アプリ版:「スタンプのみでお礼する」機能のリリースについて

c言語のmまたはnが13以上となる場合に正しい解を求めることができない、なぜなら、13の階乗は6,227,020,800であり、この値はint型変数で扱うことのできる範囲をこえてしまっているからである。以下のプログラムを改良しなるべく大きなmやnの値でも正しく解を求めることができるプログラムをおしえてください。しかし、変数はあくまでint型を用い、floatやdouble型は使用しない方法でお願いします
int add( int a, int b )
{
int i;
int d = ( b>0 ? 1 : -1 );
int n = ( b>0 ? b : -b );
for( i=0; i<n; ++i )
{ a += d; }

return a;
}

int mul( int a, int b )
{
int i;
int r = 0;
int n = ( b>0 ? b : -b );
for( i=0; i<n; ++i )
{ r = add( r, a ); }

return ( b>0 ? r : -r );
}

int fn( int kitten )
{
return ( kitten>1 ? mul( kitten, fn(kitten-1) ) : 1 );
}

A 回答 (2件)

log_2(13!)≒32.54なので、 intが33bit以上ある処理系を使えば解決。


intが64bitなら、20!まで正しく求められます。

C言語の規格では、 intのビット幅は決まっていません。
あなたが使っているものが、たまたま int=32bitなだけです。


これって、 関数fn が、 階乗を求める関数ですよね?
ということは、int型を返す 関数fn では、どんなに中で正しく計算できても、 fn()は 32bitまでしか正しく返すことはできません。
つまり、intをやめない限り、 fn(13) は正しく求められません。


どこにも m ,n が出てきていませんが、もしかして順列 (nPm) または組合せ(nCm) を求めるために、階乗を使っているのでしょうか?
それなら、階乗を使った公式ではなく、別の計算方法を使います。

nPm なら n・(n-1)・…・'(n-m+1) を計算する。
nCm なら、 分子と分母に分けて、オーバーフローしないように掛けて、約分して既約分数にしてを繰り返す。

それでも、組合せの性質上、nは大して大きくできませんが。
    • good
    • 4

質問の意図に反しますか、double型は2^53までの整数を、誤差無しで


保持出来ることは知っておいた方が良いでしょう。

多倍長演算は随分色々作りましたが、ここでちょろっと教えられるようなコード量
ではないです。検索しましょう。
Javaやpythonなら、何も作らず利用出来ます。
    • good
    • 0
この回答へのお礼

ありがとうございます

お礼日時:2017/07/07 23:19

お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!