🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
C

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

Q&A

解決済

1回答

383閲覧

C言語での簡単な構造体リストの実現

wataamenoarashi

総合スコア9

C

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

0グッド

0クリップ

投稿2019/12/10 09:13

前提・実現したいこと

ここに質問の内容を詳しく書いてください。
C言語で一番シンプルな構造体リストを実現しようとしているのですが、できずに困っています。

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

エラーメッセージ

Segmentation fault (core dumped)

該当のソースコード

C言語

#include <stdio.h>
#include <malloc.h>

typedef struct _Node {
int x;
struct _Node *next;
} Node;

Node *gen_node(void);

int main(void){
printf("flag0");//デバッガ
Node *head, *p;
// headはリストの先頭を,pは新規ノードを指す
printf("flag1"); //デバッガ

// リストの作成
head=NULL;
while(p=gen_node(),scanf("%d",p->x)!=EOF){
printf("flag2"); //デバッガ
p->next=head; // 新規ノードのnextは,これまでのリストの先頭を指す
head=p; // リストの先頭を新規ノードに更新
}
printf("flag3"); //デバッガ
// リストの表示
p=head;
while(p!=NULL){
printf("%d\n",p->x);
p=p->next;
}

return 0;
}

Node *gen_node(void)
{
return (Node *)malloc(sizeof(Node));
}

//実行時には、このプログラムをexeとおいて、
./exe < test.txt
でtest.txtを読み込ませた。
test.txtは以下の通り。

1
4
6
8
9
11
13
16
21
22
27
28
31
35
37
39
41
44
48

試したこと

printf("flag")を入れてまずどこがエラーの原因になっているか絞ろうとした。
結果 → mainの一番最初に入れたflag0も表示されないので、もはやどこに問題があるのかもわからない。

補足情報(FW/ツールのバージョンなど)

CentOS Linux 7を使っています。

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

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

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

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

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

guest

回答1

0

ベストアンサー

C

while(p=gen_node(),scanf("%d",p->x)!=EOF){

scanfの第二引数以降はアドレスを渡すので、scanf("%d", &p->x) が正しいです。

コンパイラの警告オプションをオンにしておくと、ちゃんと叱って貰えます。
例えばこんなふうに。

gcc

1prog.c:19:28: warning: format '%d' expects argument of type 'int *', but argument 2 has type 'int' [-Wformat=] 2 19 | while(p=gen_node(),scanf("%d", p->x)!=EOF){ 3 | ~^ ~~~~ 4 | | | 5 | | int 6 | int *

printf("flag")を入れてまずどこがエラーの原因になっているか絞ろうとした。

結果 → mainの一番最初に入れたflag0も表示されないので、もはやどこに問題があるのかもわからない。

デバッグプリントを用いること自体は非常に良いアイデアです。
しかしこのようなケースでは、バッファに文字列を溜め込まないよう工夫しなければなりません。

普通は printf("flag\n") などと改行するだけでも改善するでしょう。

コードの貼り方について

teratailには、コードを見易く表示する機能があります。
質問編集画面を開き、コードを選択した状態で<code>ボタンを押して下さい。
C

投稿2019/12/10 09:19

編集2019/12/10 09:23
LouiS0616

総合スコア35668

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

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

asm

2019/12/10 09:36

> 普通は printf("flag\n") などと改行するだけでも改善するでしょう。 putsでいいですし、デバッグプリントならばfprintf(stderr, "flag");を使うと stderrは既定でバッファリングされない事も覚えておくと便利ですね。
wataamenoarashi

2019/12/10 09:39

丁寧な解説本当にありがとうございます!
LouiS0616

2019/12/10 09:40

@asm さん なるほど、確かにこのようなときは標準エラーに吐く方がより適切ですね。 コメントありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.36%

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

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

質問する

関連した質問