bzoj 1210 [HNOI2004] 邮递员 插头dp
插头dp板子题??
搞了我一晚上,还tm全是抄的标程。。
还有高精,哈希混入,还是我比较弱,orz各种dalao
有不明白的可以去看原论文。。
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#define base (int)1e9
#define maxh 2650
using namespace std;
int n,m;
int get(int s,int x){
return (s>>((x-1)<<1))&3;
}//查询对于状态s,第x位的状态. 三进制,所以左移一位.&3,得0,1,2
void change(int &s,int x,int y){
int w=(x-1)<<1;
s&=~(3<<w);
s|=y<<w;
}//将s状态的第x位改为y
int find(int s,int x){
for(int i=x,d=0,pos=get(s,x)==1?1:-1;;i+=pos){
int nn=get(s,i);
if(nn==1) d++; if(nn==2) d--;
if(!d) return i;
}
}//找与该位匹配的括号
struct bignum{//高精没啥好说的...
int a[5];
void clear(){memset(a,0,sizeof a);}
bignum(){clear();}
void set(int x){clear();while(x){a[++a[0]]=x%base;x/=base;}}
void print(){
printf("%d",a[a[0]]);
for(int i=a[0]-1;i>0;i--)printf("%09d",a[i]);
}
bignum operator + (bignum b){
static bignum c;c.clear();c.a[0]=max(a[0],b.a[0])+1;
for(int i=1;i<=c.a[0];i++){
c.a[i]+=a[i]+b.a[i];
c.a[i+1]+=c.a[i]/base;
c.a[i]%=base;
}
while(!c.a[c.a[0]])c.a[0]--;
return c;
}
void operator += (bignum b){*this=*this+b;}
void operator = (int x){set(x);}
}ans;
struct hashtable{//骚骚的哈希
int key[maxh];
int cnt,hash[maxh];
bignum val[maxh];
void clear(){
memset(key,-1,sizeof key);
memset(hash,0,sizeof hash);
memset(val,0,sizeof val);
cnt=0;
}
void newnode(int i,int x){
hash[i]=++cnt;
key[cnt]=x;
}
bignum & operator[](const int hh){
int k;
for(int i=hh%maxh;;i++){
if(i==maxh)i=0;
if(!hash[i]) newnode(i,hh);
if(key[hash[i]]==hh){k=hash[i]; break;}
}
return val[k];
}
}f[2];
int now,last;
void dp(int i,int j){//插头dp
now^=1;last^=1;
int tot=f[last].cnt;
f[now].clear();
for(int k=1,x,y;k<=tot;k++){
int s=f[last].key[k];
bignum vv=f[last].val[k];
x=get(s,j); y=get(s,j+1);
if(!x&&!y){if(i!=n&&j!=m){change(s,j,1);change(s,j+1,2);f[now][s]+=vv;}}
else if(x==1&&y==1){
change(s,find(s,j+1),1);
change(s,j,0); change(s,j+1,0);
f[now][s]+=vv;
}
else if(x==2&&y==2){
change(s,find(s,j),2);
change(s,j,0); change(s,j+1,0);
f[now][s]+=vv;
}
else if(x==1&&y==2){
if(i==n&&j==m) ans+=vv;
}
else if(x==2&&y==1){
change(s,j,0); change(s,j+1,0);
f[now][s]+=vv;
}
else if(x){
if(i!=n) f[now][s]+=vv;
if(j!=m){change(s,j,0);change(s,j+1,x);f[now][s]+=vv;}
}
else if(y){
if(j!=m) f[now][s]+=vv;
if(i!=n){change(s,j+1,0);change(s,j,y);f[now][s]+=vv;}
}
}
}
int main(){
freopen("postman.in","r",stdin);
freopen("postman.out","w",stdout);
scanf("%d%d",&n,&m);
if(n==1||m==1){printf("1\n");return 0;}
if(n<m)swap(n,m);
f[0][0]=1;
now=0;last=1;
for(int i=1;i<=n;i++){
for(int j=1;j<=m;j++)dp(i,j);
if(i!=n)for(int j=1,k=f[now].cnt;j<=k;j++)
f[now].key[j]<<=2;
}
ans+=ans;
ans.print();
return 0;
}
bzoj 1210 [HNOI2004] 邮递员 插头dp的更多相关文章
- BZOJ.1210.[HNOI2004]邮递员(插头DP Hash 高精)
BZOJ 洛谷 http://www.cnblogs.com/LadyLex/p/7326874.html 插头DP.\(m+1\)个插头的状态需要用三进制表示:\(0\)表示无插头,\(1\)表示是 ...
- 【BZOJ1210】[HNOI2004]邮递员 插头DP+高精度
[BZOJ1210][HNOI2004]邮递员 Description Smith在P市的邮政局工作,他每天的工作是从邮局出发,到自己所管辖的所有邮筒取信件,然后带回邮局.他所管辖的邮筒非常巧地排成了 ...
- vijos 1110小胖邮递员;bzoj 1210: [HNOI2004]邮递员
Description Smith在P市的邮政局工作,他每天的工作是从邮局出发,到自己所管辖的所有邮筒取信件,然后带回邮局.他所管辖的邮筒非常巧地排成了一个m*n的点阵(点阵中的间距都是相等的).左上 ...
- 无聊的 邮递员 插头dp
邮递员想知道,如果他每天都用不同路线走过10×20个点阵邮筒,他必须活过多少个世纪才能走遍所有方案? 7:00 改完T1,开始肝插头dp 7:10 放弃,颓博客 7:20 学习插头dp 7:21 放弃 ...
- bzoj 2331: [SCOI2011]地板 插头DP
2331: [SCOI2011]地板 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 541 Solved: 239[Submit][Status] D ...
- 【BZOJ】2310: ParkII 插头DP
[题意]给定m*n的整数矩阵,求经过所有点至多一次路径的最大数值和.n<=8,m<=100. [算法]插头DP [题解]最小表示法确实十分通用,处理简单路径问题只需要状态多加一位表示独立插 ...
- BZOJ 2331 [SCOI2011]地板 ——插头DP
[题目分析] 经典题目,插头DP. switch 套 switch 代码瞬间清爽了. [代码] #include <cstdio> #include <cstring> #in ...
- BZOJ 1212: [HNOI2004]L语言( dp + trie )
因为单词很短...用trie然后每次dp暴力查找...用哈希+dp应该也是可以的.... ------------------------------------------------------- ...
- BZOJ 1207: [HNOI2004]打鼹鼠( dp )
dp.. dp[ i ] = max( dp[ j ] + 1 ) ------------------------------------------------------------------ ...
随机推荐
- RunTime运行时在iOS中的应用之UITextField占位符placeholder
RunTime运行时机制 runtime是一套比较底层的纯C语言API, 属于1个C语言库, 包含了很多底层的C语言API. 在我们平时编写的Objective-C代码中, 程序运行过程时, 其实最终 ...
- 听晴明老师从头讲React Native(原价399)百度云下载 百度网盘
适用人群 能使用至少一门主流编程语言:有基本的面向对象的概念:最好有一些web相关的知识和概念. 课程概述 新颖.实用.详尽的ReactNative零基础课程,由国内权威的ReactNative中文网 ...
- 小dai浅谈通信网络(一)——引子
说起通信网络,首先来看一个场景: 场景模式: 小明和小刚在闹市碰面. 小明对小刚大声喊道:"小刚,你好啊!" 小刚摇手答到:"你好,小明!" 就这么几句简单的话 ...
- Linux命令之文件搜索
locate 文件名 locate只能搜索文件名,不能搜索文件大小.搜索速度快. locate并不会搜索到那些新加入的文件.新加入文件后,使用updatedb,更新数据库后,再使用locate搜索. ...
- spiral matrix 螺旋矩阵
Given a matrix of m x n elements (m rows, n columns), return all elements of the matrix in spiral or ...
- HTML DOM - 导航
HTML DOM 节点列表长度 length 属性定义节点列表中节点的数量. 您可以使用 length 属性来循环节点列表: x=document.getElementsByTagName(" ...
- 多重影分身——C#中多线程的使用一(基础)
首先明确几个概念: 什么是进程? 当一个程序开始运行时,它就是一个进程,进程包括运行中的程序和程序所使用到的内存和系统资源. 而一个进程又是由多个线程所组成的. 一个程序通常只有一个进程(不包括exe ...
- Qt中的ui指针和this指针
初学qt,对其ui指针和this指针产生疑问,画了个把小时终于搞懂了. 首先看ui指针的定义: 在mainwindow.h中 private: Ui::MainWindow *ui; Ui又是什么? ...
- Android Gradle使用总结
转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/77678577 本文出自[赵彦军的博客] 其他 Groovy 使用完全解析 http ...
- 当配置 DispatcherServlet拦截“/”,SpringMVC访问静态资源的三种方式
如何你的DispatcherServlet拦截 *.do这样的URL,就不存在访问不到静态资源的问题.如果你的DispatcherServlet拦截“/”,拦截了所有的请求,同时对*.js,*.jpg ...