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

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

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

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

Q&A

解決済

5回答

9921閲覧

C言語についての質問です。

welld

総合スコア7

C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

0グッド

0クリップ

投稿2016/05/26 01:35

最近、C言語を学習し始めた者です。
以下のソースコードを実行したところ、エラーが発生しました。
エラーの原因が知りたいです。基礎的な質問だと思いますがよろしくお願いします。

###該当のソースコード

C言語

1#include <stdio.h> 2 3void func(int *p,int i); 4 5int main(void){ 6 int *p,i; 7 func(p,i); 8 printf("%d\n",*p); 9 return 0; 10} 11 12void func(int *p,int i){ 13 i=100; 14 p=&i; 15 return; 16}

###発生している問題・エラーメッセージ

Segmentation fault (コアダンプ)

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

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

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

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

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

guest

回答5

0

ベストアンサー

C言語では、引数はすべて値渡しです(構造体引数すら、コピーして値渡しという挙動になります)。つまり、funcの中で代入した変数は、mainの中の変数とは別なので、呼び出したmainの側に影響することはありません。

そして、main内でのint *pは初期化されないままなので、それを逆参照した結果は「未定義」です。つまり、「コンパイルに失敗する」「何事もなかったかのようにどこかわからない場所を参照する」「コアダンプする」「システムごとクラッシュする」など、何が起きるともしれないプログラム、ということです。

関数から外側に値を伝える方法は3つあります。

  1. 返り値を返す
  2. 外部変数(グローバル or ファイル単位のstatic)を書き換える
  3. ポインタで受け取って、そのポインタ経由で書き込む(ポインタ自体を書き換えたいなら、int **のようにダブルポインタとする必要があります)

投稿2016/05/26 01:55

maisumakun

総合スコア145184

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

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

welld

2016/05/26 01:57

回答有り難うございます。
guest

0

エラーとしては、他の回答者の通り、main関数内のpの値が初期化されていないことが原因です。

C

1void func(int *p,int i){ 2 i=100; 3 p=&i; 4 return; 5}

について、

  1. p=&i;にて、関数func内の変数pに変数iのアドレスを代入していますが、main関数の変数pはそのアドレスを指しません。

(main関数のpと関数funcのpは別の変数であるため。)
2. 変数pはmain関数で初期化されていないので、printf("%d\n",*p);で参照した時に妥当なアドレスを見ていません。(ここでエラー)

printf("%d\n",*p);で関数funcに与えた変数iの値を表示したいならば、

  1. main関数内でpが妥当な領域を指せるようにする(p=&iをfunc内ではなくてmain関数に書くなど)
  2. 関数func内でpに値を代入する(p = &i)のではなく、pが指す領域に値が入るよう(*p = i)にする

と良いと思います。

投稿2016/05/26 01:46

編集2016/05/26 05:17
toko

総合スコア144

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

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

welld

2016/05/26 01:50

回答ありがとうございます。
maisumakun

2016/05/26 02:22 編集

(回答が修正されたので、コメントも取り下げておきます)
toko

2016/05/26 02:16 編集

ありがとうございます。早とちりしていました。修正します。(ベストアンサーは取り下げてください。)
HiroshiWatanabe

2016/05/26 02:18 編集

(コメント内容が変更されたのでそれに対する私のコメントも適当ではなくなってしまったので内容削除しておきます)
toko

2016/05/26 02:17 編集

修正しました。最初の回答は誤っていました。申し訳ありませんでした。
toko

2016/05/26 02:42

コメントは履歴が残らないのですね。HiroshiWatanabeさんの削除を促した変更点ですが、最初のコメントに"(ベストアンサーは取り下げてください。)"を追記したことです。当初は誤った回答に対する指摘とベストアンサーを訝しむコメントを頂いておりましたが、ご指摘まで削除させてしまいました。申し訳ありません。
guest

0

main の printf で参照している *p が未初期化(アドレス未定義)な状態だからだと思います。
p を func で設定しているつもりかと思われますがこれでは main の p にアドレスは設定されていません。

投稿2016/05/26 01:41

HiroshiWatanabe

総合スコア2160

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

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

welld

2016/05/26 01:49

回答ありがとうございます。
guest

0

いくつかの回答で網羅されていますが、2つの誤解が組み合わさっているように思います(実引数のアドレスを返せるという誤解+値渡しに関する誤解)。

おそらくやりたいことを実現するには以下のようにする必要があります。

#include <stdio.h> void func(int **p, int *i); int main(void){ int *p,i; func(&p, &i); printf("%d\n", *p); return 0; } void func(int **p, int *i){ *i = 100; *p = i; }

それと、コンパイラには警告という機能があるので、基本これをONにして使うよいです。例えばgccでは普通は-Wallオプションを使います。すると以下のように怪しい箇所を警告してくれます。

ファイル名:行: warning: ‘p’ is used uninitialized in this function [-Wuninitialized] ファイル名:行: warning: ‘i’ is used uninitialized in this function [-Wuninitialized]

投稿2016/05/26 02:00

sharow

総合スコア1149

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

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

welld

2016/05/26 02:11 編集

回答有り難うございます。
guest

0

int *p はポインタです。
使用する前にアドレスを設定する必要があります。
例えば、int a という変数を作成し、p = &a; とします。

C

1int main(void){ 2 int *p,i,a; 3 p = &a; 4 func(p,i); 5 printf("%d\n",*p); 6 return 0; 7}

これでpはaを参照することになります。
つまり、
printf("%d\n",*p);

printf("%d\n",a);
は同じ値になります。

投稿2016/05/26 01:45

ttyp03

総合スコア16998

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

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

welld

2016/05/26 01:50

回答ありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問