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

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

新規登録して質問してみよう
ただいま回答率
85.35%
for

for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

Qt

QtはGUIプログラムの開発で広く使われているクロスプラットフォーム開発のフレームワークです。

C++

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

Qt Creator

Qt Creatorは、Qtアプリケーションを開発するための開発統合環境(IDE)です。

Q&A

解決済

2回答

4843閲覧

Qtのsplitについてです

wave_vague

総合スコア21

for

for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

Qt

QtはGUIプログラムの開発で広く使われているクロスプラットフォーム開発のフレームワークです。

C++

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

Qt Creator

Qt Creatorは、Qtアプリケーションを開発するための開発統合環境(IDE)です。

0グッド

0クリップ

投稿2020/06/25 10:33

sample.txt
の中身が
apple=150
orange=300
grape=200
のときにsample.txtを読み込んで(全読み込み)"\n"で1行ずつsplitした後に、"="でsplitしてパラメータ名が正しければcorrectを表示したいのですが、下記コードだと、grapeがある場合でもgrapeがない場合でもelse if(key != "grape")に引っかかってしまいます。
grapeがある場合、else if(key != "grape")に引っかからないようにするにはどのように修正すべきなのでしょうか。
grapeがない場合やgrapeの綴りが間違っている(例glape)時だけelse if(key != "grape")の処理に入りたいです。

#include "MainWindow.h" #include <QApplication> #include <QDebug> #include <QFile> #include <QTextCodec> int main(int argc, char *argv[]) { //QApplication a(argc, argv); //MainWindow w; //w.show(); QFile file("C:\Users\Desktop\sample.txt"); if (!file.open(QIODevice::ReadOnly)) { qDebug() << "can not open file." ; return 0; } QString str; QTextStream in(&file); str = in.readAll(); qDebug() << str ; QStringList list1 = str.split("\n"); for (int i=0; i < list1.count(); i++) { QString txt = list1[i]; QStringList list2 = txt.split("="); QString key = list2[0]; QString value = list2[1]; if(key == "apple"){ qDebug() << "correct"; } if(key == "grape"){ qDebug() << "correct"; } else if (key != "grape"){ qDebug() << "incorrect"; //break; } if(key == "orange"){ qDebug() << "correct"; } } file.close(); //return a.exec(); }

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

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

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

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

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

yuki23

2020/06/25 11:16

「grapeがある場合」とはどういう意味ですか? 手元で実行してみましたが、ちゃんとkeyがgrapeの行では else if (key != "grape"){ の分岐は通りませんでしたが?
wave_vague

2020/06/25 11:24

私が書いたコードだと、keyにappleが入っている際にelse if (key != "grape")の分岐を通ってしまいます。 else if (key != "grape")はsample.txtにgrapeがない場合のみ通ってほしいです。
guest

回答2

0

元々の質問の回答は @tyu_ru_cpp さんが示されているように QSet(や QList や QHash など)でキーの集合を管理して、それに対して確認をするということになると思います。

単に キー=値 形式のファイルを扱いたいのであれば、.ini フォーマットに対応している QSettings を使う方法もあります。

Qt

1#include <QDebug> 2#include <QSettings> 3 4int main(int argc, char **argv) 5{ 6 QSettings settings("./sample.txt", QSettings::IniFormat); 7 8 if (settings.contains("apple")) 9 qDebug() << "correct"; 10 11 if (settings.contains("grape")) 12 qDebug() << "correct"; 13 else 14 qWarning() << "incorrect"; 15 16 if (settings.contains("orange")) 17 qDebug() << "correct"; 18 return 0; 19} 20

投稿2020/06/25 11:30

tasuku.

総合スコア347

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

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

0

ベストアンサー

c++

1if (key == "apple"){ // ① 2 qDebug() << "correct"; 3} 4 5 6if(key == "grape"){ // ② 7 qDebug() << "correct"; 8} 9else if (key != "grape") { // ③ 10 qDebug() << "incorrect"; 11 //break; 12}

keyが"apple"の時に、①のif文を抜けた後③のif文に入ってしまいます。

このようなときには正しい単語の集合を持っておき、keyがそれに含まれるかを調べるのが良いと思います

c++

1QSet<QString> fruit = {"apple", "grape", "orange"}; 2 3if (fruit.contains(key)) { 4 qDebug() << "correct"; 5} else { 6 qDebug() << "incorrect"; 7} 8

余談ですが、QString("apple=150\norange=300\ngrape=200\n").split("\n");の結果はQList("apple=150","orange=300","grape=200","")の4つになります。

このとき、質問文のコードのように

c++

1QStringList list2 = txt.split("="); 2QString key = list2[0]; 3QString value = list2[1];

としてしまうと最後の空文字列に対してsplit("=")した結果の'[1]'を参照しようとするので範囲外参照になってしまいます。

コレを防ぐために、split後のlistのサイズはしっかり調べましょう。また、この問題の直接の対策にはなりませんが、.split("=", QString::SkipEmptyParts)とすると空文字列を飛ばしてくれるので覚えておくと良いと思います

投稿2020/06/25 10:57

編集2020/06/25 11:00
tyu_ru_cpp

総合スコア40

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

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

wave_vague

2020/06/25 11:20

回答ありがとうございます。 あらかじめ、keyがあるかどうかの確認をしたい場合は正しい単語の集合を持てばよろしいのですね。大変参考になりました。 ちなみに、keyがある場合はそのkeyの値をメンバ変数に代入したいのですが、下記方法であっていますでしょうか? if (key == "apple"){ m_apple = value.toInt();//appleの値である150が代入される } if (key == "grape"){ m_grape = value.toInt();//grapeの値である200が代入される }
wave_vague

2020/06/25 11:26

この場合だとappleの分岐を通った後 else if (key != "grape"){ qDebug() << "incorrect"; } に引っかかってしまいますよね?
tyu_ru_cpp

2020/06/25 11:31

そのような場合は、 if (key == "apple"){ m_apple = value.toInt();//appleの値である150が代入される } else if (key == "grape"){ m_grape = value.toInt();//grapeの値である200が代入される } else if (...) { ... } else { qDebug() << "incorrect"; } といったふうにif-elseを書くのがよいと思います。
tyu_ru_cpp

2020/06/25 11:35

また、連想配列を用いて QMap<QString, int> fruit; fruit["apple"] = 0; // デフォルト値 fruit["orange"] = 0; // デフォルト値 fruit["grape"] = 0; // デフォルト値 if (fruit.contains(key)) { fruit[key] = value.to_int(); } else { qDebug() << "incorrect"; } みたいに記述すると拡張性を高くすることができます
tyu_ru_cpp

2020/06/25 11:41 編集

あ、tasukuさんの回答内容と被ってしまった QSettingsはQMapのキーをQStringにvalueをJSONの値と似たようなものに特殊化して、設定ファイルの入出力にも対応させたような便利なやつです
wave_vague

2020/06/25 11:43

ありがとうございます。if-elseを使う場合は下記コードのようになるという認識であっていますでしょうか。sample.txtに記載されているkeyが正しければvalueをメンバ変数に代入、keyが正しくないおよびkeyがない場合はincorrectを出力する。 for (int i=0; i < list1.count(); i++) { QString txt = list1[i]; QStringList list2 = txt.split("="); QString key = list2[0]; QString value = list2[1]; if (key == "apple"){ m_apple = value.toInt();//appleの値である150が代入される } else if (key == "grape"){ m_grape = value.toInt();//grapeの値である200が代入される } else if (key == "orange") { m_orange = value.toInt();//orangeの値である300が代入される } else { qDebug() << "incorrect"; } }
tyu_ru_cpp

2020/06/25 11:47

(valueの中身が整数になっていないようなケースを考慮しなければ)おおよそそんなコードになると思います valueが正しくパースできたかは.toInt()の第一引数にチェック用のbool変数へのポインタを渡せばできます 詳しくは公式リファレンスをどうぞ https://doc.qt.io/qt-5/qstring.html#toInt
wave_vague

2020/06/25 11:53

ありがとうございます。大変助かりました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問