質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

2回答

1942閲覧

c++のmultimapで特定のキーのi番目の要素を取得する方法を探しています。

tada_tadaa

総合スコア111

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2017/08/29 02:55

c++のmultimapで特定のキーのi番目の要素を取得する方法を探しています。
例えば以下のようなプログラム文があります。

c++

1 2 3#include <stdio.h> 4#include <algorithm> 5#include <iostream> 6#include <string> 7#include <vector> 8#include <functional> 9#include <map> 10 11using namespace std; 12 13int main() { 14int i; 15 16multimap<string, int> mpp; 17 18for (i = 0; i < 10; i++) { 19if (i % 2 == 0) { 20mpp.insert(pair<string, int>("A", i)); 21} 22else { 23mpp.insert(pair<string, int>("B", i)); 24} 25} 26 27std::multimap<std::string, int>::iterator p; 28 29for (p = mpp.begin(); p != mpp.end(); p++) { 30cout << p->first << " " << p->second << endl; 31} 32 33getchar(); 34getchar(); 35return 0; 36}

実行結果は
A 0
A 2
A 4
A 6
A 8
B 1
B 3
B 5
B 7
B 9

となりますが、キー「B」の要素「5」だけを表示する方法はありますでしょうか。
この場合、「5」を検索したりするのではなく、キー「B」の2番目の要素を指し示す
方法を知りたいです。
よろしくお願いいたします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

ベストアンサー

こんにちは。

ちょっと面倒ですが、std::multimap::lower_boundで指定キーの「先頭」イテレータを得て、必要回数インクリメント(①)すれば基本はOKです。
注意点として、std::multimap::lower_boundは指定キーがない場合でも指定キーを挿入するべき位置を返却するので、①が指定キーを持つのか確認する必要があります。

std::multimapはstd::unordered_multimapではないので順序付けはなされるようですが、std::pair::secondも使ってソートが必要ならばstd::multisetを使った方が良いと思います。std::tieを使えば比較関数を極簡単に書けます

投稿2017/08/29 04:06

Chironian

総合スコア23272

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

mattn

2017/08/29 04:54

multimap は順序付けされるのはキーだけという認識だったのですが。
Chironian

2017/08/29 05:06

その通りですね。
mattn

2017/08/29 05:31 編集

キーや値での順保障はないですが、挿入順は保証されるようでした。この意味からするとキー内では挿入順になるので Chironian さんので問題なさそうでした。(C++11から)すみません。 http://en.cppreference.com/w/cpp/container/multimap
tada_tadaa

2017/08/29 05:50

回答ありがとうございます。 Chironian様のおっしゃるようにプログラムを作ってみました。 間違いもあるかもしれませんが、一応動いていると思います。 #include <stdio.h> #include <algorithm> #include <iostream> #include <string> #include <vector> #include <functional> #include <map> using namespace std; int main() { int i; multimap<string, int> mpp; for (i = 0; i < 10; i++) { if (i % 2 == 0) { mpp.insert(pair<string, int>("A", i)); } else { mpp.insert(pair<string, int>("B", i)); } } multimap<std::string, int>::iterator p; for (p = mpp.begin(); p != mpp.end(); p++) { cout << p->first << " " << p->second << endl; } // 「B」キーがあるかどうか調べる int cnt_a = mpp.count("B"); if (cnt_a == 0) { cout << "キーが存在しません" << endl; } else if( cnt_a > 0 ){ cout << "キーは存在します" << endl; cout << cnt_a << endl; p = mpp.lower_bound("B"); p++; cout << "p->" << p->second << endl; p++; cout << "p->" << p->second << endl; } getchar(); getchar(); return 0; } 実行結果は A 0 A 2 A 4 A 6 A 8 B 1 B 3 B 5 B 7 B 9 キーは存在します 5 p->3 p->5 になり、うまくキー「B」の2番目の要素(ゼロから数えて)を得ることができました。 それで、ついでに上記プログラムのmultiset版も作ってみようかと思ったのですが、よく分からないで います。 multimapの文字列をmultisetにしただけなのですが、エラーがでてコンパイルできません。 int i; //multimap<string, int> mpp; multiset <string, int> mpp; for (i = 0; i < 10; i++) { if (i % 2 == 0) { mpp.insert(pair<string, int>("A", i)); } else { mpp.insert(pair<string, int>("B", i)); } } そもそもmultisetは"文字をキーにして、"要素を格納できるものなのでしょうか? 度々質問してすみません。
Chironian

2017/08/29 07:29

std::multisetは、要素全てをキーにして格納します。標準の型でない場合、コンパイラは大小比較方法が分からないので、大小比較関数を定義して渡してあげる必要があります。http://vivi.dyndns.org/tech/cpp/multiset.html この比較関数って意外に面倒です。firstキー不一致の時はfirstキーで大小比較し、firstキーが一致しているはsecondキーで比較するため、3分岐します。 std::tieはその辺よしなにしてくれるようです。http://d.hatena.ne.jp/yohhoy/20120218/p1
guest

0

multimap は ordered map ではないので i 番目は分かりません。

投稿2017/08/29 03:25

編集2017/08/29 03:29
mattn

総合スコア5030

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

tada_tadaa

2017/08/29 06:02

回答ありがとうございます。 Chironian様がうまい方法を教えてくださいました。 ありがとうございました。
mattn

2017/08/29 06:09 編集

うまい方法といいますか、僕の知識が C++11 より前で止まっていたので multimap は挿入順では無いと思っていました。もちろん C++11 未満では期待しない動作になるかもしれないのでご使用の際はお気をつけて。
tada_tadaa

2017/08/29 06:11

>C++11 未満では期待しない動作になるかもしれないのでご使用の際はお気をつけて。 了解しました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問