HDU 4332 Contest 4
顶好的一道题。其实,是POJ 2411的升级版。但POJ 2411我用的插头DP来做,一时没想到那道题怎么用状态DP,于是回头看POJ 2411那一道的状态DP,其实也很简单,就是每一行都设一个状态,用位来表示,如果上一行为0,则当前行必定是要竖着放的。填1.否则,当前行的位置可以横放填两个格子为1,也可以不放,为0.于是,看上一行的状态能转移到哪些状态,就可以了。
这一道也是一样的想做法,DFS看开始时有哪些状态,记在一个一维矩阵里。因为有一些状态最开始是达不到的,如奇数个1,或者一些不连续1的状态。然后求转移矩阵(这个地方想了很久才想到,因为要做很多重复的工作,所以想到用矩阵)。求矩阵时,也用DFS就可以了。
此处在矩阵相乘时有一个小优化,因为转移矩阵中含0较多,所以把中间变量K提到第一个循环。当含有0时就终止。因为这个地方由TLE,优化到了8000MS,也是满足了。
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#define LL __int64
#define mod 1000000007
using namespace std; struct Matrix{
LL ans[130][130];
};
Matrix pp[31]; LL bgn[130];
LL res[130];
int only;
int number[260];
Matrix transfer;
Matrix per;
bool ok(int s){
int c=0;
for(int i=0;i<8;i++)
if(s&(1<<i)) c++;
if(c%2==0) return true;
return false;
} Matrix operator *(Matrix a,Matrix b){
Matrix c;
for(int i=0;i<only;i++)
for(int j=0;j<only;j++)
c.ans[i][j]=0;
for(int k=0;k<only;k++) //把k提到这里来就过了。。。哈哈,这也是一种优化啊,因为矩阵中0较多,避免了不必要的计算。
for(int i=0;i<only;i++){
if(a.ans[i][k]==0) continue;
for(int j=0;j<only;j++){
c.ans[i][j]=(c.ans[i][j]+(a.ans[i][k]*b.ans[k][j])%mod)%mod;
}
}
return c;
} void find_bgn(int state,int pos){
if(pos>=8) {
bgn[number[state]]++;
return ;
}
int tmp=state;
int i=pos;
if((state&(1<<i))==0&&(state&(1<<((i+1)%8)))==0){
tmp=state|(1<<i)|(1<<((i+1)%8));
find_bgn(tmp,i+2);
}
find_bgn(state,i+1);
} void dfs(int pre,int state,int pos){
if(pos>=8) {
transfer.ans[number[pre]][number[state]]++;
return ;
}
int tmp=state;
int i=pos;
if((state&(1<<i))==0&&(state&(1<<((i+1)%8)))==0){
tmp=state|(1<<i)|(1<<((i+1)%8));
dfs(pre,tmp,i+2);
}
dfs(pre,state,i+1);
} /*
LL can(int a,int b)
{
int st,i,j;
bool end=false;
if(a==0)return b==255? 1LL:0;
for(i=0;i<8;i++)
{
j=(i+7)%8;
if((a&(1<<i))&&(a&(1<<j))==0)
{
st=i;
break;
}
}
for(i=st;i!=st||end==false;i=(i+1)%8)
{
end=true;
j=(i+1)%8;
if(0==(a&(1<<i)))
{
if(0==(b&(1<<i)))return 0;
}
else
{
if((a&(1<<j))&&(b&(1<<i))&&(b&(1<<j)))i++;
else if((b&(1<<i))==0)continue;
else return 0;
}
}
return 1LL;
}
*/
void initial(){
only=0;
memset(number,-1,sizeof(number));
memset(transfer.ans,0,sizeof(transfer.ans));
for(int i=0;i<(1<<8);i++)
if(ok(i)) number[i]=only++; for(int i=0;i<only;i++)
for(int j=0;j<only;j++)
if(i==j) per.ans[i][i]=1;
else per.ans[i][j]=0; memset(bgn,0,sizeof(bgn));
find_bgn(0,0); for(int i=0;i<(1<<8);i++){
if(number[i]!=-1){
dfs(i,255-i,0);
}
}
/*
for(int i=0;i<1<<8;i++)
for(int j=i+1;j<1<<8;j++)
if(number[i]!=-1&&number[j]!=-1)
transfer.ans[number[j]][number[i]]=transfer.ans[number[i]][number[j]]=can(i,j);
transfer.ans[only-1][only-1]=2;
*/
for(int i=1;i<31;i++){
if(i==1)
pp[i]=transfer;
else
pp[i]=pp[i-1]*pp[i-1];
}
} LL pow(int k){
Matrix ans=per,ats=transfer;
int c=1;
while(k){
if(k&1) ans=ans*pp[c];
k>>=1;
c++;
}
LL tmp=0;
for(int j=0;j<only;j++)
tmp=(tmp+(bgn[j]*ans.ans[j][only-1])%mod)%mod;
return tmp;
} int main(){
initial();
int n,T,kase=0;
scanf("%d",&T);
while(T--){
scanf("%d",&n);
LL resul=pow(n-1);
printf("Case %d: %I64d\n",++kase,resul);
}
return 0;
}
HDU 4332 Contest 4的更多相关文章
- HDU 5045 Contest(状压DP)
Problem Description In the ACM International Collegiate Programming Contest, each team consist of th ...
- hdu - 5045 - Contest(国家压缩dp)
意甲冠军:N个人M通过主打歌有自己的期望,每个问题发送人玩.它不能超过随机播放的次数1,追求最大业绩预期 (1 ≤ N ≤ 10,1 ≤ M ≤ 1000). 主题链接:pid=5045" ...
- [ACM] hdu 5045 Contest (减少国家Dp)
Contest Problem Description In the ACM International Collegiate Programming Contest, each team consi ...
- HDU–5988-Coding Contest(最小费用最大流变形)
Coding Contest Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)To ...
- hdu 5045 Contest(状态压缩DP)
题解:我们使用一个二位数组dp[i][j]记录进行到第i个任务时,人组合为j时的最大和(这里的j我们用二进制的每位相应一个人). 详细见代码: #include <iostream> #i ...
- HDU 5045 Contest
pid=5045">主题链接~~> 做题感悟:比赛时这题后来才写的,有点小尴尬.两个人商议着写写了非常久才写出来,I want to Powerful ,I believe me ...
- HDU 4335 Contest 4
利用降幂公式..呃,还是自己去搜题解吧.知道降幂公式后,就不难了. #include <iostream> #include <cstdio> #include <alg ...
- HDU 4339 Contest 4
树状数组,主要是抓住要求连续1的个数.这样,初始时,相同的加1,不同的加0. 查询时,用二分搜索右边界.就是比较当前mid-l+1的值与他们之间1的个数(这可以通过树状数组求区间和得出),记录右边界即 ...
- HDU 4334 Contest 4
本来以为是一道水题,好吧,做了才知道,出题的人有多牛.二分搜索是不可能的了,因为会超内存... 看到别人的搜索两个集合的提示,我就自己一边去想了.终于想出来了: 可以这样做,先把每两个集合的和值枚举出 ...
随机推荐
- [SharePoint2010开发入门经典]SPS2010列表编程
本章概要: 1.理解SPS2010列表的结构和功能 2.使用客户端和服务器端对象模型,web service,wcf和RESTful service进行列表编程. 3.理解方法的使用 4.类表编程
- HDU 4454
想了很久,发现其实就只需要三分枚举圆上的点,到矩形的最短很容易就可以求到了.开始时考虑要不要根据矩形相对圆的方位来划分枚举区间,后来发现一定不能这样做的. 注意题目给的是矩形的对角形,但没说哪一条对角 ...
- what we benefit from big data
大数据玩的是什么,趋势,故障,寿命? 物联网拉动的是终端厂商的销量,作为终端设备生产商(OEM).无论是汽车.手机.家电行业.最有理由推动物联网的普及,可是作为传统行业,玩"网"并 ...
- 朴素的UNIX之-调度器细节
0.多进程调度的本质 我们都知道UNIX上有一个著名的nice调用.何谓nice,当然是"好"了.常规的想法是nice值越大越好,实际上,nice值越好,自己的优先级越低.那么为何 ...
- MAME 0.201 发布,重温童年的街机模拟器
MAME 0.201 已发布,MAME 最初是街机模拟器,随着时间的推移,MAME 吸收了姊妹项目 MESS(多机种模拟器超级系统),所以 MAME 现在还记录了各种各样的(大多是老式的)电脑游戏.掌 ...
- C#将内容导出到Word到指定模板
昨天做了下导入导出Excel文件,今天研究了下导出Word文件. 从网上找了半天才找到了一个能导出到指定模板的,在这里总结下. 导出模板原理就是利用的替换占位符. 我这里先建立好了一个模板, 接下来写 ...
- Controller总结
下图显示了组建之间的基本控制流程 1.1控制器工厂.动作调用器 控制器工厂负责创建对请求进行服务的控制器实例 动作调用其负责查找并调用控制器类中的动作方法. 1.2自定义控制器工厂 namespace ...
- Excel基础视频教程在线观看
也许你已经在Excel中完成过上百张财务报表,也许你已利用Excel函数实现过上千次的复杂运算,也许你认为Excel也不过如此,甚至了无新意.但我们平日里无数次重复的得心应手的使用方法只不过是Exce ...
- Powerpivot PowerBI相关组件下载安装(附操作截图)
加载方式:com加载项加载方法: 点击Excel界面[文件]→[选项]→[加载项]→[COM加载项]→[转到] Excel2013加载PowerView Excel216PowerQuery不需要加载 ...
- ZBrush中如何反选遮罩
通过对ZBrush的学习,我们知道了如何手动创建遮罩,手动创建遮罩相对来说是最简单有效的方法,在某些特定的使用场合会起到事半功倍的效果.创建遮罩我们可以结合Ctrl键在物体保持编辑的状态下来执行,您可 ...