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

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

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

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

Q&A

解決済

2回答

460閲覧

競技プログラミング(AtCoder)c++に関する疑問

houki

総合スコア22

C++

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

0グッド

0クリップ

投稿2019/08/16 11:35

前提・実現したいこと

AtCoder問題「C - 数式の書き換え」URL:https://atcoder.jp/contests/abc033/tasks/abc033_c

コードの考え方:数字部分と記号部分に分けて探索する方針になっています。

以下の該当のコードの(*)でstring tと書いた場合、sの文字列の長さ(=n)が一定値を越えると、1つ目のfor文がi<nまで処理されず(nが大きい値の時、最大でもi=47までしか処理が行われず)に、int ans=0;以下の処理が始まります。
char t[100010]にすると、上記のようなことは起こらないのですが。

なぜstringにした場合、上記のようなことが起こるのでしょうか。
また、競技プログラミングに限らず、stringよりもcharを使う方が良いですか?

よろしくお願いします。

該当のソースコード

c++

1#include <bits/stdc++.h> 2 3using namespace std; 4typedef long long ll; 5#define rep(i, j, n) for (int i = j; i < n; i++) 6 7int main() { 8 9 string s;cin>>s; 10 int n=s.size(); 11 int num[100010]; 12 string t;//------------------------------(*) 13 14 for(int i=0;i<n;i++){ 15 if(i%2==0)num[i/2]=s[i]-'0'; 16 else t[i/2]=s[i]; 17 } 18 19 int ans=0; 20 bool zero=false; 21 rep(i,0,(n-1)/2){ 22 if(num[i]==0)zero=true; 23 if(t[i]=='+'){ 24 if(!zero)ans++; 25 else zero=false; 26 } 27 } 28 if(num[(n-1)/2]!=0&&!zero)ans++; 29 30 cout<<ans<<endl; 31 32 return 0; 33} 34

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

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

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

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

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

guest

回答2

0

string t;としたとき、tのサイズは0です。問題は、t[i/2]という所はサイズ0より大きい位置の参照となり、そこに代入しようとしていることです。その場合はどうなるのかというと、

  • C++03以下: 未定義動作です。
  • C++11以上: char()(ヌル文字)の値を持った参照を返しますが、代入した場合は未定義動作です。

となり、いずれの場合も未定義動作になります。未定義動作なのでその後の動作がどうなるかは予測不可能であり、コンパイラ・環境・実行タイミング等で結果も変わってくるでしょう。(今回は、質問者さんの環境では、たまたまたi=47まで行われて次の処理に進んだように見えただけで、場合によってはエラーで落ちる等もあり得ると言うことです。)

string t(n/2, '\0');等のようにサイズを指定して初期化するか、動的に増やしながら代入してください。

参考
std::basic_string<CharT,Traits,Allocator>::operator[] - cppreference.com
basic_string::operator[] - cpprefjp C++日本語リファレンス

投稿2019/08/16 12:42

編集2019/08/16 12:44
raccy

総合スコア21735

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

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

houki

2019/08/16 13:53

stringはvectorのように自動で動的に動くものと誤認識していました。自ら動的に書く必要があることが分かりました。サイズの未定義が起こすパソコン内の動作についても回答していただき大変勉強になりました。本当にありがとうございます。 回答者のお二人方ともに分かりやすく回答していただき、投稿時間が早かったset0gut1さんの回答をベストアンサーにしたいと思います。
guest

0

ベストアンサー

1つ目の for ループ中で n の値を出力すると、あるところから n の値が変わってしまうのが観測できると思います。
string t で確保した領域を超えて t[i/2] = s[i] と値を代入していて、ここでメモリ上の周囲の変数の値を破壊してます。

次のように変更するとACしました。

diff

1 for(int i=0;i<n;i++){ 2 if(i%2==0)num[i/2]=s[i]-'0'; 3- else t[i/2]=s[i]; 4+ else t += s[i]; 5 }

投稿2019/08/16 12:32

set0gut1

総合スコア2413

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

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

houki

2019/08/16 13:53

今回の問題は、変数の領域サイズの未定義が起こすエラーということが理解できました。確かにnが途中で変化していました。また、charではなく、stringのままで動的に追加していくやり方も提示して下さり、助かりました。ありがとうございます。 回答者のお二人方ともに分かりやすく回答していただき、投稿時間が早かったset0gut1さんの回答をベストアンサーにしたいと思います。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問