有 n 个有信仰的人,信仰共m 种,每个人只有一种信仰,当两个相同信仰的人遇到一起,他们便会逃走,求有多少中排列方式使得可能有人逃走, 即存在两个相同信仰的人相邻,答案对100003 取模。
第一种:首先我我们考虑 f [ i ] 代表前i 个连续的人可能逃走的可能方案。当前面 i - 1 个人中已经会有人逃跑了,第 i 个人的信仰就无关紧要,可以是 m 中的任意一种,则 f [ i ] = f [ i - 1 ] * m;当前面i - 1 个人中没有要越狱的,那么第 i 个人只能与前一个人的信仰相同,那么也就是等于前 i - 1 个人没有相邻的情况 = 总的 - 相邻的,则 f [ i ] = m ^ ( i - 1 ) - f [ i - 1]。第二种:设 f [ i ] 代表前 i 个连续的人不逃走的可能方案数,那么可能的方案数 = 总方案 ( m ^ n ) - f [ n ]。当前面i - 1 个人中已经会有人要逃跑了,第 i 个人加进去也会发生越狱,那么情况为 0;当前面 i - 1 个人中没有要越狱的,那么第 i 个人不能和前一个人的信仰相同,否则就会逃跑,就有 m - 1 种选择,则 f [ i ] = f [ i - 1 ] * ( m - 1 )。f [ 1 ] = m,第一个人可以任意选择。
显然第二种比较好,快速幂一下。
#include <cstdio>
using namespace std;
const int mod = 100003;
typedef long long LL;
LL m, n;
LL pom(LL m, LL n)
{
LL ret = 1;
while(n){
if(n & 1) ret = (ret * m) % mod;
m = (m * m) % mod;
n >>= 1;
}
return ret;
}
int main()
{
scanf("%lld%lld", &m, &n);
LL ans = pom(m, n) % mod;
ans = (ans - m * pom(m - 1, n - 1)) % mod;
ans += mod; ans %= mod;
printf("%lld\n", ans);
return 0;
}