BZOJ

洛谷

http://www.cnblogs.com/LadyLex/p/7326874.html

插头DP。\(m+1\)个插头的状态需要用三进制表示:\(0\)表示无插头,\(1\)表示是左括号插头,\(2\)表示是右括号插头。为了方便用两位的二进制写。所以还需要个哈希表存状态。

转移的时候,对于左边上边这两个插头,如果某个插头为\(0\),很好转移。否则就分\(4\)种情况讨论下。不写了。。见上面的链接。

还需要高精度。其它就是些细节了。

转移时特判下边界外有插头就不转移,会方便很多。

实际方案数还不是特别多,高精可以用两个long long 实现。

因为状态也不多,线性探测的效率比链式哈希高很多。。

还有别忘把答案\(*2\)(反向走)。。

优化后:

//944kb	116ms
#include <cstdio>
#include <cstring>
#include <algorithm>
typedef long long LL;
const LL Base=(LL)1e18; int n,m;
struct BigInt
{
LL a,b;
BigInt() {a=0, b=0;}
inline void Init() {a=0, b=0;}
inline void Set(int x) {a=x, b=0;}
inline void operator =(const int x) {Set(x);}
inline void operator +=(const BigInt &x)//不能写 ++b,b+=x.b 因为是&...
{
a+=x.a, a>=Base?(b+=x.b+1,a-=Base):b+=x.b;
}
inline void Print()
{
b ? printf("%lld%018lld",b,a) : printf("%lld\n",a);
putchar('\n');
}
}Ans;
struct Hash_Table
{
#define mod 2501//状态数最多大概2200?
int top,sk[mod],hash[mod];
BigInt val[mod];
Hash_Table() {top=0, memset(hash,0xff,sizeof hash);}
inline void Init() {while(top) hash[sk[top--]]=-1;}
inline BigInt& operator [](const int s)
{//s可能是0
for(int x=s%mod; ; x=x+1==mod?0:x+1)
{
if(!~hash[x]) hash[x]=s, sk[++top]=x, val[x].Init();
if(hash[x]==s) return val[x];
}
}
}f[2]; inline int Get(const int s,const int bit) {return s>>(bit<<1)&3;}//s在bit位置的插头
inline void Upd(int &s,int bit,int v) {bit<<=1, s|=3<<bit, s^=3<<bit, s|=v<<bit;}//将s在bit位置的插头改为v
inline int Find(const int s,const int y,const int p)//找到与y位置的插头p 所对应的插头
{
int delta=p==1?1:-1/*向左/向右找*/,sum=delta;//if(!p) return y;
for(int i=y+delta,v; ~i&&i<=m; i+=delta)//i=0~m-1!
if(v=Get(s,i),sum+=v==2?-1:v==1,!sum) return i;
return -1;
}
void Work(const int n,const int m,const int x,const int y,Hash_Table &f,Hash_Table &g)
{
f.Init();
for(int i=1,tot=g.top; i<=tot; ++i)
{
int id=g.sk[i],s=g.hash[id],p1=Get(s,y-1),p2=Get(s,y),t1=p1?Find(s,y-1,p1):0,t2=p2?Find(s,y,p2):0;
if(t1==-1||t2==-1) continue;//
BigInt v=g.val[id];
if(!v.a) continue;
if(!p1&&!p2) {if(x!=n&&y!=m) Upd(s,y-1,1), Upd(s,y,2), f[s]+=v;}
else if(!p1&&p2)
{
if(y!=m) f[s]+=v;
if(x!=n) Upd(s,y-1,p2), Upd(s,y,0), f[s]+=v;
}
else if(p1&&!p2)
{
if(x!=n) f[s]+=v;
if(y!=m) Upd(s,y-1,0), Upd(s,y,p1), f[s]+=v;
}
else if(p1==1&&p2==1) Upd(s,y-1,0), Upd(s,y,0), Upd(s,t2,1), f[s]+=v;
else if(p1==1&&p2==2) {if(x==n&&y==m) Ans+=v;}
else if(p2==1) Upd(s,y-1,0), Upd(s,y,0), f[s]+=v;
else if(p2==2) Upd(s,y-1,0), Upd(s,y,0), Upd(s,t1,2), f[s]+=v;
}
} int main()
{
int n,m; scanf("%d%d",&n,&m);
if(m>n) std::swap(n,m); ::n=n, ::m=m;
if(m==1) return puts("1"),0;//!
int p=0; f[p].Init(), f[p][0]=1;
for(int i=1; i<=n; ++i)
{
for(int j=1; j<=m; ++j) p^=1, Work(n,m,i,j,f[p],f[p^1]);
if(i!=n)
{
for(int i=1,tot=f[p].top; i<=tot; ++i)
f[p].hash[f[p].sk[i]]<<=2;//这样会多花额外时间找状态吧=-= 不管了反正是快
// f[p^1].Init();
// for(int i=1,tot=f[p].top,s; i<=tot; ++i)
// s=f[p].hash[f[p].sk[i]], f[p^1][s<<2]=f[p][s];
// p^=1;
}
}
Ans+=Ans, Ans.Print(); return 0;
}

优化前:

//154108kb	1396ms
#include <cstdio>
#include <cstring>
#include <algorithm>
#define Base 1000000000
typedef long long LL; int n,m;
struct BigInt
{
int a[5];
BigInt() {memset(a,0,sizeof a);}
inline void Init() {memset(a,0,sizeof a);}
inline void Set(int x) {a[0]=0; while(x) a[++a[0]]=x%Base, x/=Base;}
inline int& operator [](int x) {return a[x];}
inline BigInt operator +(const BigInt &x)const
{
BigInt res;
int l=std::max(a[0],x.a[0]);
for(int i=1; i<=l; ++i) res.a[i]+=a[i]+x.a[i], res[i]>=Base&&(++res.a[i+1]/*+=res.a[i]/Base*/, res.a[i]-=Base);
++l;
while(!res.a[l]) --l;
res.a[0]=l;
return res;
}
inline void operator =(const int x) {Set(x);}
inline void operator +=(const BigInt &x) {*this=*this+x;}
inline void Print()
{
printf("%d",a[a[0]]);
for(int i=a[0]-1; i>0; --i) printf("%09d",a[i]);
putchar('\n');
}
}Ans;
struct Hash_Table
{
#define mod 10007
#define N 2800000//1010101010101010101010 //4^{11}=4194304 不是3^{11}。。
int tot,top,sk[mod],H[mod],nxt[N],sta[N];
BigInt val[N];
Hash_Table() {tot=top=0;}
inline void AE(int u,int v) {nxt[v]=H[u], H[u]=v;}
inline void Init() {tot=0; while(top) H[sk[top--]]=0;}
inline BigInt& operator [](const int s)
{
int x=s%mod;//s可能是0,边表里的需要是s+1。也可以初始化H[x]=-1。
for(int i=H[x]; i; i=nxt[i])
if(i==s+1) return val[s];
if(!H[x]) sk[++top]=x;
AE(x,s+1), sta[++tot]=s, val[s].Init();
return val[s];
}
}f[2]; inline int Get(const int s,const int bit) {return s>>(bit<<1)&3;}//s在bit位置的插头
inline void Upd(int &s,int bit,int v) {bit<<=1, s|=3<<bit, s^=3<<bit, s|=v<<bit;}//将s在bit位置的插头改为v
inline int Find(const int s,const int y,const int p)//找到与y位置的插头p 所对应的插头
{
int delta=p==1?1:-1/*向左/向右找*/,sum=delta;//if(!p) return y;
for(int i=y+delta,v; ~i&&i<=m; i+=delta)//i=0~m-1!
if(v=Get(s,i),sum+=v==1?1:(v==2?-1:0),!sum) return i;
return -1;
}
void Work(const int n,const int m,const int x,const int y,Hash_Table &f,Hash_Table &g)
{
f.Init();
for(int i=1,tot=g.tot; i<=tot; ++i)
{
int s=g.sta[i],p1=Get(s,y-1),p2=Get(s,y),t1=p1?Find(s,y-1,p1):0,t2=p2?Find(s,y,p2):0;
if(t1==-1||t2==-1) continue;//
BigInt v=g.val[s];
if(!v[0]) continue;
if(!p1&&!p2) {if(x!=n&&y!=m) Upd(s,y-1,1), Upd(s,y,2), f[s]+=v;}
else if(!p1&&p2)
{
if(y!=m) f[s]+=v;
if(x!=n) Upd(s,y-1,p2), Upd(s,y,0), f[s]+=v;
}
else if(p1&&!p2)
{
if(x!=n) f[s]+=v;
if(y!=m) Upd(s,y-1,0), Upd(s,y,p1), f[s]+=v;
}
else if(p1==1&&p2==1) Upd(s,y-1,0), Upd(s,y,0), Upd(s,t2,1), f[s]+=v;
else if(p1==1&&p2==2) {if(x==n&&y==m) Ans+=v;}
else if(p2==1) Upd(s,y-1,0), Upd(s,y,0), f[s]+=v;
else if(p2==2) Upd(s,y-1,0), Upd(s,y,0), Upd(s,t1,2), f[s]+=v;
}
} int main()
{
int n,m; scanf("%d%d",&n,&m);
if(m>n) std::swap(n,m); ::n=n, ::m=m;
if(m==1) return puts("1"),0;//!
int p=0; f[p].Init(), f[p][0]=1;
for(int i=1; i<=n; ++i)
{
for(int j=1; j<=m; ++j) p^=1, Work(n,m,i,j,f[p],f[p^1]);
if(i!=n)
{
p^=1, f[p].Init();
for(int i=1,tot=f[p^1].tot,s; i<=tot; ++i)
s=f[p^1].sta[i], f[p][s<<2]=f[p^1][s];
}
}
Ans+=Ans, Ans.Print(); return 0;
}

BZOJ.1210.[HNOI2004]邮递员(插头DP Hash 高精)的更多相关文章

  1. bzoj 1210 [HNOI2004] 邮递员 插头dp

    插头dp板子题?? 搞了我一晚上,还tm全是抄的标程.. 还有高精,哈希混入,还是我比较弱,orz各种dalao 有不明白的可以去看原论文.. #include<cstdio> #incl ...

  2. 【BZOJ1210】[HNOI2004]邮递员 插头DP+高精度

    [BZOJ1210][HNOI2004]邮递员 Description Smith在P市的邮政局工作,他每天的工作是从邮局出发,到自己所管辖的所有邮筒取信件,然后带回邮局.他所管辖的邮筒非常巧地排成了 ...

  3. vijos 1110小胖邮递员;bzoj 1210: [HNOI2004]邮递员

    Description Smith在P市的邮政局工作,他每天的工作是从邮局出发,到自己所管辖的所有邮筒取信件,然后带回邮局.他所管辖的邮筒非常巧地排成了一个m*n的点阵(点阵中的间距都是相等的).左上 ...

  4. 无聊的 邮递员 插头dp

    邮递员想知道,如果他每天都用不同路线走过10×20个点阵邮筒,他必须活过多少个世纪才能走遍所有方案? 7:00 改完T1,开始肝插头dp 7:10 放弃,颓博客 7:20 学习插头dp 7:21 放弃 ...

  5. bzoj 2331: [SCOI2011]地板 插头DP

    2331: [SCOI2011]地板 Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 541  Solved: 239[Submit][Status] D ...

  6. 【BZOJ】2310: ParkII 插头DP

    [题意]给定m*n的整数矩阵,求经过所有点至多一次路径的最大数值和.n<=8,m<=100. [算法]插头DP [题解]最小表示法确实十分通用,处理简单路径问题只需要状态多加一位表示独立插 ...

  7. BZOJ 2331 [SCOI2011]地板 ——插头DP

    [题目分析] 经典题目,插头DP. switch 套 switch 代码瞬间清爽了. [代码] #include <cstdio> #include <cstring> #in ...

  8. 「NOIP模拟赛」数位和乘积(dp,高精)

    统计方案数,要么组合数,要么递推(dp)了. 这是有模拟赛历史以来爆炸最狠的一次 T1写了正解,也想到开long long,但是开错了地方然后数组开大了结果100->0 T3看错题本来简单模拟又 ...

  9. BZOJ.1005.[HNOI2008]明明的烦恼(Prufer 高精 排列组合)

    题目链接 若点数确定那么ans = (n-2)!/[(d1-1)!(d2-1)!...(dn-1)!] 现在把那些不确定的点一起考虑(假设有m个),它们在Prufer序列中总出现数就是left=n-2 ...

随机推荐

  1. java web----MINA框架使用

    前期准备 1.下载 http://mina.apache.org/ 2.将依赖包添加到工程目录下(在工程目录下创建libs(directory目录)) 3.将 slf4j-api-1.7.26.jar ...

  2. redis 单实例安装

    单实例安装 近些年,由于内存技术的提升.造价的下降,越来越多企业的服务器内存已增加到几百G.这样的内存容量给了内存数据库一个良好的发展环境. 而使用Redis是内存数据库的一股清流,渐有洪大之势.下面 ...

  3. jdk1.8学习、jdk1.9学习、jdk10.0学习和总结

    由于中文参考资料很少,参考链接: https://www.oschina.net/translate/109-new-features-in-jdk-10 http://chuansong.me/n/ ...

  4. MySql定期存档数据

    # 创建归档表(只复制表结构和索引) CREATE TABLE Orders_2016 like Orders; # 删除归档表的所有索引(这样可以提高插入速度) # 以前的数据导入归档表 INSER ...

  5. 用Web api /Nancy 通过Owin Self Host简易实现一个 Http 服务器

    过去做 端游的Http 服务器 用的WebApi 或者Mvc架构,都是放在iis...而我已经是懒出一个地步,并不想去配iis,或者去管理iis,所以我很喜欢 Self host 的启动方式. C#做 ...

  6. [SDOI2018]荣誉称号

    题解: 并不需要什么算法 首先我们随便画一画就会发现 能画出一颗满二叉树 然后要满足每个点从上往下的路径和都相同(%m意义下) 一个点上可能对应了多个点 然后这样我们可以暴力dp $2^k*m^2+n ...

  7. python全栈开发day92-day96 Vue总结

    -- ES6常用语法 -- var let const -- 模板字符串 -- 反引号 -- ${} -- 箭头函数 -- 普通函数取决于函数最近的调用者 -- 箭头函数取决当前环境 -- 类 -- ...

  8. net core体系-web应用程序-4net core2.0大白话带你入门-11asp.net core 2.0 cookie的使用

    asp.net core 2.0 cookie的使用   本文假设读者已经了解cookie的概念和作用,并且在传统的.net framework平台上使用过. cookie的使用方法和之前的相比也有所 ...

  9. UOJ#191. 【集训队互测2016】Unknown 点分治 分治 整体二分 凸包 计算几何

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ191.html 题目传送门 - UOJ191 题意 自行移步集训队论文2016中罗哲正的论文. 题解 自行 ...

  10. java大数相加

    import java.math.BigInteger; import java.util.Scanner; public class Bignum{    public static void ma ...