C言語について教えてください。
大学の課題でCを書いています。
freeの仕方について教えてください。
構造体
struct word {
struct word *next;
char *st;
int *line_no;
int line_cnt;
};
を用意し、char* stにstrdupにて文字列を格納します。
そして、簡単なリストを作り、表示をさせfreeをするプログラムです。
freeをする際にfreeしてはいけないメモリをfreeしているようです。
しかし、正しい場所をfreeしていると思うので、なぜエラーなのか分かりません。
理由と解決方法を御教授願えませんでんしょうか。
よろしくお願いします。
/************以下ソース****************/
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
struct word {
struct word *next;
char *st;
int *line_no;
int line_cnt;
};
struct word *head;
void Add_List(char *buf,int no){
struct word* node = malloc(sizeof(struct word*));
struct word* crnt;
/*nodeの準備*/
node->next = NULL;
node->st = strdup(buf);
node->line_no = malloc(sizeof(int*));
node->line_no[0] = 1;
node->line_cnt= 1;
if(head == NULL) {
head = node;
} else {
crnt = head;
while( crnt->next != NULL) {
crnt = crnt->next;
}
crnt->next = node;
}
}
void printlist(){
struct word *crnt = head;
while( crnt != NULL ) {
printf("%s\n",crnt->st);
crnt = crnt->next;
}
}
int main(){
head = NULL;
char buf1[] = {'A','B','C','D','E','\0'};
char buf2[] = {'F','G','H','I','J','\0'};
char buf3[] = {'K','L','M','N','O','\0'};
Add_List(buf1,1);
Add_List(buf2,1);
Add_List(buf3,1);
printlist();
/*この部分がうまくいかない*/
free(head->st);
/*********************/
}
(以下実行結果です。OS:CentOS)
[yuichiro-ito@localhost test]$ ./a.out
ABCDE
FGHIJ
KLMNO
*** glibc detected *** ./a.out: free(): invalid pointer: 0x0000000001a01030 ***
======= Backtrace: =========
/lib64/libc.so.6[0x365a4760e6]
./a.out[0x40072f]
/lib64/libc.so.6(__libc_start_main+0xfd)[0x365a41ecdd]
./a.out[0x4004d9]
======= Memory map: ========
00400000-00401000 r-xp 00000000 08:02 390975 /home/yuichiro-ito/test/a.out
00600000-00601000 rw-p 00000000 08:02 390975 /home/yuichiro-ito/test/a.out
01a01000-01a22000 rw-p 00000000 00:00 0 [heap]
3659c00000-3659c20000 r-xp 00000000 08:02 785966 /lib64/ld-2.12.so
3659e1f000-3659e20000 r--p 0001f000 08:02 785966 /lib64/ld-2.12.so
3659e20000-3659e21000 rw-p 00020000 08:02 785966 /lib64/ld-2.12.so
3659e21000-3659e22000 rw-p 00000000 00:00 0
365a400000-365a58a000 r-xp 00000000 08:02 785967 /lib64/libc-2.12.so
365a58a000-365a789000 ---p 0018a000 08:02 785967 /lib64/libc-2.12.so
365a789000-365a78d000 r--p 00189000 08:02 785967 /lib64/libc-2.12.so
365a78d000-365a78e000 rw-p 0018d000 08:02 785967 /lib64/libc-2.12.so
365a78e000-365a793000 rw-p 00000000 00:00 0
3666c00000-3666c16000 r-xp 00000000 08:02 785985 /lib64/libgcc_s-4.4.7-20120601.so.1
3666c16000-3666e15000 ---p 00016000 08:02 785985 /lib64/libgcc_s-4.4.7-20120601.so.1
3666e15000-3666e16000 rw-p 00015000 08:02 785985 /lib64/libgcc_s-4.4.7-20120601.so.1
7fc7b9e92000-7fc7b9e95000 rw-p 00000000 00:00 0
7fc7b9ea0000-7fc7b9ea3000 rw-p 00000000 00:00 0
7fffbce3c000-7fffbce51000 rw-p 00000000 00:00 0 [stack]
7fffbce9d000-7fffbce9e000 r-xp 00000000 00:00 0 [vdso]
ffffffffff600000-ffffffffff601000 r-xp 00000000 00:00 0 [vsyscall]
A 回答 (1件)
- 最新から表示
- 回答順に表示
No.1
- 回答日時:
free()の使い方ではなくてmalloc()の使い方が間違ってます.
> struct word* node = malloc(sizeof(struct word*));
これだと sizeof(struct word*) バイト、つまりポインタのバイト数 (普通、4バイトとか8バイト)のメモリ しか
確保されていません。
そこに無理矢理構造体のデータを書き込んでしまっているのでメモリ管理が破綻してます。
構造体本体を確保する場合はこうします。
struct word* node = malloc(sizeof(struct word));
> node->line_no = malloc(sizeof(int*));
こちらも同様。 ポインタ line_no で示される位置に格納されるのは int* ではなくて int ですから
node->line_no = malloc(sizeof(int));
とします。
余談ですが、malloc()やstrdup()などで動的にメモリを確保した際には、必ずNULLかどうかチェックしてメモリ不足を捕まえておかないと後で苦労する事になります。
(本格的なプログラムを書き始めると、それこそどこが原因でエラーが起きてるのか判らなくなります)
今からCを習得されるのでしたら、C++から入った方が楽です。
malloc()しません.し、
( struct word* node = new struct word; とします)
もし間違って struct word* node = new struct word*;
と書いてしまった場合ばコンパイラがエラー吐いて教えてくれますから.
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
このQ&Aを見た人はこんなQ&Aも見ています
-
餃子を食べるとき、何をつけますか?
みんな大好き餃子。 ふと素朴な疑問ですが、餃子には何をつけて食べますか? 王道は醤油とお酢でしょうか。
-
大人になっても苦手な食べ物、ありますか?
大人になっても、我慢してもどうしても食べれないほど苦手なものってありますよね。 あなたにとっての今でもどうしても苦手なものはなんですか?
-
CDの保有枚数を教えてください
ひとむかし前はCDを買ったり借りたりが主流でしたが、サブスクで簡単に音楽が聴ける今、CDを手に取ることも減ってきたかと思います。皆さんは2024年現在、何枚くらいCDをお持ちですか?
-
許せない心理テスト
私は「あなたの目の前にケーキがあります。ろうそくは何本刺さっていますか」と言われ「12本」と答えたら「ろうそくの数はあなたが好きな人の数です」と言われ浮気者扱いされたことをいまだに根に持っています。
-
あなたの習慣について教えてください!!
あなたが習慣だと思って実践しているものを共有してくださいませんか? 筋肉トレーニングでも朝シャワーでも、あなたが習慣だなと思えば何でも構いません
-
C言語で、メモリを解放しないで終わるプログラム
C言語・C++・C#
-
Enterキーを押されたら次の処理に移るという事をしたい。
C言語・C++・C#
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・【大喜利】【投稿~11/12】 急に朝起こしてきた母親に言われた一言とは?
- ・好きな和訳タイトルを教えてください
- ・うちのカレーにはこれが入ってる!って食材ありますか?
- ・好きな「お肉」は?
- ・あなたは何にトキメキますか?
- ・おすすめのモーニング・朝食メニューを教えて!
- ・「覚え間違い」を教えてください!
- ・とっておきの手土産を教えて
- ・「平成」を感じるもの
- ・秘密基地、どこに作った?
- ・【お題】NEW演歌
- ・カンパ〜イ!←最初の1杯目、なに頼む?
- ・一回も披露したことのない豆知識
- ・これ何て呼びますか
- ・チョコミントアイス
- ・初めて自分の家と他人の家が違う、と意識した時
- ・「これはヤバかったな」という遅刻エピソード
- ・これ何て呼びますか Part2
- ・許せない心理テスト
- ・この人頭いいなと思ったエピソード
- ・牛、豚、鶏、どれか一つ食べられなくなるとしたら?
- ・あなたの習慣について教えてください!!
- ・ハマっている「お菓子」を教えて!
- ・高校三年生の合唱祭で何を歌いましたか?
- ・【大喜利】【投稿~11/1】 存在しそうで存在しないモノマネ芸人の名前を教えてください
- ・好きなおでんの具材ドラフト会議しましょう
- ・餃子を食べるとき、何をつけますか?
- ・あなたの「必」の書き順を教えてください
- ・ギリギリ行けるお一人様のライン
- ・10代と話して驚いたこと
- ・家の中でのこだわりスペースはどこですか?
- ・つい集めてしまうものはなんですか?
- ・自分のセンスや笑いの好みに影響を受けた作品を教えて
- ・【お題】引っかけ問題(締め切り10月27日(日)23時)
- ・大人になっても苦手な食べ物、ありますか?
- ・14歳の自分に衝撃の事実を告げてください
- ・架空の映画のネタバレレビュー
- ・「お昼の放送」の思い出
- ・昨日見た夢を教えて下さい
- ・ちょっと先の未来クイズ第4問
- ・【大喜利】【投稿~10/21(月)】買ったばかりの自転車を分解してひと言
- ・メモのコツを教えてください!
- ・CDの保有枚数を教えてください
- ・ホテルを選ぶとき、これだけは譲れない条件TOP3は?
- ・家・車以外で、人生で一番奮発した買い物
- ・人生最悪の忘れ物
- ・【コナン30周年】嘘でしょ!?と思った○○周年を教えて【ハルヒ20周年】
- ・あなたの習慣について教えてください!!
- ・都道府県穴埋めゲーム
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
文字列から空白を取り除きたい...
-
fgetsなどのときのstdinのバッ...
-
テキストデータをそのままバイ...
-
配列をnビットシフトする
-
charでの計算?
-
sprintfに同じ変数は使えるか
-
C言語の入力した文字を反転させ...
-
CStringをwchar_tに変換したい
-
YUV⇔RGB変換がうまくいきません。
-
絶対パスからのファイル名の切...
-
C言語です
-
strncpyと_tcsncpy_sのヌルの扱...
-
C言語のfor文です。 繰り返しの...
-
atoi( ) の反対をやりたい
-
C言語のポインターで詰まっている
-
データの取得をしたいのですが…
-
下記のプログラムがコンパイラ...
-
間接参照のレベルが異なっています
-
型変換
-
C言語 ミリ秒を日付に変換には
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
charでの計算?
-
文字列から空白を取り除きたい...
-
CStringをwchar_tに変換したい
-
C言語のfor文です。 繰り返しの...
-
charからLPTSTRへの変換方法
-
fgetsなどのときのstdinのバッ...
-
'const char *' 型は 'char *' ...
-
間接参照のレベルが異なっています
-
double型の値をchar配列に変換...
-
atoi( ) の反対をやりたい
-
間接操作のレベルとは
-
ネットワークにつながっている...
-
型変換
-
テキストデータをそのままバイ...
-
文字列ポインタを結合
-
3桁区切(コンマ)記号をつけ...
-
C言語です
-
Win32APIでのエディットボック...
-
TCHAR文字列?の特定部分の数字...
-
絶対パスからのファイル名の切...
おすすめ情報