bzoj1008.越狱(排列组合 && 快速幂)

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;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值