CF 398 E(动态规划)
传送门:
http://codeforces.com/problemset/problem/398/E
题解:
首先答案不超过2。
最长环=1时,ans=0
最长环=2时,ans=1
否则,ans=2
考虑有长度大于2的环时如何两步出解。
那么第一步肯定是把大环拆成若干长度不超过2的环。
不妨确定一个x,设它指向y,指向它的是z,那么肯定将y、z交换,这样x、y在一个环里,然后剩下一个len-2的环,不过因为z不能再动了,所以对这个环的拆分就唯一了,一直下去可以把环拆开,并且只考虑这个环的方案数是len。
有多个环时,这些环的选择时可以相交的。
现在有两个环,
一定有一步交换位于不同环上的两个点,如果依然想拆成若干长度不超过2的环,那么剩下的交换也是唯一的。
由于对称问题,也只有len种。
显然两个环的时候必须长度相等才有解。
然后我并不会证更多环没有解。
然后就可以设个\(f[i][j]\)表示长度为i的有j个环的方案数
\(f[i][j]=f[i][j-1]*i+f[i][j-2]*(j-1)*i\),复杂度是调和级数
那么接下来\(O(k!)\)暴力的话也TLE了。
考虑确定每个点就是把若干条链拼起来,那么就只用集合划分了。
Code:
#include<bits/stdc++.h>
#define fo(i, x, y) for(int i = x, B = y; i <= B; i ++)
#define ff(i, x, y) for(int i = x, B = y; i < B; i ++)
#define fd(i, x, y) for(int i = x, B = y; i >= B; i --)
#define ll long long
#define pp printf
#define hh pp("\n")
using namespace std;
const int mo = 1e9 + 7;
ll ksm(ll x, ll y) {
ll s = 1;
for(; y; y /= 2, x = x * x % mo)
if(y & 1) s = s * x % mo;
return s;
}
const int N = 1e6 + 5;
int n, k, a[N], r[N], q[N];
ll fac[15];
vector<ll> f[N], nf[N];
int b[N], b0, cnt[N];
int c[N], c0, d[N];
ll ans, sum, s2;
void dg(int x) {
if(x > b0) {
ll s = sum, s3 = s2;
fo(i, 1, c0) {
s3 += c[i] > 2;
s = s * nf[c[i]][cnt[c[i]]] % mo;
cnt[c[i]] ++;
s = s * f[c[i]][cnt[c[i]]] % mo;
}
ll xs = 1;
fo(i, 1, c0) xs = xs * fac[d[i] - 1] % mo;
ans = (ans + (s3 ? s : 1) * xs) % mo;
fo(i, 1, c0) cnt[c[i]] --;
return;
}
fo(i, 1, c0) {
c[i] += b[x];
d[i] ++;
dg(x + 1);
d[i] --;
c[i] -= b[x];
}
c[++ c0] = b[x]; d[c0] = 1;
dg(x + 1);
d[c0] = 0; c0 --;
}
int main() {
freopen("determination.in", "r", stdin);
freopen("determination.out", "w", stdout);
fac[0] = 1; fo(i, 1, 15) fac[i] = fac[i - 1] * i % mo;
scanf("%d %d", &n, &k);
fo(i, 1, n) scanf("%d", &a[i]), r[a[i]] ++;
fo(i, 1, n) {
f[i].resize(n / i + 1);
nf[i].resize(n / i + 1);
f[i][0] = 1;
fo(j, 1, n / i) {
f[i][j] = f[i][j - 1];
if(j >= 2) f[i][j] = (f[i][j] + f[i][j - 2] * (j - 1)) % mo;
f[i][j] = f[i][j] * i % mo;
}
ll s = 1;
fo(j, 1, n / i) s = s * f[i][j] % mo;
s = ksm(s, mo - 2);
fd(j, n / i, 0) nf[i][j] = s, s = s * f[i][j] % mo;
s = 1;
fo(j, 1, n / i) {
nf[i][j] = nf[i][j] * s % mo;
s = s * f[i][j] % mo;
}
}
fo(i, 1, n) if(!r[i]) {
int x = i; q[x] = 1;
b[++ b0] = 0;
do {
b[b0] ++;
x = a[x];
q[x] = 1;
} while(x != 0);
}
fo(i, 1, n) if(!q[i]) {
int x = i, len = 0;
do {
len ++;
x = a[x];
q[x] = 1;
} while(x != i);
cnt[len] ++;
}
sum = 1;
fo(i, 1, n) sum = sum * f[i][cnt[i]] % mo, s2 += cnt[i] * (i > 2);
dg(1);
pp("%lld", ans);
}
CF 398 E(动态规划)的更多相关文章
- CF 848E(动态规划+分治NTT)
传送门: http://codeforces.com/problemset/problem/848/E 题解: 假设0-n一定有一条边,我们得到了一个方案,那么显然是可以旋转得到其他方案的. 记最大的 ...
- 四角递推(CF Working out,动态规划递推)
题目:假如有A,B两个人,在一个m*n的矩阵,然后A在(1,1),B在(m,1),A要走到(m,n),B要走到(1,n),两人走的过程中可以捡起格子上的数字,而且两人速度不一样,可以同时到一个点(哪怕 ...
- CF 1096D Easy Problem [动态规划]
题目链接:http://codeforces.com/problemset/problem/1096/D 题意: 有一长度为n的字符串,每一字符都有一个权值,要求现在从中取出若干个字符,使得字符串中没 ...
- CF思维联系–CodeForces - 225C. Barcode(二路动态规划)
ACM思维题训练集合 Desciption You've got an n × m pixel picture. Each pixel can be white or black. Your task ...
- CF 494 F. Abbreviation(动态规划)
题目链接:[http://codeforces.com/contest/1003/problem/F] 题意:给出一个n字符串,这些字符串按顺序组成一个文本,字符串之间用空格隔开,文本的大小是字母+空 ...
- CF 414B Mashmokh and ACM 动态规划
题意: 给你两个数n和k.求满足以下条件的数列有多少个. 这个数列的长度是k: b[1], b[2], ……, b[k]. 并且 b[1] <= b[2] <= …… <= b[k] ...
- CF思维联系– Codeforces-987C - Three displays ( 动态规划)
ACM思维题训练集合 It is the middle of 2018 and Maria Stepanovna, who lives outside Krasnokamensk (a town in ...
- 【DP专辑】ACM动态规划总结
转载请注明出处,谢谢. http://blog.csdn.net/cc_again?viewmode=list ---------- Accagain 2014年5月15日 ...
- 【CF932G】Palindrome Partition(回文树,动态规划)
[CF932G]Palindrome Partition(回文树,动态规划) 题面 CF 翻译: 给定一个串,把串分为偶数段 假设分为了\(s1,s2,s3....sk\) 求,满足\(s_1=s_k ...
随机推荐
- MySQL 复制参数详解
log-bin 二进制日志 server-id 早起版本必须添加 1-pow(2,32)-1 推荐使用 端口号+ip最后一位 5.6后可以动态修改 server-uuid (5.6以后) 默认存在 ...
- C# System.Windows.Forms.Panel
UserControl 定义的界面 输出到panel 实现界面切换
- JavaScript性能优化之摇树
作者|Jeremy Wagner译者|薛命灯 现代 Web 应用程序可能会变得非常巨大,特别是它们的 JavaScript 部分.HTTP Archive 网站的数据显示,截至 2018 年中,传输到 ...
- Qt:代码里存在中文时带来的问题
一.报错: 常量中有换行符 方法1: 把文本文件转化为unicode或者utf-8, 同是还要带上QString::fromLocal8Bit() 还有其他方法,感觉不靠谱 二.显示异常:乱码 QSt ...
- 记录我个人对Telegram的了解
对Telegram(电报) 开始的了解是以为提供了Telegram API,就可以基于它进行开发自己的即时通讯(Instant Messaging)程序. 大概使用过: webogram 和 tele ...
- Thread.Join理解
Thread.Join:Blocks the calling thread until a thread terminates MainThread里面起了一个SubThread,从SubThread ...
- Delphi 鼠标控制函数GetCursorPos、SetCursorPos
GetCursorPos函数 获取鼠标的位置 var P: TPoint; begin GetCursorPos(P); //获取鼠标位置 end; SetCursorPos函数 设置鼠标的位置 v ...
- 【索引】位图BitMap索引
位图(BitMap)索引 前段时间听同事分享,偶尔讲起Oracle数据库的位图索引,顿时大感兴趣.说来惭愧,在这之前对位图索引一无所知,因此趁此机会写篇博文介绍下位图索引. 1. 案例 有张表名为ta ...
- 线段树双tag+差分数组——cf1208E
写了一上午 /* 对于每个数组a[],先排序然后从大到小把a[i]放进线段树更新 设a[i]的位置是pos,那么其可更新的区间是[pos,w-(li-pos)] 线段树结点保存 tag=now表示当前 ...
- 可持化永久树 的 STL ( rope )
rope 的基本操作 #include <ext/rope> using namespace __gnu_cxx; ]; rope<int> x; rope<int> ...