
(環境はmacで, gcc 4.0.1を用いています)
#include <stdio.h>
main(void)
{
int n, k, nck;
printf("n = ");
scanf("%d", &n);
printf(" k nCk\n");
k = 0;
nck = 1;
printf("%12d%12d\n", k, nck);
for(k=1; k <= n; k++) {
nck = nck * (n-k+1)/k;
printf("%12d%12d\n", k, nck);
}
}
とした場合, 入力した値n=29までは正しい答えが得られるのですが
n=30以降では途中から答えが狂い始めます.
この現象はなぜ起こるのでしょうか?
No.3ベストアンサー
- 回答日時:
この方法で、この環境だと、n=30, k=14 までは正しく求めることができます。
30C14=145,422,675に(30-15+1)=16を掛けると、intで取り扱うことができる最大の数値(2の31乗よりも1小さい値:2,147,483,647)よりも大きくなってしまいます(オーバーフローです)。以降は正しくないnckの値を使って次の値を求めているので当然正しい値は得られません。
#1の回答の「(n-k+1)/kは恒に整数か?」は何を指摘されているのかわかりません。nck*(n-k+1)は、オーバーフローしない限り、kで割り切れます。
また、「nckの最大値は?」については、n≦33であればnCkの値がintの最大値を超えることはありません。
ご丁寧にありがとうございます.
電卓を取り出して確認していたらnckを求める式の途中でintの範囲を超えていることに気づきました.
そうこうしていると皆さんへのお礼が遅くなりました, 申し訳ないです.
今回はakayoroshiさんの回答を見ることで自分の考えが正しいのかどうかが確認できましたのでベストアンサーとさせて頂きました.
回答して頂いた皆さん, あらためてありがとうございました.
これでスッキリして眠れます...
No.4
- 回答日時:
integer overflowです。
正しい結果がほしいなら、gmpなどを使うことを検討すると良いかもしれません。
http://gmplib.org/
参考までに、こんな感じに書き換えられます。 gmpのライブラリーとリンクしてください。
#include <gmp.h>
#include <stdio.h>
void
print_nck(unsigned long k, mpz_t nck) {
char* nck_str;
nck_str = mpz_get_str(NULL, 10, nck);
printf("%12lu\t%s\n", k, nck_str);
free(nck_str);
}
int
main(void)
{
unsigned long n, k;
mpz_t nck;
printf("n = ");
scanf("%lu", &n);
printf(" k nCk\n");
k = 0;
mpz_init_set_ui(nck, 1UL);
print_nck(k, nck);
for (k = 1; k <= n; k++) {
mpz_mul_ui(nck, nck, (n - k + 1));
mpz_fdiv_q_ui(nck, nck, k);
print_nck(k, nck);
}
return 0;
}
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
cshの文字列操作(0埋め)
-
C言語のソースをC++言語に変換...
-
一番大きい奇数を表示する
-
C言語での数字の花形表示
-
switch分のケースを範囲数?に...
-
コンパイルエラーについて
-
コマンドラインに出力した文字...
-
C言語です このプログラミング...
-
strcmp
-
現在時刻の表示について
-
C言語のプログラミングについて...
-
文字と数字の判定について
-
%P と %X の違い
-
srand(time(NULL))の使い方
-
int型 00 を表示するのに0とな...
-
値を変数に代入してprintfで表...
-
文字を動かしたい
-
Aの値からBの値を除するとは??
-
信頼区間の1.96や1.65ってどこ...
-
20'(角度)の計算がわかりま...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
10個出力で改行したいのですが...
-
コンパイルエラーについて
-
c言語でAからZまでを表示する...
-
コマンドプロンプトがすぐ消える
-
C言語の勉強しています。すみま...
-
テキストカーソル位置の取得
-
【C言語教えてください】sin波...
-
コマンドラインに出力した文字...
-
三角形の判別
-
(C言語)めちゃくちゃな値にな...
-
cshの文字列操作(0埋め)
-
scanfに文字が入力されたときに...
-
入力したお金の金額からお札の...
-
unsigned int型について
-
8人分のテストの点数を入力し、...
-
プログラミングについて質問で...
-
LU分解法のピボット選択機能実...
-
勝率をプログラムに
-
CTRL+Dでループを抜けるには
-
C言語
おすすめ情報