zoukankan      html  css  js  c++  java
  • 模板—插头dp(Ural 1519 Formula 1)

    括号表示法:

    据说比下一个要快而且灵活。

      1 #include<iostream>
      2 #include<cstring>
      3 #include<cstdio>
      4 #define LL long long
      5 #define MAXN 20000
      6 #define HASH 23333
      7 #define ma(x,y) memset(x,y,sizeof(x))
      8 using namespace std;
      9 struct Hash_map
     10 {
     11     int size,first[HASH],next[MAXN];
     12     #define f(x) first[x]
     13     #define n(x) next[x]
     14     LL sta[MAXN],sum[MAXN];
     15     void init(){size=0;ma(first,-1);ma(sum,0);}
     16     void push(LL states,LL Sum)
     17     {
     18         int pos=(states%HASH+HASH)%HASH;
     19         for(int i=f(pos);i>=0;i=n(i))
     20             if(sta[i]==states){sum[i]+=Sum;return;}    
     21         sta[size]=states;
     22         sum[size]=Sum;
     23         n(size)=f(pos);f(pos)=size++;
     24     }
     25 }dp[2];
     26 int n,m,map[15][15],bin[16],fx,fy;//最后一个可行点坐标
     27 int now,pre;
     28 int mov[13]={0,2,4,6,8,10,12,14,16,18,20,22,24};
     29 inline int getbit(LL st,int k){return (st>>mov[k])&3;}//第k位状态
     30 inline int pybit(LL st,int k){return st<<mov[k];}//平移k位
     31 inline LL clrbit(LL st,int i,int j){return st&(~(3<<mov[i]))&(~(3<<mov[j]));}//清空i,j位
     32 inline int fl(LL st,int pos)//从左往右找和当前pos位置匹配的右括号
     33 {
     34     int cnt=1;
     35     for(int i=pos+1;i<=m;i++)
     36     {
     37         int k=getbit(st,i);
     38         if(k==1)cnt++;
     39         else if(k==2)cnt--;
     40         if(!cnt)return i;
     41     }
     42 }
     43 inline int fr(LL st,int pos)//从右往左找和当前pos位置匹配的左括号
     44 {
     45     int cnt=1;
     46     for(int i=pos-1;i>=0;i--)
     47     {
     48         int k=getbit(st,i);
     49         if(k==1)cnt--;
     50         else if(k==2)cnt++;
     51         if(!cnt)return i;
     52     }
     53 }
     54 void DP(int x,int y,int k)
     55 {
     56     int l=getbit(dp[pre].sta[k],y-1);
     57     int up=getbit(dp[pre].sta[k],y);
     58     LL st=clrbit(dp[pre].sta[k],y-1,y);
     59     LL v=dp[pre].sum[k];
     60     if(!l&&!up)
     61     {
     62         if(!map[x][y]){dp[now].push(st,v);return;}    
     63         if(x<n&&y<m&&map[x+1][y]&&map[x][y+1])
     64             dp[now].push(st|pybit(1,y-1)|pybit(2,y),v);
     65     }
     66     else if(!l||!up)
     67     {
     68         int e=l+up;
     69         if(x<n&&map[x+1][y])
     70             dp[now].push(st|pybit(e,y-1),v);
     71         if(y<m&&map[x][y+1])
     72             dp[now].push(st|pybit(e,y),v);
     73     }
     74     else if(l==1&&up==1)
     75         dp[now].push(st^pybit(3,fl(st,y)),v);
     76     else if(l==2&&up==2)
     77         dp[now].push(st^pybit(3,fr(st,y-1)),v);
     78     else if(l==2&&up==1)
     79         dp[now].push(st,v);
     80     else if(x==fx&&y==fy)
     81         dp[now].push(st,v);
     82 }
     83 LL solve()
     84 {
     85     dp[0].init();
     86     dp[0].push(0,1);
     87     now=0,pre=1;
     88     for(int i=1;i<=n;i++)
     89     {
     90         pre=now,now^=1;dp[now].init();
     91         for(int k=0;k<dp[pre].size;k++)
     92             dp[now].push(pybit(dp[pre].sta[k],1),dp[pre].sum[k]);
     93         for(int j=1;j<=m;j++)
     94         {    
     95             pre=now,now^=1;dp[now].init();
     96             for(int k=0;k<dp[pre].size;k++)
     97                 DP(i,j,k);
     98         }
     99     }
    100     for(int i=0;i<dp[now].size;i++)
    101         if(dp[now].sta[i]==0)
    102             return dp[now].sum[i];
    103     return 0;
    104 }
    105 signed main()
    106 {
    107 //    freopen("in.txt","r",stdin);                                    
    108 
    109     bin[0]=1;for(int i=1;i<=15;i++)bin[i]=bin[i-1]*2;
    110     cin>>n>>m;char t;
    111     for(int i=1;i<=n;i++)
    112         for(int j=1;j<=m;j++)
    113         {    
    114             cin>>t;
    115             if(t=='*')map[i][j]=0;
    116             else map[i][j]=1;
    117         }
    118     fx=0;
    119     for(int i=n;i>0&&!fx;i--)
    120         for(int j=m;j>0&&!fx;j--)
    121         if(map[i][j])
    122             fx=i,fy=j;
    123     if(fx==0)puts("0");
    124     else cout<<solve()<<endl;
    125 }
    括号表示法(这个是自己写的)

    最小表示法(看不懂,下面是标程):

    不过连我都能看出来它慢那它就是真的慢了。而且我也并没有觉得它好理解……

     找到一个和自己码风相似的不容易啊……

      1 #include<bits/stdc++.h>
      2 
      3 #define LL long long  
      4 using namespace std;  
      5 const int maxn=30001,inc=3,bit=7;//3位二进制以及111的表示  
      6 int n,m,now,pre,code[20],bin[20],res[20];//用来表示状态的每一位的数值  
      7 char gp[20][20],fx,fy;//图和最后的可行点  
      8 struct node//离散化hash  
      9 {  
     10     int head[maxn],next[maxn],size;  
     11     LL sum[maxn],sta[maxn];  
     12     void clear()  
     13     {  
     14         memset(head,-1,sizeof(head));  
     15         size=0;  
     16     }  
     17     void push(LL st,const LL v)  
     18     {  
     19         LL hash=st%maxn;  
     20         for(int i=head[hash];i>=0;i=next[i])  
     21         {  
     22             if(sta[i]==st)  
     23             {  
     24                 sum[i]+=v;  
     25                 return ;  
     26             }  
     27         }  
     28         sta[size]=st,sum[size]=v;  
     29         next[size]=head[hash],head[hash]=size++;  
     30     }  
     31 }dp[2];  
     32 inline LL encode(int m)//将code转换成状态  
     33 {  
     34     LL st=0;  
     35     int cnt=1;  
     36     memset(bin,-1,sizeof(bin));  
     37     bin[0]=0;  
     38     for(int i=m;i>=0;i--)  
     39     {  
     40         if(bin[code[i]]==-1)  
     41             bin[code[i]]=cnt++;  
     42         code[i]=bin[code[i]];  
     43         st<<=inc;  
     44         st|=code[i];  
     45     }  
     46     return st;  
     47 }  
     48 inline void decode(LL st,int m)//将状态转换成code  
     49 {  
     50     for(int i=0;i<=m;i++)  
     51     {  
     52         code[i]=st&bit;  
     53         st>>=inc;  
     54     }  
     55 }  
     56 void DP(int x,int y,int k)//dp具体情况具体分析  
     57 {  
     58     decode(dp[pre].sta[k],m);  
     59     int l=code[y-1];  
     60     int up=code[y];  
     61     code[y-1]=code[y]=0;  
     62     memcpy(res,code,sizeof(code));  
     63     LL v=dp[pre].sum[k];  
     64     if(!l&&!up)  
     65     {  
     66         if(gp[x][y]=='*')  
     67             dp[now].push(encode(m),v);  
     68         else if(x<n&&y<m&&gp[x+1][y]=='.'&&gp[x][y+1]=='.')  
     69         {  
     70             code[y]=code[y-1]=bit;  
     71             dp[now].push(encode(m),v);  
     72         }  
     73     }  
     74     else if(!l||!up)  
     75     {  
     76         int e=l+up;  
     77         if(x<n&&gp[x+1][y]=='.')  
     78         {  
     79             code[y-1]=e;  
     80             dp[now].push(encode(m),v);  
     81             memcpy(code,res,sizeof(res));  
     82         }  
     83         if(y<m&&gp[x][y+1]=='.')  
     84         {  
     85             code[y]=e;  
     86             dp[now].push(encode(m),v);  
     87         }  
     88     }  
     89     else if(l!=up)  
     90     {  
     91         for(int i=0;i<=m;i++)  
     92             if(code[i]==up)  
     93                 code[i]=l;  
     94         dp[now].push(encode(m),v);  
     95     }  
     96     else if(x==fx&&y==fy)  
     97         dp[now].push(encode(m),v);  
     98 }  
     99 LL solve()  
    100 {  
    101     dp[0].clear();//初始化状态  
    102     dp[0].push(0,1);  
    103     now=0,pre=1;  
    104     for(int i=1;i<=n;i++)//逐格逐状态枚举转移  
    105     {  
    106         pre=now,now^=1,dp[now].clear();  
    107         for(int k=0;k<dp[pre].size;k++)//轮廓线行转移  
    108             dp[now].push(dp[pre].sta[k]<<inc,dp[pre].sum[k]);  
    109         for(int j=1;j<=m;j++)  
    110         {  
    111             pre=now,now^=1,dp[now].clear();  
    112             for(int k=0;k<dp[pre].size;k++)  
    113             {  
    114                 DP(i,j,k);  
    115             }  
    116         }  
    117     }  
    118     for(int i=0;i<dp[now].size;i++)  
    119         if(dp[now].sta[i]==0)  
    120             return dp[now].sum[i];  
    121     return 0;  
    122 }  
    123 int main()  
    124 {  
    125     while(~scanf("%d%d",&n,&m))  
    126     {  
    127         for(int i=1;i<=n;i++)//都是从1开始  
    128             scanf("%s",&gp[i][1]);  
    129         fx=fy=0;  
    130         for(int i=n;i>0&&!fx;i--)//寻找最终的位置  
    131             for(int j=m;j>0&!fx;j--)  
    132                 if(gp[i][j]=='.')  
    133                     fx=i,fy=j;  
    134         if(fx==0)puts("0");  
    135         else cout<<solve()<<endl;  
    136     }  
    137 }
    最小表示法(摘自某大佬)
  • 相关阅读:
    PL/SQL Developer 远程连接Oracle数据库
    常用js代码学习
    三元运算 多条件嵌套
    在后台CS文件里面,隐藏和显示Repeater里面控件
    最小二乘线性及平面拟合原理及C++实现
    C++的顶层const 和 底层const
    vs2019上遇到的一个未预指定引用函数的静态库导致的问题
    一个基础QItemDelegate检查器类
    linux 基础命令及快捷键
    lambda表达式
  • 原文地址:https://www.cnblogs.com/Al-Ca/p/11259485.html
Copyright © 2011-2022 走看看