zoukankan      html  css  js  c++  java
  • 洛谷 P1758 [NOI2009] 管道取珠(dp)

    传送门


    解题思路

    重要思想:求 (sum{a_i^2}) 相当于是取两遍球,取出来的序列相同的方案数。
    于是设dp[i][j][l][r]表示第一次取上下分别取i/j个,第二次取上下分别取l/r个,两次取出相同序列的方案数。
    转移就判断s1[i]/s2[j]和s1[l]/s2[r]之间的相等关系并进行转移即可。
    最后答案就是dp[n][m][n][m]。
    考虑优化。
    因为i+j==l+r,所以最后一维省去,r=i+j-l。
    再把第一维用滚动数组优化一下即可。
    还需要卡一下常数,本人用string第二个点一直过不去,卡了一晚上,改了char数组就好了。。
    注意细节。

    AC代码

    #include<cstdio>
    #include<iostream>
    #include<cstring>
    #include<iomanip>
    #include<cmath>
    #include<algorithm>
    using namespace std;
    const int maxn=505;
    const int mod=1024523;
    int n,m,dp[2][maxn][maxn];
    char s1[maxn],s2[maxn];
    int main(){
    	ios::sync_with_stdio(false);
    	cin>>n>>m;
    	for(int i=0;i<n;i++) cin>>s1[i];
    	for(int i=0;i<m;i++) cin>>s2[i]; 
    	reverse(s1,s1+n);
    	reverse(s2,s2+m);
    	dp[0][0][0]=1;
    	for(int i=0;i<=n;i++){
    		for(int j=0;j<=m;j++){
    			for(int l=0;l<=n;l++){
    				int r=i+j-l;
    				if(r<0) break;
    				if(r>m) continue;
    				dp[i&1][j][l]=0;
    				if(i==0&&j==0&&l==0) dp[0][0][0]=1;
    				if(i>0&&l>0&&s1[i-1]==s1[l-1]) dp[i&1][j][l]+=dp[(i+1)&1][j][l-1];
    				if(i>0&&r>0&&s1[i-1]==s2[r-1]) dp[i&1][j][l]+=dp[(i+1)&1][j][l];
    				if(j>0&&l>0&&s2[j-1]==s1[l-1]) dp[i&1][j][l]+=dp[i&1][j-1][l-1];
    				if(j>0&&r>0&&s2[j-1]==s2[r-1]) dp[i&1][j][l]+=dp[i&1][j-1][l];
    				while(dp[i&1][j][l]>=mod) dp[i&1][j][l]-=mod;
    			}
    		}
    	}
    	cout<<dp[n&1][m][n];
        return 0;
    }
    
  • 相关阅读:
    Storm监控文件夹变化 统计文件单词数量
    Storm默认配置 default.yaml
    Storm集群搭建
    Storm概念
    zookeeper安装
    zookeeper
    zookeeper应用
    zookeeper应用
    zookeeper应用
    HDU 3473 Minimum Sum (划分树求区间第k大带求和)(转)
  • 原文地址:https://www.cnblogs.com/yinyuqin/p/15257336.html
Copyright © 2011-2022 走看看