In one of the refactoring steps, I extracted the code that filters out not numeric tokens to a separated helper method: filterOutNotNumericTokens:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<std::string> NumbersExtractor::filterOutNotNumericTokens(const std::vector<std::string> & tokens) const { | |
std::vector<std::string> numericTokens; | |
for (unsigned int i = 0; i < tokens.size(); ++i) { | |
std::string token = tokens[i]; | |
if (isNotAnInteger(token)) { | |
continue; | |
} | |
numericTokens.push_back(tokens[i]); | |
return numericTokens; | |
} | |
bool NumbersExtractor::isNotAnInteger(const std::string & str) const { | |
return ! StringUtils::isAnInteger(str); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<std::string> NumbersExtractor::filterOutNotNumericTokens(const std::vector<std::string> & tokens) const { | |
std::vector<std::string> numericTokens; | |
std::copy_if(tokens.begin(), tokens.end(), | |
std::back_inserter(numericTokens), | |
StringUtils::isAnInteger); | |
return numericTokens; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<int> NumbersFilter::ignoreTooBig(const std::vector<int> & numbers) const { | |
std::vector<int> filteredNumbers; | |
for (unsigned int i = 0; i < numbers.size(); ++i) { | |
if (notTooBig(numbers[i])) { | |
filteredNumbers.push_back(numbers[i]); | |
} | |
} | |
return filteredNumbers; | |
} | |
bool NumbersFilter::notTooBig(int number) const { | |
return !(number > 1000); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
bool notTooBig(int number); | |
std::vector<int> NumbersFilter::ignoreTooBig(const std::vector<int> & numbers) const { | |
std::vector<int> filteredNumbers; | |
std::copy_if(numbers.begin(), numbers.end(), | |
std::back_inserter(filteredNumbers), | |
notTooBig); | |
return filteredNumbers; | |
} | |
bool notTooBig(int number) { | |
return !(number > 1000); | |
} |
Notice that we had to make notTooBig a free function in order to pass it to copy_if.
To keep it as a member function we'd need to write this less readable version:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<int> NumbersFilter::ignoreTooBig(const std::vector<int> & numbers) const { | |
std::vector<int> filteredNumbers; | |
std::copy_if(numbers.begin(), numbers.end(), | |
std::back_inserter(filteredNumbers), | |
std::bind1st(std::mem_fun(&NumbersFilter::notTooBig), this)); | |
return filteredNumbers; | |
} | |
bool NumbersFilter::notTooBig(int number) const { | |
return !(number > 1000); | |
} |
In this case, I think it's probably better to use a lambda:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<int> NumbersFilter::ignoreTooBig(const std::vector<int> & numbers) const { | |
std::vector<int> filteredNumbers; | |
std::copy_if(numbers.begin(), numbers.end(), | |
std::back_inserter(filteredNumbers), | |
[](const int number) { return !(number > 1000); }); | |
return filteredNumbers; | |
} |
We can improve the readability of this code introducing a helper templated functions, filter:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
namespace VectorUtils { | |
template<typename T, typename Function> | |
std::vector<T> filter(const std::vector<T> & original, Function pred) { | |
std::vector<T> filtered; | |
std::copy_if(original.begin(), original.end(), | |
std::back_inserter(filtered), | |
pred); | |
return filtered; | |
} | |
template<typename T2, typename T1, typename Function> | |
std::vector<T2> map(const std::vector<T1> & original, Function f) { | |
std::vector<T2> mapped; | |
std::transform(original.begin(), original.end(), | |
std::back_inserter(mapped), | |
f); | |
return mapped; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<std::string> NumbersExtractor::filterOutNotNumericTokens( | |
const std::vector<std::string> & tokens) const { | |
return VectorUtils::filter(tokens, StringUtils::isAnInteger); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<int> NumbersFilter::ignoreTooBig(const std::vector<int> & numbers) const { | |
return VectorUtils::filter(numbers, | |
[](int number) { return !(number > 1000); }); | |
} |
We can also use filter to refactor the getNegatives method:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<int> NumbersValidator::getNegatives(const std::vector<int> & numbers) const { | |
return VectorUtils::filter(numbers, [](int number) { return number < 0; }); | |
} |
The same procedure can be applied to have a map templated function that uses the transform algorithm which can then be used to refactor the convertToInts method:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::vector<int> NumbersExtractor::convertToInts( | |
const std::vector<std::string> & numbersStrings) const { | |
return VectorUtils::map<int, std::string>(numbersStrings, StringUtils::convertToInt); | |
} |
This map and filter functions that we'd refactored out from the StringCalculator code are pure functions that are highly reusable and will help us to avoid duplication in another projects.
No comments:
Post a Comment