[解決済み] std::accumulateを理解する
質問
なぜか知りたい
std::accumulate
(aka reduce) 3番目のパラメータが必要です。をご存じない方のために
accumulate
は、このように使います。
vector<int> V{1,2,3};
int sum = accumulate(V.begin(), V.end(), 0);
// sum == 6
への呼び出し
accumulate
とは同等である。
sum = 0; // 0 - value of 3rd param
for (auto x : V) sum += x;
また、オプションで第4パラメータがあり、加算を他の操作に置き換えることができます。
その理由は、例えばベクトルの要素を加算するのではなく、乗算する必要がある場合、他の(ゼロでない)初期値が必要だからだと聞いたことがあります。
vector<int> V{1,2,3};
int product = accumulate(V.begin(), V.end(), 1, multiplies<int>());
しかし、Pythonのように、初期値を
V.begin()
から始まる範囲を使用します。
V.begin()+1
. このような感じです。
int sum = accumulate(V.begin()+1, V.end(), V.begin());
これは、どのようなオペでも動作します。なぜ第3パラメータが必要なのでしょうか?
解決方法は?
このままでは、ある範囲が空でないことを確実に知っていて、その範囲の最初の要素から累積を開始したいコードにとって迷惑な話です。積算に使用する演算によっては、使用すべき「ゼロ」値が必ずしも明らかではありません。
一方、空でない範囲を必要とするバージョンしか提供しない場合、呼び出し側にとって、自分の範囲が空でないことを確実に知らないのは迷惑な話です。追加の負担がかかるのです。
一つの視点として、両方の機能を提供することがベストであることは言うまでもありません。例として、Haskellは
foldl1
と
foldr1
(空でないリストを必要とする) と共に
foldl
と
foldr
(これは
std::transform
).
もうひとつの視点は、一方は些細な変換で他方から実装できるので(あなたが実証してくれたように。
std::transform(std::next(b), e, *b, f)
--
std::next
はC++11ですが、要点は同じです)、表現力を失うことなく、できる限りインターフェースを小さくすることが望ましいのです。
関連
-
[解決済み】C++ 非推奨の文字列定数から「char*」への変換について
-
[解決済み】非静的メンバ関数への参照を呼び出す必要がある
-
[解決済み】C++ - 解放されるポインタが割り当てられていないエラー
-
[解決済み】「std::operator」で「operator<<」にマッチするものがない。
-
[解決済み】C++ - 適切なデフォルトコンストラクタがない [重複]。
-
[解決済み] 配列のベクトルを扱う正しい方法
-
[解決済み] 変数サイズのオブジェクトが初期化されないことがある c++
-
[解決済み] using namespace std;」はなぜバッドプラクティスだと言われるのですか?
-
[解決済み] std::vectorをハードコードされた要素で初期化する最も簡単な方法は何ですか?
-
[解決済み] std::vector<> からインデックスで要素を消すにはどうしたらいいですか?
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
[解決済み] テスト
-
[解決済み】coutはstdのメンバではない
-
[解決済み】C++ クラスヘッダが含まれているときに「不明な型」があるのはなぜですか?重複
-
[解決済み] エラーが発生する。ISO C++は型を持たない宣言を禁じています。
-
[解決済み】C++プログラムでのコンソールの一時停止
-
[解決済み】エラー。switchステートメントでcaseラベルにジャンプする
-
[解決済み】Visual Studio 2013および2015でC++コンパイラーエラーC2280「削除された関数を参照しようとした」が発生する
-
[解決済み】標準ライブラリにstd::endlに相当するタブはあるか?
-
[解決済み】Enterキーを押して続行する
-
[解決済み】std::cin.getline( ) vs. std::cin