この記事で使うアルゴリズム
逆元
はじめに
カテゴリー競プロ初中級者100問では、Qiita
にて@e869120
さんがレッドコーダーが教える、競プロ・AtCoder上達のガイドライン【中級編:目指せ水色コーダー!】としてまとめられている100問をPython
で解いています。
全問題の一覧はこちらです
問題
https://atcoder.jp/contests/abc034/tasks/abc034_c
この問題では逆元の計算が求められます。
方針
組み合わせの問題として解くことができます。 以下の値を計算すれば良いことになります。
((W-1)+(H-1))!/((W-1)!(H-1)!) % MOD
階乗の部分が巨大になるため、計算途中で剰余の計算を行います。しかしながら、除算が含まれるため逆元と呼ばれる概念を導入する必要があります。
逆元とは
a
とm
が互いに素である場合、m
を法としてax≡1
となるようなx
が存在します(必要十分条件)。これをmを法とするaの逆元
と言います。
逆元は複数存在し得ますが、x
に対してさらにm
を法とするとただ一つに定まります。この問題では合同式で計算が進むため、逆元は単一の整数と考えて問題はありません。
こちらを参考にさせていただきました。
フェルマーの小定理
p
を素数として、p
と互いに素な数a
についてa^(p-1)≡1
が成立します。これをフェルマーの小定理と呼びます。
逆元と組み合わせて考える
さて、a
とp
は互いに素ですので、逆元a^(-1)
を考えることができます。つまり、逆元をフェルマーの小定理を使うことで計算できます。a^(-1)≡a^(p-2)
冪剰余
さて次はa^(p-2) mod p
を求めれば良いですが、この問題ではp
は巨大であるため、先にa^(p-2)
を計算してから剰余を取ろうとすると、計算効率が著しく悪くなります。そこで、計算の最中に随時剰余の計算を行う必要があります。Python
の場合は、この処理をpow(a,p-2,p)
とすることで計算できます。
コード
1 | MOD = 10**9+7 |
記事情報
- 投稿日:2020年5月1日
- 最終更新日:2020年5月9日