想了很久觉得自己做法肯定T啊,就算是CF机子的3s时限,但我毕竟是 O ( C(15,7)*7!*log ) ....

果然在n=15的点T了...贱兮兮地特判了15过掉了,结果发现题解说就是打表...

(卒,享年16岁)

总之啊总之,要灵活啊

回归机房以后发现Div2的D都要想一个世纪了......不过感觉这题真的挺好的?

命不久矣,所以想最后写写题解吧。

题目大意:求满足  a+b=c( a,b,c均为长度为n的排列)的有序对{a,b}的对数

(说明:a,b,c均为0~(n-1)的排列  +运算为题中定义的+的等价修改  即ci=(ai+bi)mod n )

∆初步分析,有 2*n*(n-1)/2 = n*k + n*(n-1)/2 (k为整数) 则有 k=(n-1)/2 则n必须为奇数 否则答案为0

∆显然,{a,c}和{a,b}是一一对应的,所以我们不妨求{a,c}的有序对数(实际上这个转换并没有什么用处,只是我感觉爽....)

∆进一步化简,不妨将a固定为{0,1,2...n-1},则最终所求方案数* n! 即为答案。

∆考虑c需要满足的条件:

  (1)i-ci=j-cj(mod n) 对于任意 i!=j 均不成立

  (2)ci为0~n-1 且互不相同

(2)较容易解决,记录出现的数字即可。考虑如何满足(1)。想象你是在玩一个类似数独的游戏,在一个一个填数字,那么你只要保证:1.已经填上的数字彼此之间满足(1).(2)  2.新填入的第i+1个数字和前i个数字不重复、不冲突。

由此,我们可以像填数独一样,列出下一个格子不能取的值。那么我们就有一个大致思路,即状压,记录两维状态v1,v2(即由条件(1)、(2),下一位不能取的数字有哪些)。转移只需枚举下一位的可行数字x,在v1并入x,在v2并入x并循环右移1位( ci !=( val=cj-j+i )   ci+1 != (val+1=cj-j+i+1) ) 。

∆然鹅,这样复杂度很高(上界可达 O(n*22n) ?总之打表估计都GG),因此我们考虑进一步优化。

优化的本质就是探寻性质。我们发现,对于f[v1][v2]>0,v1和v2中的1个数一定相同,即在v2中并入数字时一定不会并入一个已经存在的位。可以用反证法证明,若并入ci对ci+1的影响时,该位已有cj对ci+1的影响,那么i、j两位就一定不符合(2)。

也就是说,v1,v2是可以「拆分」的。

∆「拆分」具体是什么意思呢?我们来模拟一下。

假设n=7,当前已经放了4个数:

v1=1100110

v2=1001011

现在我们挨个放入剩下的数字。例如:

v1=1120110

v2=0210111

v1=1123110

v2=2131110

v1=1123114

v2=1311142

将末状态和初状态做差,得到:

v1=0023004

v2=0300042

v1=0011001

v2=0100011

我们发现 对于初状态的{v1,v2} 差状态即为{1111111^v1, turn(1111111^v1,3) }

(turn(j,ct)表示将j循环右移ct位)

∆于是 我们就可以开心地折半搜索了!复杂度见第一行

(以下是打表代码,如果直接复制会在n=15的点TLE)

 #include <bits/stdc++.h>
using namespace std; #define rep(i,l,r) for(int i=l;i<=r;++i)
#define per(i,r,l) for(int i=r;i>=l;--i)
#define mp make_pair
#define fir first
#define sec second typedef long long ll;
typedef pair<int,int> pii; const int p=,V=1e6+; int n,full,semi,turn[V];
map<pii,int>f[];
map<pii,int>::iterator it; int rev(int j){//j循环右移1位
int ans=;
if(j>=semi){
ans=;
j-=semi;
} return ans|(j<<);
} int main(){
scanf("%d",&n); if(n==){
printf("");
return ;
} if(n%==){
printf("");
return ;
} full=(<<n)-;
semi=(<<(n-)); int u=(n>>); f[][mp(,)]=; //多出的一维[0/1]是在愚蠢地省空间...
bool oi=;
rep(o,,u){
f[!oi].clear(); for(it=f[oi].begin();it!=f[oi].end();++it){
pii P=it->fir;int val=it->sec;
int L=P.fir,R=P.sec; int j=(L|R);
if(j==full) continue; rep(i,,n-) if( ( (<<i)&L ) == && ( (<<i)&R ) == )
(f[!oi][mp( ( L|(<<i) ) , rev( R|(<<i) ) )]+=val)%=p; } oi=!oi;
} int pre=(<<(u+))-,suf=full^pre;
rep(i,,full) turn[i]=((i&pre)<<u)|((i&suf)>>(u+));
//turn[i]即i循环右移(n/2)位 ll ans=;
for(it=f[oi].begin();it!=f[oi].end();++it){
pii P=it->fir;ll val=it->sec;
int L=P.fir,R=P.sec; (ans+=val*f[!oi][mp(full^L,turn[full^R])]%p)%=p;
} rep(i,,n) (ans*=i)%=p; printf("%lld",ans); return ;
}

CF285D.Permutation Sum的更多相关文章

  1. CF285D.D. Permutation Sum

    CF285D. Permutation Sum 题目 大意 寻找a,b两个排列从0到n-1,有c[i]=(a[i]+b[i])%n+1,使得c[i]也为全排列的排列方式 思路 a中元素和b中元素的对应 ...

  2. codeforces 285 D. Permutation Sum 状压 dfs打表

    题意: 如果有2个排列a,b,定义序列c为: c[i] = (a[i] + b[i] - 2) % n + 1 但是,明显c不一定是一个排列 现在,给出排列的长度n (1 <= n <= ...

  3. SPOJ:Elegant Permuted Sum(贪心)

    Special Thanks: Jane Alam Jan*At moment in University of Texas at San Antonio - USA You will be give ...

  4. Codeforces Round #175 (Div. 2)

    A. Slightly Decreasing Permutations 后\(k\)个倒序放前面,前\(n-k\)个顺序放后面. B. Find Marble 模拟. C. Building Perm ...

  5. Codeforces Round #175 (Div. 2) A~D 题解

    A.Slightly Decreasing Permutations Permutation p is an ordered set of integers p1,  p2,  ...,  pn, c ...

  6. combination sum、permutation、subset(组合和、全排列、子集)

    combination sum I.permutation I.subsets  I 是组合和.全排列.子集的第一种情况,给定数组中没有重复的元素. combination sum II.permut ...

  7. UVA11525 Permutation[康托展开 树状数组求第k小值]

    UVA - 11525 Permutation 题意:输出1~n的所有排列,字典序大小第∑k1Si∗(K−i)!个 学了好多知识 1.康托展开 X=a[n]*(n-1)!+a[n-1]*(n-2)!+ ...

  8. 266. Palindrome Permutation

    题目: Given a string, determine if a permutation of the string could form a palindrome. For example,&q ...

  9. Educational Codeforces Round 7 D. Optimal Number Permutation 构造题

    D. Optimal Number Permutation 题目连接: http://www.codeforces.com/contest/622/problem/D Description You ...

随机推荐

  1. 第一次实验报告x

    C程序设计实验报告 实验项目:2.3.3字符与ASCII码,2.3.4运算符的表达式与应用,2.3.5顺序结构应用程序,3.3.1数学函数的算法描述,3.3.2鸡兔同笼的算法描述,3.3.3确定坐标的 ...

  2. 从SQLServer转储数据到MySQL

    前一段时间,由于项目需要将数据库从SQLServer迁移到MySQL,在网上百度了很久,基本都是通过SQLyog实现的.其实使用平时常用的数据库管理软件Navicat Premium也能做到,并且操作 ...

  3. left join on 后and 和 where 的区别

    SELECT * FROM student a LEFT JOIN sc b ON a.Sid = b.Sid AND a.Sname="赵雷" 结果:(left join 左连接 ...

  4. ios调用Google地图

    现在的ios版本一般只支持https协议,而引用谷歌地图API时只提供src="http://maps.google.cn/maps/api/js..",https协议无法使用,解 ...

  5. Sublime Text3 调色板 ColorPicker插件安装及快捷键

    一.安装 第一步:打开菜单栏下的tools>command palette或是快捷键ctrl+shift+p输入PI 点击第一个安装包等待跳出窗口,输入ColorPicker,待安装完成 第二步 ...

  6. 【SoftwareTesting】Homework3

    (a) (b) 数组越界问题 (c) n=0 (d) 点覆盖:[1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16] 边覆盖:[(1,2),(2,3),(3,4),(4,5) ...

  7. Problem 2: Even Fibonacci numbers

    Each new term in the Fibonacci sequence is generated by adding the previous two terms. By starting w ...

  8. 初读"Thinking in Java"读书笔记之第七章 --- 复用类

    组合语法 将对象引用置于新类中,即形成类的组合. 引用初始化方法 在定义处初始化. 在类的构造器中初始化. 在使用这些对象之前,进行"惰性初始化". 使用实例初始化. 继承语法 J ...

  9. `TypeError: torch.mm received an invalid combination of arguments - got (torch.FloatTensor, Variable),

    `TypeError: torch.mm received an invalid combination of arguments - got (torch.FloatTensor, Variable ...

  10. python+appium 自动化2--元素定位uiautomatorviewer

    出处:https://www.cnblogs.com/yoyoketang/p/6128741.html 前言: 可以打开手机上的app了,下一步元素定位uiautomatorviewer,通过定位到 ...