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

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

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

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

Q&A

解決済

4回答

1888閲覧

C言語で数字のカウントについて

aomi333

総合スコア1

C

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

0グッド

0クリップ

投稿2022/05/19 14:59

編集2022/05/20 01:39

txtファイルを読み込み0~9までの数字のカウントを行い、一番多い数字を出力したいです。

txtファイルにある数字の0~9までのカウントはできているのですが、
以下の2つの対応がどのようしたら良いかわかりません。
・標準入力にある桁数のみカウントを行う。(下の例ですと24桁ありますが、20桁のみカウントしたいです)
・カウントした中で、最大値のみを標準出力する。(同数が複数あれば若い番号順に出力する)

数字のカウントだけはできていると思っていますが、初学者のため、あまり難しいコードはわかりません。
ご教授いただければ幸いです。
よろしくお願いします。

【txtファイル】
3456543154897978017060606
【標準入力】
20

列1【現在標準出力に出力されている結果】【20桁(標準入力にて指定)のみのカウント結果】
042
122
322
433
533
641
733
822
922

【標準出力に出力したい結果】
4:3
5:3
7:3

C

1#include <stdio.h> 2 3int main(int argc, char *argv[]) { 4 5 FILE *fp; 6 char ch; 7 int i; 8 9 //数字の出現回数の初期化 10 int cnt[10] = {0}; 11 12 //ファイルを読み取り専用で開く。失敗するとNULLを返す。 13 fp = fopen(argv[1], "r"); 14 if (fp == NULL) { 15 printf("%s : file not opened.\n", argv[1]); 16 return 1; 17 } 18 19 //chに1文字ずつ読み込む 20 while ((ch = fgetc(fp)) != EOF) { 21 if((ch >='0') && (ch <= '9')) { 22 cnt[ch - '0']++; 23 } 24 } 25 for (i = 0; i <= 9; i++) { 26 if (cnt[i] >= 1){ 27 printf("%c:%d\n", '0' + i, cnt[i]); 28 } 29 } 30 31 fclose(fp); 32 33 return 0; 34}

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

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

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

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

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

jimbe

2022/05/19 16:44 編集

>数字のカウントについて >文字のカウントだけはできている それで、何をご質問なのでしょうか。
episteme

2022/05/19 16:43

> scanf("%s", file); コレは何のため?
dodox86

2022/05/19 23:34

何が分からなくて詰まっているのか、ちゃんと説明しましょう。 例えば > カウントを行う桁数は標準入力で決まっています。 ですが、現状のコードに反映されていないように見えるので標準入力から桁数を読み込むのが仕様なのか、その方法が知りたいのか分かりません。 teratailは「何でも相談室」のような場所ではないので、要望だけ書いても意図を汲むように読まないと分かりづらいです。 [質問するときのヒント] https://teratail.com/help/question-tips [推奨していない質問] https://teratail.com/help/avoid-asking また初心者であれば初心者マーク(若葉マーク)を質問につけることができ、閲覧した方々の目に留まります。
BeatStar

2022/05/20 02:38

絶対にダメということはありませんが、一つ目のif文とその次のwhile文、さらにその次のfor文のインデントがちょっと変です。 C言語タイプの言語だと問題はそこまでないですが、Pythonみたいなインデントでブロックを表現する言語だと意味が相当変わってきます。そうなると意図しない動作になり、頭を抱えることになりますからインデントをそろえる癖をつけた方がいいですよ。 それにfor文の中にfor文があり、その中にfor文が…みたいに入れ子になるとインデントがぐちゃぐちゃだと可読性が相当低くなりますし。 一応C言語系だとそこまで問題はないですが。
dodox86

2022/05/20 06:06

> ・標準入力にある桁数のみカウントを行う。 ひとつには、用語の使い方から推測するに標準入力からのデータ、ここでは整数で桁数の取り込みについて具体的な方法がピンと来ていないのではないかな、と。標準入力が良く分かっていないのか。一般的にはコンソールからの入力を指します。scanf関数での入力です。 コンソールが良く分かっていない可能性もありますし、取り込んだ値をループなどで具体的に、どのようにコードに反映したら良いか分からない、とかかもしれませんが。
guest

回答4

0

最大値のみを標準出力する。(同数が複数あれば若い番号順に出力する)

っていうのを達成するための手続きがわからん,というのであれば,この文言をもっと丁寧に書き下すとよいのかもしれない.
例えば,以下のようになるだろうか.

  • cnt[0] が最大値ならば cnt[0] を出力する
  • cnt[1] が最大値ならば cnt[1] を出力する
  • …(中略)
  • cnt[9] が最大値ならば cnt[9] を出力する

これを{素直に,ありのままに,愚直に}実装するならば,必要なもの(この時点で明確になっていないもの)は「cnt[X]が最大値かどうかを判定する手段」だけである.
でも,その判定は簡単だよね.
10個の数値 cnt[0]cnt[9] を全部チェックして,cnt[X] よりも大きい値が1つも無いならば,cnt[X]は最大値であるハズだ.

…っていうことで,必要な手続きが全て明確になれば,あとは実装するだけだ.
完遂.

#もちろん,この話だと「効率悪そう」なことは明確と思えるが,処理効率がどうのとかいう話は,とりあえずそういうのを度外視した簡素な物が正しく動く状態を作ってから別途考えても遅くないんじゃないかな(このような,勉強目的なものに取り組んでいる状況においては).

投稿2022/05/20 05:35

fana

総合スコア11658

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

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

fana

2022/05/20 05:40

他の回答にある, まず最大値を求める → cnt[X]が最大値かどうかの判定は「cnt[X]が先に求めておいた最大値と同一であるか」という方法で行う という話は,「効率の面で一歩進んだ方法」と言えるであろう.
guest

0

ベストアンサー

以下のようにしてください。
不明点は補足してください。
char ch;ではなく
int ch;にしました。(fgetcの戻り値はint型です)

C

1#include <stdio.h> 2 3int main(int argc, char *argv[]) 4{ 5 6 FILE *fp; 7 int ch; 8 int i; 9 int inputd; 10 int ctr = 0; 11 int max = -1; 12 //数字の出現回数の初期化 13 int cnt[10] = { 0 }; 14 printf("input value(1-100)=>"); 15 scanf("%d",&inputd); 16 if (inputd < 1 || inputd > 100){ 17 printf("input error!! <value=%d>\n",inputd); 18 return 1; 19 } 20 //ファイルを読み取り専用で開く。失敗するとNULLを返す。 21 fp = fopen(argv[1], "r"); 22 if (fp == NULL) { 23 printf("%s : file not opened.\n", argv[1]); 24 return 1; 25 } 26 //chに1文字ずつ読み込む 27 while ((ch = fgetc(fp)) != EOF) { 28 if ((ch >= '0') && (ch <= '9')) { 29 cnt[ch - '0']++; 30 } 31 ctr++; 32 if (ctr >= inputd) break; 33 } 34 for (i = 0; i < 9; i++) { 35 if (cnt[i] > max) { 36 max = cnt[i]; 37 } 38 } 39 for (i = 0; i < 9; i++) { 40 if (cnt[i] == max) { 41 printf("%c:%d\n", '0' + i, cnt[i]); 42 } 43 } 44 45 fclose(fp); 46 47 return 0; 48} 49

投稿2022/05/20 09:07

tatsu99

総合スコア5438

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

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

aomi333

2022/05/22 02:18

tatsu99さん ありがとうございます。 想像している通りの動きになりました。 最後のforのところがはこういう記述にすればよかったのですね。 1つ質問なのですが、int max = -1となっていますが、-1となっているのはなぜでしょうか? 0からではだめなのでしょうか?
tatsu99

2022/05/22 02:52

すみません。前の2つのコメントは不適切でした。無視してください。
tatsu99

2022/05/22 02:57

int max = 0; でも問題ありません。同じ結果になります。
aomi333

2022/05/22 03:41

tatsu99さん 検証していただきましてありがとうございました。
aomi333

2022/05/22 03:44

tatsu99さん 質問にもご回答いただき、すごく勉強になりました。 大変ありがとうございました。
guest

0

数字のカウントだけはできていると思っていますが、初学者のため、あまり難しいコードはわかりません。

なんか根本的に勘違いしている気がする…
そもそもプログラミングっていうのは既存のコードを切り貼りする行為…ではなく、丸暗記して挑むこと…でもありません。プログラミングっていうのは「現実世界で、人間がやっている手作業を機械に任せるために命令する事」です。だから「現実世界でやっている手順を書けばいい」のです。もちろんデバッグやテストも含めて。

ちなみに数字のカウントについては大体OKです。

・標準入力にある桁数のみカウントを行う。(下の例ですと24桁ありますが、20桁のみカウントしたいです)

質問者さんは手作業で上記のような処理をする場合、どのようにしますか?
現実世界でいえば、

[依頼] Aさんが桁数を言うからその桁数分、ノートにある数字を黒板に書いてくれないか?

みたいな依頼かなんかだと考えてみてください。
私なら

1. Aさんに「桁数は?」と聞く 2. (1)で受け取った数字を桁数と仮定して以下を行う 2.1. (1)の桁数分ループして以下を処理 2.1.1. 一文字読み込んで 2.1.2. (2.1.1)を黒板に一文字書く

的な処理をすると思います。

3456543154897978017060606がノートに書かれていて、Aさんが「5」と言ったとする。
その場合はまず、(1)で質問して「5」が返ってくるので、桁数 = 5 つまり 5桁読み込んで書くという事です。

なので(2) -> (2.1)で5桁分読み込んで書く。その方法が(2.1.1)で一文字分読み込んで(2.1.2)で書く。
最初の数字は3なので 3を黒板に書く。
で(2.1)に戻って(2.1.1)に行って二文字目、つまり4を黒板に書く。
…と5文字分書く。

これだけです。これを実装してデバッグするだけです。

・カウントした中で、最大値のみを標準出力する。(同数が複数あれば若い番号順に出力する)

これも同じように考える。

最大値とはなんでしょうか? 最大値とは『最も大きい値』(← tautologyやね)です。
つまり、対象範囲の中にあるデータの中で一番大きい数字を探せばいいのです。
で、その大きい数字を表示するとか。

ではどういう最大値でしょうか? 今回は個数ですよね。ということは個数をキーにして最大値を計算すればいいのです。

厄介なのが「同数が複数あれば若い番号順に出力する」ですね。
つまり複数個の数字が答えになる可能性があると。そういう場合は現実世界だとどうしますか?

私なら、

1. (個数の)最大値を計算する 2. データの個数分、以下をループする 2.1. 最大値と同じ個数かどうか 2.1.1. (同じであれば)書きだす

的な感じになりますね。これを実装すればいいだけです。

投稿2022/05/20 03:10

編集2022/05/20 03:15
BeatStar

総合スコア4958

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

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

aomi333

2022/05/22 03:46

BeatStarさん 解説していただきまして、ありがとうございました。 フローを書いていくとわかりやすいですね。 プログラムを書こうとすると??になる部分がありますが。。。
guest

0

0から9までの数字の中で、出現回数の最大は何回か? (0回からスタートして、直前のものより回数が多ければ、次からはそれを基準にする)

0から9までの数字の中で、上で出した最大回数と同じ数字だけを対象に、数字と回数を出力する。

この2つはできそうですか?

投稿2022/05/19 22:34

majiponi

総合スコア1720

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

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

aomi333

2022/05/20 02:34

majiponiさん 最大回数の3を出すことはできました。 上で出した最大回数と同じ数字と回数を出力するやり方が思いつきません。 #include <stdio.h> int main(int argc, char *argv[]) { FILE *fp; char ch; int i; // 出現回数の最大値 int max = 0; //数字の出現回数の初期化 int cnt[10] = {0}; //ファイルを読み取り専用で開く。失敗するとNULLを返す。 fp = fopen(argv[1], "r"); if (fp == NULL) { printf("%s : file not opened.\n", argv[1]); return 1; } //chに1文字ずつ読み込む while ((ch = fgetc(fp)) != EOF) { if((ch >='0') && (ch <= '9')) { cnt[ch - '0']++; } } for (i = 0; i <= 9; i++) { if (cnt[i] >= max){ max = cnt[i]; } } printf("%d", max);//最大回数を表示させる fclose(fp); return 0; }
jimbe

2022/05/20 02:59 編集

コードは出来るだけご質問に追加編集する形でお提示ください。 最大回数を求めるループの if 文の式を >= から == に変えたループを追加して、成立したら表示する…という考え方は如何でしょうか。
episteme

2022/05/20 02:59

> 最大回数と同じ数字と回数を出力するやり方が思いつきません。 i = 0, 1, ... 9 に対し、cnt[i] == 最大回数 なら出力する。でよくね?
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問