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

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

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

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

Q&A

解決済

2回答

295閲覧

for文で配列をつくると、Segmentation faultとなる

Yuto_0125

総合スコア2

C++

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

0グッド

0クリップ

投稿2024/04/09 13:20

実現したいこと

入力した数だけの要素数を持つ配列を作り、各要素を入力して格納し、それらを表示したい。

発生している問題・分からないこと

 要素数numにどんな数値を打ち込んでも、その後配列の要素を4つ受け付けた後Segmentation faultと出てしまう。
ただ、numを1にしたときのみ、cin >> array[i]を1回だけ受け付けた後
0番目は:〇〇
と成功する。

エラーメッセージ

error

1Running /home/ec2-user/environment/sample2.cpp 2要素数は?4 312 434 556 678 7bash: line 17: 33343 Segmentation fault (core dumped) "$file.o" 8 9

該当のソースコード

C++

1#include <stdio.h> 2#include <iostream> 3#include <iomanip> 4using namespace std; 5 6int main() 7{ 8 int num; 9 int array[num]; 10 int i; 11 cout << "要素数は?"; 12 cin >> num; 13 for(i = 0; i < num; i++) 14 { 15 cin >> array[i]; 16 } 17 18 for(i = 0; i < num; i++) 19 { 20 cout << i << "番目は:" << array[i] << "\n"; 21 } 22 return 0; 23} 24

試したこと・調べたこと

  • teratailやGoogle等で検索した
  • ソースコードを自分なりに変更した
  • 知人に聞いた
  • その他
上記の詳細・結果

元々、C++初心者向けの動画を真似て作ったので、ソースコードは同じはず。
2つ目のfor文をコメントアウトし、array[0]だけ出力などを試したが結果は変わらない。

補足

特になし

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

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

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

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

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

guest

回答2

0

他の質問のコメントとして書き始めたのですが結構文字数が多くなったので回答欄に独立させました。なので、若干他の回答とそのコメントのやりとりに依存している記述になっています。

「numが不定だから」先にnumに値を与えましょう、というのならちょっと正確な言い方ではないです。

重要な要素として、C/C++のプログラムは、少なくとも一つの関数内で繰り返しやジャンプなどの構文を伴わない場合は「上から下に順次実行」し、勝手に過去に遡ってなにかが適用されることはありません。
int array[num];
と記述されていたら、「その時点でのnumの値でこの宣言の処理が為される」ことになり、その後でnumの値が何に変わろうとも一旦確定してしまった構造体arrayにはなんらの影響もありません。これが「先にnumを確定しておかなければならない」理由とも言えます。

それと、「不定」という言葉をちゃんとわかっているのかなぁ、というのがちょっと心配。
C/C++では、初期化していないローカル変数(とりあえず関数の中で宣言した変数を思ってください)の値は「不定」です。
不定というのは、言語規格としてはプログラムを実行してみるまで値が決まらないし、何の値になるかについて事前になんらの期待をしてもいけない、ということ。プログラムが
int main()
{
int num;
まで進んだところでnumという変数が生成されます。C/C++の変数には「何も入っていない」という状態はないので何らかの値を持つのですが、その時点でのnumに入っているデータはもしかしたら0かも、1かも、あるいは-10億かも20億かも知れない、そういう状態である、いうことです。
もちろん、実際の機械でプログラムを実行する場合には、諸条件によって実行毎にある決まった値に安定して「なってしまう」ことはあります。でもそれは、もしかしたら再起動したら別の値になるかも知れない、OSにパッチが当たったら違う値になるかもしれない、そういう危うい安定なのかもしれません。
(余談:「何も入っていない」について nullptrは? と思う人がいるかもしれまぜんが、それはポインタがどこも指していないということを示す特別な値のことなので、決して変数自体に値が何も入っていないということではありません)

もし事前には想定できないnumがたまたま1であったなら、
int array[num];
は1個だけの領域が確保され、以降の要素へのアクセスは「未定義の動作」(何が起こっても文句は言えない)を引き起こします。今回のsegmentatil faultはその現れであった、ということでしょう。

投稿2024/04/09 23:53

thkana

総合スコア7652

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

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

0

ベストアンサー

Windows 11、Visual Studio 2022で確認しました。

numが不定な状態で、int array[num];は宣言できません。

#include <stdio.h> #include <iostream> #include <iomanip> // 1行追加 #include <vector> using namespace std; int main() { int num; // 1行変更 vector<int> array; int i; cout << "要素数は?"; cin >> num; // 1行追加 array.resize(num); for(i = 0; i < num; i++) { cin >> array[i]; } for(i = 0; i < num; i++) { cout << i << "番目は:" << array[i] << "\n"; } return 0; }

投稿2024/04/09 14:48

hiroki-o

総合スコア483

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

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

Yuto_0125

2024/04/09 14:57

回答ありがとうございます! ちなみにですが、「numが不定なのが原因」という回答を見て、元々のコードの int array[num] を cin >> num の後に持ってきても解決しました。
hiroki-o

2024/04/09 15:02

処理系によっては、それでもビルドが通るのでしょうか? 少なくともVisual Studio 2022では通りませんし、C/C++的に間違いだと思います。 やらないほうがいいと思います。
actorbug

2024/04/09 21:19

配列のサイズに変数を入れる機能は、C言語のVLA(可変長配列)という機能で、C++でも使えるようにしている処理系が多いです。主な処理系であるg++やclang++だと使用可能で、使えないのは少数派だったりします。一応C言語の機能なので、「C/C++的に間違いだと思います」は少し言い過ぎです。まあ、C++の機能ではないので、「やらないほうがいいと思います」には同意です。
hiroki-o

2024/04/09 21:29

actorbugさん 勉強になりました。ありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.47%

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

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

質問する

関連した質問