BZOJ1566 【NOI2009】管道取珠

题面

这是一道DP神题,直到我写下这句题解时也没有想明白……

首先,这道题要我们求所有(不同输出序列的方案数)的平方和,于是我们当然就想到求所有不同输出序列的方案数……(大雾) 。这道题一个巧妙的地方就在于对问题的转化。(以下摘自BYVoid大神的题解

假设同时有两个人X & Y在玩这个游戏,设X从up取了i个珠子(不一定连续),从down取了j个珠子,取出来的珠子组成的序列为Q,操作序列为x,Y从up取了k个珠子,从down取了l个珠子,取出来的珠子组成的序列也为Q,操作序列为y,那么我们就得到了一个有序对(x,y),f[i][j][k][l]即表示有序对(x,y)的数量。两个有序对不相同当且仅当x和y不同时相同。

下面证明f[i][j][k][l]即为所求。

已知:取出珠子的序列为Q,x和y分别为一种取珠方法(可相同), 取出Q的方案数为a;

求证:有序对(x,y)的数量等于a2

因为取出Q的方案数为a,所以x & y都有a种取值,且x & y彼此独立,故对于x的每一个取值,y都有a种取值,故有序对(x,y)的数量为a2,命题得证。

博主是个超级大傻*,连空间优化到n2都不会,请各路大神指教。

 1 #include <map>
 2 #include <set>
 3 #include <cmath>
 4 #include <queue>
 5 #include <stack>
 6 #include <cstdio>
 7 #include <string>
 8 #include <vector>
 9 #include <cstring>
10 #include <complex>
11 #include <cstdlib>
12 #include <iostream>
13 #include <algorithm>
14 #define rg register
15 #define ll long long
16 using namespace std;
17
18 inline int gi()
19 {
20     rg int r = 0; rg bool b = 1; rg char c = getchar();
21     while (c < ‘0‘ || c > ‘9‘) { if (c == ‘-‘) b = 0; c = getchar(); }
22     while (c >= ‘0‘ && c <= ‘9‘) { r = r * 10 + c - ‘0‘, c = getchar(); }
23     if (b) return r; return -r;
24 }
25
26 const int inf = 2147483647, N = 505, MOD = 1024523;
27 int n,m,f[N][N][N];
28 char S[N],X[N];
29
30 inline void input()
31 {
32     freopen ("!.in", "r", stdin);
33     n=gi(), m=gi();
34     scanf("%s%s",S+1,X+1);
35 }
36
37 inline void output()
38 {
39     freopen ("!.out", "w", stdout);
40     printf("%d\n",f[n][m][n]);
41 }
42
43 inline void cal(int &t,int d) { t+=d; if (t >= MOD) t-=MOD; }
44
45 inline void solve()
46 {
47     int i,j,k,l,tmp;
48     f[0][0][0]=1;
49     for (i=0; i<=n; i++)
50         for (j=0; j<=m; j++)
51             for (k=0; k<=n; k++)
52                 {
53                     tmp=f[i][j][k], l=i+j-k;
54                     if (!tmp || !l || l > m) continue;
55                     if (S[i+1] == S[k+1])
56                         cal(f[i+1][j][k+1],tmp);
57                     if (X[j+1] == S[k+1])
58                         cal(f[i][j+1][k+1],tmp);
59                     if (S[i+1] == X[l+1])
60                         cal(f[i+1][j][k],tmp);
61                     if (X[j+1] == X[l+1])
62                         cal(f[i][j+1][k],tmp);
63                 }
64 }
65
66 int main()
67 {
68     input();
69     solve();
70     output();
71     return 0;
72 }
时间: 2024-10-05 11:40:48

BZOJ1566 【NOI2009】管道取珠的相关文章

bzoj1566[noi2009]管道取珠

bzoj1566[noi2009]管道取珠 题意: 有个装置,左侧有上下两条管道分别有n个和m个不同颜色的两种球,右侧一条空管道.每次可以选左侧的一条管道将最右侧的球推到右侧管道,经过n+m次操作,右侧管道从右到左形成一个输出序列.求不同种类的输出序列的产生方式数的平方之和.n,m≤500 题解: 将题目转化成两个人同时取,取出来的序列相同的可能性有多少种.于是做dp.方程见代码.f[i][j][k][l]表示第一个人取了i个Aj个B第二个人取了k个Al个B. 代码: 1 #include <c

洛谷P1758 [NOI2009]管道取珠

题目:https://www.luogu.org/problemnew/show/P1758 题目描述 管道取珠是小X很喜欢的一款游戏.在本题中,我们将考虑该游戏的一个简单改版.游戏画面如图1所示: (图1) 游戏初始时,左侧上下两个管道分别有一定数量的小球(有深色球和浅色球两种类型),而右侧输出管道为空.每一次操作,可以从左侧选择一个管道,并将该管道中最右侧的球推入右边输出管道. 例如:我们首先从下管道中移一个球到输出管道中,将得到图2所示的情况. (图2) 假设上管道中有n个球, 下管道中有

【Luogu】P1758 [NOI2009] 管道取珠

终于开始正式写了.... 题目描述 管道取珠是小X很喜欢的一款游戏.在本题中,我们将考虑该游戏的一个简单改版.游戏画面如图1所示: (图1) 游戏初始时,左侧上下两个管道分别有一定数量的小球(有深色球和浅色球两种类型),而右侧输出管道为空.每一次操作,可以从左侧选择一个管道,并将该管道中最右侧的球推入右边输出管道. 例如:我们首先从下管道中移一个球到输出管道中,将得到图2所示的情况. (图2) 假设上管道中有n个球, 下管道中有m个球,则整个游戏过程需要进行n+m次操作,即将所有左侧管道中的球移

bzoj 1566: [NOI2009]管道取珠

Description   Input 第一行包含两个整数n, m,分别表示上下两个管道中球的数目. 第二行为一个AB字符串,长度为n,表示上管道中从左到右球的类型.其中A表示浅色球,B表示深色球. 第三行为一个AB字符串,长度为m,表示下管道中的情形. Output 仅包含一行,即为 Sigma(Ai^2) i从1到k 除以1024523的余数. Sample Input 2 1 AB B Sample Output 5 HINT 样例即为文中(图3).共有两种不同的输出序列形式,序列BAB有

1566: [NOI2009]管道取珠 - BZOJ

Description Input第一行包含两个整数n, m,分别表示上下两个管道中球的数目. 第二行为一个AB字符串,长度为n,表示上管道中从左到右球的类型.其中A表示浅色球,B表示深色球. 第三行为一个AB字符串,长度为m,表示下管道中的情形.Output仅包含一行,即为 Sigma(Ai^2) i从1到k 除以1024523的余数.Sample Input2 1ABB Sample Output5HINT 样例即为文中(图3).共有两种不同的输出序列形式,序列BAB有1种产生方式,而序列B

BZOJ 1566 管道取珠(DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1566 题意: 思路:假设得到同一个序列S 的方案有ai=3,x,y,z,那么ai^2就是(x,x)(x,y)(x,z)(y,x)(y,y)(y,z)(z,x)(z,y)(z,z),因此我 们可以将原来的一套装置看做两套.f[i][j][p][q]表示从第一套装置的上面取i个下面取j个.从第二套装置的上面取p个下面取q个,且两套装置 得到的两个序列相同的方案数,则答案就是f[n][m][

BZOJ 1566 管道取珠(DP)

求方案数的平方之和.这个看起来很难解决.如果转化为求方案数的有序对的个数.那么就相当于求A和B同时取,最后序列一样的种数. 令dp[i][j][k]表示A在上管道取了i个,下管道取了j个,B在上管道取了k个,下管道取了i+j-k个珠子的序列相同的种数. 那么状态转移方程显然可得. # include <cstdio> # include <cstring> # include <cstdlib> # include <iostream> # include

管道取珠

平方转化为两个人取到相同的方案,这是一个小trick 为什么这样是对的? 假设第一个人取方案是$x$,第二个人取方案是$y$,根据乘法原理就是$x*y$,又因为两个人取得方案数相同所以$x==y$,即$x^2$ 所以设f[a][b][c][d]表示第一个人从第一个管道取a,第一个人从管道取b,第二个人从管道取c,第二个人从管道取d 优化一下可以去掉一维 #include<bits/stdc++.h> using namespace std; #define ll long long #defi

bzoj1566【Noi2009】管道取珠

题意:http://www.lydsy.com/JudgeOnline/problem.php?id=1566 两个栈不断pop,共C(n+m,n)种,ai表示每个相同序列的方案数,求∑(ai^2) sol  :首先,将相同的序列看做两个人选取后相同的方案数 考虑Dp,dp[i][j][k][l]表示第一个人从上面选i个,下面选j个,第二个人上k个下l个的答案 显然第四维状态可以由前三维决定 不过还是不太好转移,将状态换为dp[i][j][k]表示选了i个点,第一个人从上面选了j个,第二个人从上