次のベクトルが与えられた場合:
std::vector<int> foo{1,2,3,4,5,6,7};
std::vector<int> bar{10,20,30,40,50,60,70};
最後に、foo
に値{ 10, 2, 30, 4, 50, 6, 70 }
を含めて、すべての奇数の値を置き換えることを意味します。
アルゴリズムstd :: replace_ifを試しましたが、バーに対応する値にアクセスするにはどうすればよいですか?
// replace_copy_if example
#include <iostream> // std::cout
#include <algorithm> // std::replace_copy_if
#include <vector> // std::vector
bool IsOdd (int i) { return ((i%2)==1); }
int main () {
std::vector<int> foo{1,2,3,4,5,6,7};
std::vector<int> bar{10,20,30,40,50,60,70};
std::replace_if (foo.begin(), foo.end(), IsOdd, 0);
std::cout << "foo contains:";
for (auto i: foo){ std::cout << ' ' << i; }
std::cout << '\n';
return 0;
}
// output : foo contains: 0 2 0 4 0 6 0
// desired output : foo contains: 10 2 30 4 50 6 70
std::replace_if
は、最後の引数として指定された定数値を持つ要素の置換のみをサポートします。ただし、自分にとって意味があると感じた場合は、独自のreplace_if
関数は 参照実装 に基づいており、ある種のジェネレータを最後のパラメータとして受け入れます。例えば:
template<class ForwardIt, class UnaryPredicate, class Generator>
void my_replace_if(ForwardIt first, ForwardIt last,
UnaryPredicate p, Generator g)
{
for (; first != last; ++first) {
if(p(*first)) {
*first = g();
}
}
}
bool IsOdd (int i) { return ((i%2)==1); }
int main(int argc, const char * argv[]) {
std::vector<int> foo{1,2,3,4,5,6,7};
std::vector<int> bar{10,20,30,40,50,60,70};
int i = 0;
my_replace_if (foo.begin(), foo.end(), IsOdd, [&bar, &i]()
{
return bar[i++];
});
for (auto el : foo)
{
std::cout << el << std::endl;
}
return 0;
}
=====編集=====
Anastaciuが指摘したように、ジェネレーターが呼び出されたときだけでなく、ジェネレーターのインデックスは各ステップでインクリメントする必要があります。
例えば。:
my_replace_if (foo.begin(), foo.end(),
[&i](int v) { i++; return ((v%2)==1); },
[&bar, &i]() { return bar[i-1]; });
しかし、正直に言うと、この場合はやり過ぎであり、見栄えもよくありません。
std::vector<int> result;
std::transform(
foo.begin(), foo.end(), bar.begin(),
std::back_inserter(result),
[](const auto& a, const auto& b) {
if (a % 2 == 0) {
return a;
}
return b;
}
);
// replace_copy_if example
#include <iostream> // std::cout
#include <vector> // std::vector
int main () {
std::vector<int> foo{1,2,3,4,5,6,7};
std::vector<int> bar{10,20,30,40,50,60,70};
for (size_t i = 0; i < bar.size(); i++)
{
if (foo[i]%2==1) foo[i]=bar[i];
}
std::cout << "bar contains:";
for (auto i: foo){ std::cout << ' ' << i; }
std::cout << '\n';
return 0;
}