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

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

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

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

アルゴリズム

アルゴリズムとは、定められた目的を達成するために、プログラムの理論的な動作を定義するものです。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

2回答

7041閲覧

大きな数字でもコンビネーションの計算を速く求めるには?

manman

総合スコア233

Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

アルゴリズム

アルゴリズムとは、定められた目的を達成するために、プログラムの理論的な動作を定義するものです。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

2クリップ

投稿2015/05/03 07:03

編集2015/05/05 12:19

以下の工夫なしのコードでは、コンビネーションの計算をするのに
1分以上かかっています。

lang

1def ncr(n, r) 2 r = n - r if r > n - r 3 return 0 if r < 0 4 return 1 if r == 0 5 return (n - r + 1..n).inject(:*) / (1..r).inject(:*) 6end 7 8N = 1000 9cnt = 0 10(0..N).each{|a| 11 (0..a).each{|b| 12 cnt += ncr(a, b) 13 } 14} 15p cnt == 2 ** (N + 1) - 1

大きな数字でもコンビネーションの計算を
速く求められるようにするには
ncr をどのように定めればよろしいのでしょうか?

(追記)
ちなみに、Python版(工夫なし)では同じ計算を1分以内で行います。

lang

1from math import factorial 2 3def product(iterable): 4 prod = 1 5 for n in iterable: 6 prod *= n 7 return prod 8 9def npr(n, r): 10 return product(range(n - r + 1, n + 1)) 11 12def ncr(n, r): 13 if r > n // 2: 14 r = n - r 15 return npr(n, r) // factorial(r) 16 17N = 1000 18cnt = 0 19for a in xrange(N + 1): 20 for b in xrange(a + 1): 21 cnt += ncr(a, b) 22 23print cnt == 2 ** (N + 1) - 1

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

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

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

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

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

guest

回答2

0

入山徳夫氏によるnCrを高速に求めるアルゴリズム(追記:リンク先変更)
人間が手計算でnCrを計算するとき、約分して分母を消して、残った分子を掛けますが、
それをそのまんま再現したようなアルゴリズムです。

もしくはパスカルの三角形を使って加算だけで求める手もあります。

どっちが速いかは知らないので早かった方をお使い下さい

あと実装は自分でやって下さい。
リンク先をRubyにただ翻訳すればいいと思います。

投稿2015/05/03 07:54

編集2015/05/05 01:48
ozwk

総合スコア13521

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

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

manman

2015/05/04 18:45

入山徳夫氏によるnCrを高速に求めるアルゴリズム を試しましたが、nが4以上で誤った値をかえします。
ozwk

2015/05/05 01:55

リンク先の実装が間違えてますね。申し訳ありません。 リンク先差し替えてついでに検算もしておきました。
guest

0

自己解決

入山徳夫氏によるnCrを高速に求めるアルゴリズムの
Ruby版およびPython版です。
(Ruby版およびPython版どちらの場合も、)
質問の例だと残念ながら工夫なしの方が速いのですが、
N が大きいときの計算をさせると工夫なしより速くなりました。

lang

1def ncr(n, r) 2 r = [r, n - r].min 3 return 1 if r == 0 4 return n if r == 1 5 numerator = (n - r + 1..n).to_a 6 denominator = (1..r).to_a 7 (2..r).each{|p| 8 pivot = denominator[p - 1] 9 if pivot > 1 10 offset = (n - r) % p 11 (p - 1).step(r - 1, p){|k| 12 numerator[k - offset] /= pivot 13 denominator[k] /= pivot 14 } 15 end 16 } 17 result = 1 18 (0..r - 1).each{|k| 19 result *= numerator[k] if numerator[k] > 1 20 } 21 return result 22end 23 24N = 10000 25cnt = 0 26(0..N).each{|a| 27 cnt += ncr(N, a) 28} 29p cnt == 2 ** N

lang

1def ncr(n, r): 2 r = min(r, n - r) 3 if r == 0: return 1; 4 if r == 1: return n; 5 numerator = [n - r + i + 1 for i in range(r)] 6 denominator = [i + 1 for i in range(r)] 7 for p in range(2, r + 1): 8 pivot = denominator[p - 1] 9 if pivot > 1: 10 offset = (n - r) % p 11 for k in range(p - 1, r, p): 12 numerator[k - offset] /= pivot 13 denominator[k] /= pivot 14 result = 1 15 for k in range(r): 16 if numerator[k] > 1: 17 result *= numerator[k] 18 return result 19 20N = 10000 21cnt = 0 22for a in xrange(N + 1): 23 cnt += ncr(N, a) 24 25print cnt == 2 ** N

投稿2015/05/05 09:58

編集2015/12/31 20:37
manman

総合スコア233

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問