今学校で二分探索木を勉強しています。二分探索木に要素を挿入したいのですが、うまくいかないのでアドバイスをいただけないでしょうか。ファイル中の英文を単語に分けてその出現頻度をカウントするプログラムです。とりあえず二分探索木を作るところまではなんとか完成させたいです。
#include <stdio.h>
#include <stdlib.h>
#include<string.h>
#include <ctype.h>
typedef struct node Node;
struct node{
char *word;
int count;
Node *left,*right;
};
Node *root=NULL;
Node *compose(FILE *fp);
void inorder(Node *p);
int main(int argc, char *argv[])
{
FILE *fp;
Node *new;
fp=fopen(argv[1],"r");
if(fp==NULL){
puts("ファイルを開けません");
return(-1);
}
new=(Node *)malloc(sizeof(Node *));
new=compose(fp);
inorder(new);
return (0);
}
Node *compose(FILE *fp)
{
Node **p,*new;
char buf[20];
p=&root;
while(fscanf(fp,"%[-a-z-A-Z0-9_]",buf)!=EOF){
while(*p!=NULL){
(*p)->count=0; /*countを0で初期化したいけどwhileの外にこの1行を出すとエラーが出る*/
buf[0]=tolower(buf[0]);
strdup(buf);
if(strcasecmp(buf,(*p)->word)==0){ /*if文にしかはいらない…*/
(*p)->count++;
printf("%d\n",(*p)->count);
}else if(strcasecmp(buf,(*p)->word)<0){
p=&(*p)->left;
}else{
p=&(*p)->right;
}
}
new=(Node *)malloc(sizeof(Node *));
new->left=NULL;
new->right=NULL;
new->word=buf;
*p=new;
fscanf(fp,"%*[^-a-z-A-Z0-9_]");
}
return(new);
}
void inorder(Node *p)
{
if(p==NULL)
return;
printf("%s",p->word);
if(p->count!=0){
printf("%d",p->count);
}
inorder(p->right);
inorder(p->left);
}
No.4ベストアンサー
- 回答日時:
なんとなく動くようにしてみました。
参考にしていただければ。。#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
typedef struct node Node;
struct node{
char *word;
int count;
Node *left,*right;
};
Node *root=NULL;
void compose(FILE *fp);
void inorder(Node *p);
void strlower(char *s);
int main(int argc, char *argv[]) {
FILE *fp;
Node *new;
fp=fopen(argv[1],"r");
if(fp==NULL){ puts("ファイルを開けません"); return(-1); }
compose(fp);
inorder(root);
return (0);
}
/* 文字列を小文字にする */
void strlower(char *s) {
while(*s!=NULL) { *s=tolower(*s); s++; }
}
/* 木を作る */
void compose(FILE *fp) {
Node **p,*new;
char buf[256];
while(1) {
fscanf(fp,"%[^a-zA-Z0-9]",buf); /* 英数字がくるまで読み飛ばす */
if (fscanf(fp,"%[a-zA-Z0-9]",buf)==EOF) break; /* ファイルから単語らしきものを読み込む */
strlower(buf); /* 文字列を小文字化 */
if (root==NULL) { /* 最初の単語ならすぐ登録 */
new=(Node *)malloc(sizeof(Node));
new->left=NULL; new->right=NULL; new->word=strdup(buf); new->count=1;
root=new;
} else { /* 最初でなければ大小比較して挿入場所を探す */
*p=root; /* 注目ノードを根に設定 */
while(1){
if(strcmp(buf,(*p)->word)==0){ /* 同じデータがあればカウントアップして終わり */
(*p)->count++; break;
} else if(strcmp(buf,(*p)->word)<0){ /* 新単語が小さければ左を見る */
if ((*p)->left==NULL) { /* 子がなければ新規登録 */
new=(Node *)malloc(sizeof(Node));
new->left=NULL; new->right=NULL; new->word=strdup(buf); new->count=1;
(*p)->left=new; break;
} else { /* 左を探す */
*p=(*p)->left;
}
} else { /* 新単語が大きければ右を見る */
if ((*p)->right==NULL) { /* 子がなければ新規登録 */
new=(Node *)malloc(sizeof(Node));
new->left=NULL; new->right=NULL; new->word=strdup(buf); new->count=1;
(*p)->right=new; break;
}else{ /* 右を探す */
*p=(*p)->right;
}
}
}
}
}
}
/* 小さい順に表示 */
void inorder(Node *p) {
if (p==NULL) return;
inorder(p->left);
printf("%s %d\n",p->word, p->count);
inorder(p->right);
}
お礼が遅くなってしまい申し訳ありません。
友人と相談して関数をもう一つ付け加えたら挿入できるようになりました。JaritenCatさんのソースもまた勉強の材料にさせていただきます。ありがとうございました!
No.3
- 回答日時:
> if(strcasecmp(buf,(*p)->word)==0){ /*if文にしかはいらない…*/
構造体のメンバーwordには適切な値が入っていますか?
> *p=new;
> fscanf(fp,"%*[^-a-z-A-Z0-9_]");
ここのfscanf()の意味合いがわかりません。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- C言語・C++・C# c言語の問題です 3 2023/01/10 16:15
- C言語・C++・C# プログラミングの授業の課題です 1 2023/01/17 22:15
- C言語・C++・C# c言語の問題の説明、各所ごとに 5 2023/07/26 11:03
- 大学・短大 C言語線形リストの問題です 3 2022/12/22 00:45
- C言語・C++・C# バイナリファイルをコピーするのにかかる時間を測りたいのですが実行するとFatel error:gli 2 2022/11/03 01:10
- C言語・C++・C# C言語で再起関数とポインタを用いて文字列反転をする方法がわかりません。 4 2023/04/29 20:32
- C言語・C++・C# C言語のエラーについて 2 2022/07/11 13:56
- C言語・C++・C# プログラミング c言語 4 2023/03/07 01:05
- C言語・C++・C# #include <stdio.h>int main(void) { int buf[100] = 6 2022/11/01 22:45
- C言語・C++・C# このプログラミング誰か教えてくれませんか 1 2022/06/02 15:27
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・【大喜利】【投稿~11/22】このサンタクロースは偽物だと気付いた理由とは?
- ・お風呂の温度、何℃にしてますか?
- ・とっておきの「まかない飯」を教えて下さい!
- ・2024年のうちにやっておきたいこと、ここで宣言しませんか?
- ・いけず言葉しりとり
- ・土曜の昼、学校帰りの昼メシの思い出
- ・忘れられない激○○料理
- ・あなたにとってのゴールデンタイムはいつですか?
- ・とっておきの「夜食」教えて下さい
- ・これまでで一番「情けなかったとき」はいつですか?
- ・プリン+醤油=ウニみたいな組み合わせメニューを教えて!
- ・タイムマシーンがあったら、過去と未来どちらに行く?
- ・遅刻の「言い訳」選手権
- ・好きな和訳タイトルを教えてください
- ・うちのカレーにはこれが入ってる!って食材ありますか?
- ・おすすめのモーニング・朝食メニューを教えて!
- ・「覚え間違い」を教えてください!
- ・とっておきの手土産を教えて
- ・「平成」を感じるもの
- ・秘密基地、どこに作った?
- ・【お題】NEW演歌
- ・カンパ〜イ!←最初の1杯目、なに頼む?
- ・一回も披露したことのない豆知識
- ・これ何て呼びますか
- ・初めて自分の家と他人の家が違う、と意識した時
- ・「これはヤバかったな」という遅刻エピソード
- ・これ何て呼びますか Part2
- ・許せない心理テスト
- ・この人頭いいなと思ったエピソード
- ・牛、豚、鶏、どれか一つ食べられなくなるとしたら?
- ・好きなおでんの具材ドラフト会議しましょう
- ・餃子を食べるとき、何をつけますか?
- ・あなたの「必」の書き順を教えてください
- ・ギリギリ行けるお一人様のライン
- ・10代と話して驚いたこと
- ・大人になっても苦手な食べ物、ありますか?
- ・14歳の自分に衝撃の事実を告げてください
- ・家・車以外で、人生で一番奮発した買い物
- ・人生最悪の忘れ物
- ・あなたの習慣について教えてください!!
- ・都道府県穴埋めゲーム
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
隣接行列を読み込んで有向グラ...
-
ガンマ変換 C言語でプログラ...
-
複数ファイルの同時読み込みの...
-
C言語でファイル読み書きを早く...
-
C言語のファイル読み込み
-
【C言語】ファイルを読み込んで...
-
音声データを出力するCプログラ...
-
FeliCaを利用した認証システム
-
C言語でクロマキー合成をする方法
-
VisualStudioでのファイルの入...
-
クリップボードから画像取得
-
fopenの使い方
-
日本語ファイル名のFTPについて
-
C言語 連番データの読み込み
-
バイナリファイルの読み込み(C...
-
大量の入力ファイルを扱うとき...
-
Syntax Errorで困っています。
-
C言語について質問です。
-
C言語でファイル名を変数にした...
-
C言語の課題です
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
複数ファイルの同時読み込みの...
-
c言語でのfscanfについて
-
ガンマ変換 C言語でプログラ...
-
fopenでファイル名に、変数を使...
-
エラーがわかりません、、
-
ファイル出力で改行を入れたい!
-
C言語でクロマキー合成をする方法
-
C言語でファイル読み書きを早く...
-
ファイルが読み込めない・・・
-
テキストファイル内に対して, ...
-
ファイルへの書込み処理が異常...
-
fscanfでループしてしまう。
-
音声データを出力するCプログラ...
-
fgets関数の利用 c言語
-
fgets( ) の返り値は何?
-
日本語ファイル名のFTPについて
-
大量の入力ファイルを扱うとき...
-
二分探索木への挿入
-
【C言語】ファイルを読み込んで...
-
[C言語]2つのファイルを用いた...
おすすめ情報