[fix/docs]: cleanup longest_common_string.cpp (#2462)

* updating DIRECTORY.md

* fix: cleanup longest_common_string.cpp

* clang-format and clang-tidy fixes for 3280d46e

* docs: explain why utility header is needed

* refactor: add test_all()

* docs: add relevant doc-strs

* refactor: rename to tests()

* style: typo fix

* style: use string instead of str

* Apply suggestions from code review

The code after this commit requires reformatting.

Co-authored-by: David Leal <halfpacho@gmail.com>

* clang-format and clang-tidy fixes for 18203b20

* style: give an exact reason why iostream is needed

* style: make all test functions `static`

* docs: update missing docs

* chore: apply suggestions from code review

---------

Co-authored-by: github-actions[bot] <github-actions@users.noreply.github.com>
Co-authored-by: realstealthninja <68815218+realstealthninja@users.noreply.github.com>
Co-authored-by: David Leal <halfpacho@gmail.com>
This commit is contained in:
Piotr Idzik 2023-07-04 07:40:35 +02:00 committed by GitHub
parent e3f0551f98
commit 32db99c1de
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -1,53 +1,159 @@
#include <iosrteam> /**
using namespace std; * @file
* @brief contains the definition of the function ::longest_common_string_length
* @details
* the function ::longest_common_string_length computes the length
* of the longest common string which can be created of two input strings
* by removing characters from them
*
* @author [Nikhil Arora](https://github.com/nikhilarora068)
* @author [Piotr Idzik](https://github.com/vil02)
*/
int max(int a, int b) { return (a > b) ? a : b; } #include <cassert> /// for assert
#include <iostream> /// for std::cout
#include <string> /// for std::string
#include <utility> /// for std::move
#include <vector> /// for std::vector
int main() { /**
char str1[] = "DEFBCD"; * @brief computes the length of the longest common string created from input
char str2[] = "ABDEFJ"; * strings
int i, j, k; * @details has O(str_a.size()*str_b.size()) time and memory complexity
int n = strlen(str1) + 1; * @param string_a first input string
int m = strlen(str2) + 1; * @param string_b second input string
// cout<<n<<" "<<m<<"\n"; * @returns the length of the longest common string which can be strated from
int a[m][n]; * str_a and str_b
*/
std::size_t longest_common_string_length(const std::string& string_a,
const std::string& string_b) {
const auto size_a = string_a.size();
const auto size_b = string_b.size();
std::vector<std::vector<std::size_t>> sub_sols(
size_a + 1, std::vector<std::size_t>(size_b + 1, 0));
for (i = 0; i < m; i++) { const auto limit = static_cast<std::size_t>(-1);
for (j = 0; j < n; j++) { for (std::size_t pos_a = size_a - 1; pos_a != limit; --pos_a) {
if (i == 0 || j == 0) for (std::size_t pos_b = size_b - 1; pos_b != limit; --pos_b) {
a[i][j] = 0; if (string_a[pos_a] == string_b[pos_b]) {
sub_sols[pos_a][pos_b] = 1 + sub_sols[pos_a + 1][pos_b + 1];
else if (str1[i - 1] == str2[j - 1]) } else {
a[i][j] = a[i - 1][j - 1] + 1; sub_sols[pos_a][pos_b] = std::max(sub_sols[pos_a + 1][pos_b],
sub_sols[pos_a][pos_b + 1]);
else
a[i][j] = 0;
}
}
/*for(i=0;i<m;i++)
{
for(j=0;j<n;j++)
cout<<a[i][j]<<" ";
cout<<"\n";
}*/
int ma = -1;
int indi, indj;
for (i = 0; i < m; i++) {
for (j = 0; j < n; j++) {
if (a[i][j] > ma) {
ma = a[i][j];
indi = i;
indj = j;
} }
} }
} }
cout << str1 << "\n"; return sub_sols[0][0];
cout << str2 << "\n"; }
cout << "longest string size = " << ma /*<<" "<<indi<<" "<<indj*/ << "\n"; /**
for (i = indi - 3; i < indi; i++) cout << str1[i]; * @brief represents single example inputs and expected output of the function
cout << "\n"; * ::longest_common_string_length
*/
struct TestCase {
const std::string string_a;
const std::string string_b;
const std::size_t common_string_len;
TestCase(std::string string_a, std::string string_b,
const std::size_t in_common_string_len)
: string_a(std::move(string_a)),
string_b(std::move(string_b)),
common_string_len(in_common_string_len) {}
};
/**
* @return example data used in the tests of ::longest_common_string_length
*/
std::vector<TestCase> get_test_cases() {
return {TestCase("", "", 0),
TestCase("ab", "ab", 2),
TestCase("ab", "ba", 1),
TestCase("", "xyz", 0),
TestCase("abcde", "ace", 3),
TestCase("BADANA", "ANADA", 3),
TestCase("BADANA", "CANADAS", 3),
TestCase("a1a234a5aaaa6", "A1AAAA234AAA56AAAAA", 6),
TestCase("123x", "123", 3),
TestCase("12x3x", "123", 3),
TestCase("1x2x3x", "123", 3),
TestCase("x1x2x3x", "123", 3),
TestCase("x12x3x", "123", 3)};
}
/**
* @brief checks the function ::longest_common_string_length agains example data
* @param test_cases list of test cases
* @tparam type representing a list of test cases
*/
template <typename TestCases>
static void test_longest_common_string_length(const TestCases& test_cases) {
for (const auto& cur_tc : test_cases) {
assert(longest_common_string_length(cur_tc.string_a, cur_tc.string_b) ==
cur_tc.common_string_len);
}
}
/**
* @brief checks if the function ::longest_common_string_length returns the same
* result when its argument are flipped
* @param test_cases list of test cases
* @tparam type representing a list of test cases
*/
template <typename TestCases>
static void test_longest_common_string_length_is_symmetric(
const TestCases& test_cases) {
for (const auto& cur_tc : test_cases) {
assert(longest_common_string_length(cur_tc.string_b, cur_tc.string_a) ==
cur_tc.common_string_len);
}
}
/**
* @brief reverses a given string
* @param in_str input string
* @return the string in which the characters appear in the reversed order as in
* in_str
*/
std::string reverse_str(const std::string& in_str) {
return {in_str.rbegin(), in_str.rend()};
}
/**
* @brief checks if the function ::longest_common_string_length returns the same
* result when its inputs are reversed
* @param test_cases list of test cases
* @tparam type representing a list of test cases
*/
template <typename TestCases>
static void test_longest_common_string_length_for_reversed_inputs(
const TestCases& test_cases) {
for (const auto& cur_tc : test_cases) {
assert(longest_common_string_length(reverse_str(cur_tc.string_a),
reverse_str(cur_tc.string_b)) ==
cur_tc.common_string_len);
}
}
/**
* @brief runs all tests for ::longest_common_string_length funcion
*/
static void tests() {
const auto test_cases = get_test_cases();
assert(test_cases.size() > 0);
test_longest_common_string_length(test_cases);
test_longest_common_string_length_is_symmetric(test_cases);
test_longest_common_string_length_for_reversed_inputs(test_cases);
std::cout << "All tests have successfully passed!\n";
}
/**
* @brief Main function
* @returns 0 on exit
*/
int main() {
tests();
return 0;
} }