~~~题面~~~

思路:

主要难点在思路的转化,

不能看见要求$\sum{a[i]^2}$就想着求a[i],

我们可以对其进行某种意义上的拆分,即a[i]实际上可以代表什么?

假设我们现在有两种取出某一数列的方法,分别为X,Y。(X,Y可以相同)

那么这样的二元组有多少个呢?

a[i]^2个。

因为X的取法有a[i]种,Y的取法也是a[i]种,所以二元组个数实际上就是a[i]^2.

那么这样一转化有什么好处?

方便DP

因为这样的话就不在需要知道具体的a[i]了,因为二元组的个数是可以拆开来算的,

所以就可以考虑递推/DP了。

我们用f[i][j][k][l]表示第一种决策X在上面取了i个,下面取了j个,第二种决策Y在上面取了k个,在下面取了l个且产生序列相同的二元组个数

显然一开始的f[0][0][0][0]要设为1,

然后注意到我们应该要同步取,不然个数都不同,序列也不可能相同了,

因此我们数组中就可只存3维了,因为第4维可以通过i + j - k得到。

那么应该如何转移呢?

假设现在我们有f[i][j][k],显然要使下一次取出来的序列相同,只要让下一次取的珠子一样就可以了,因为其他部分现在已经一样了,

因此我们就可以分别枚举X取上面,X取下面,Y取上面,Y取下面,然后相互搭配,一共4种情况,如果满足下一次取的珠子一样就可以转移,

因为n有500,所以空间还是承受不了,观察到i的转移只涉及到i和i+1,因此我们可以用滚动数组优化。

但是我们注意到用f[i][j][k]向f[i+1][j][k+1]这类的转移的话,如果不及时清空,因为使用的是滚动数组,转移的时候又是+=,无法将原来的答案覆盖,

那就会重复统计,因此我们不能向前转移,我们应该要枚举当前状态,枚举向当前状态转移的状态,并且每次转移之前要清空数组,这样就可以了

注意将不合法的状态减掉,luogu上这题貌似有点卡常,,,

work1是另一种状态表示方法,,,我也不知道为什么比work快

 #include<bits/stdc++.h>
using namespace std;
#define R register int
#define AC 510
#define mod 1024523
/*转化为有序数对的统计之后就可以不用在意具体数列到底是什么样的了,因为只要相同就可以了,
但是由于是同步取的,所以两种决策取的次数是相同的,所以少掉一维是没关系的,
因为剩下那维可以直接推出来,是没用的*/
int n,m,now;
int f[][AC][AC];
char U[AC],D[AC];
void pre()
{
scanf("%d%d",&n, &m);
scanf("%s", U + );
scanf("%s", D + );
reverse(U+,U+n+);
reverse(D+,D+m+);//因为是从右边开始取的,所以要反过来!!!
} inline void update(int &a,int b)
{
a += b;
if(a > mod) a -= mod;
}
//本来是向前做贡献,但是这样贡献会和前面的累加,
//所以需要在做的过程中初始化,但这样就不太好掌控什么时候初始化,
//因此改成从前面取贡献,方便初始化,但是这样的话前面的判断就要麻烦点了
void work()
{
f[now][][] = ;//一开始啥也没取,,,
for(R i=;i<=n;i++)//可以取0个啊
{
for(R j=;j<=m;j++)
for(R k=;k<=n;k++)
{
if(!i && !j) continue;//这个不能初始化掉了
if(i + j - k > m) continue;//因为i,j,k枚举了,所以肯定合法,但最后一个要计算得到,不一定合法
f[now][j][k] = ;//初始化,防止叠加
if(i && k && U[i] == U[k]) update(f[now][j][k],f[now^][j][k-]);
if(i && i + j - k && U[i] == D[i + j - k]) update(f[now][j][k],f[now^][j][k]);
if(j && k && D[j] == U[k]) update(f[now][j][k],f[now][j-][k-]);
if(j && i + j - k && D[j] == D[i + j - k]) update(f[now][j][k],f[now][j-][k]);
// printf("%d %d %d = %d\n",i,j,k,f[now][j][k]);
}
now ^= ;
}
printf("%d\n",f[now^][m][n]);
printf("time used ... %lf\n",(double)clock()/CLOCKS_PER_SEC);
} void work1()
{//现在取了i个,第一种决策在第一行取了j个,第二种取了k个
f[][][] = ;//因为是滚了一维
int b = n + m;
for(R i = ; i <= b; i++)
{
int lim = min(i, n);
for(R j = ; j <= lim; j++)
{
for(R k = ; k <= lim; k++)
{
int x = i - j, y = i - k;//x为第一种决策在第二行取的,y则是第二种决策
if(x > m || y > m) continue;
f[now][j][k] = ;
if(j && U[j] == U[k]) update(f[now][j][k], f[now^][j-][k-]);
if(j && U[j] == D[y]) update(f[now][j][k], f[now^][j-][k]);
if(x && D[x] == U[k]) update(f[now][j][k], f[now^][j][k-]);
if(x && D[x] == D[y]) update(f[now][j][k], f[now^][j][k]);
// printf("%d %d %d = %d\n",i,j,k,f[now][j][k]);
}
}
now ^= ;
}
printf("%d\n",f[now^][n][n]);
// printf("time used ... %lf\n",(double)clock()/CLOCKS_PER_SEC);
} int main()
{
//freopen("in.in","r",stdin);
pre();
//work();
work1();
//fclose(stdin);
return ;
}

[NOI2009]管道取珠 DP + 递推的更多相关文章

  1. Bzoj 1566: [NOI2009]管道取珠(DP)

    1566: [NOI2009]管道取珠 Time Limit: 20 Sec Memory Limit: 650 MB Submit: 1558 Solved: 890 [Submit][Status ...

  2. BZOJ.1566.[NOI2009]管道取珠(DP 思路)

    BZOJ 洛谷 考虑\(a_i^2\)有什么意义:两个人分别操作原序列,使得得到的输出序列都为\(i\)的方案数.\(\sum a_i^2\)就是两人得到的输出序列相同的方案数. \(f[i][j][ ...

  3. bzoj1566: [NOI2009]管道取珠 DP

    题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=1566 思路 n个球,第i个球颜色为ai,对于颜色j,对答案的贡献为颜色为j的球的个数的平 ...

  4. bzoj1566 [NOI2009]管道取珠——DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1566 一眼看上去很懵... 但是答案可以转化成有两个人在同时取珠子,他们取出来一样的方案数: ...

  5. 【BZOJ 1566】 1566: [NOI2009]管道取珠 (DP)

    1566: [NOI2009]管道取珠 Time Limit: 20 Sec  Memory Limit: 650 MBSubmit: 1659  Solved: 971 Description In ...

  6. NOI2009 管道取珠 神仙DP

    原题链接 原题让求的是\(\sum\limits a_i^2\),这个东西直接求非常难求.我们考虑转化一下问题. 首先把\(a_i^2\)拆成\((1+1+...+1)(1+1+...+1)\),两个 ...

  7. BZOJ 1566 管道取珠(DP)

    求方案数的平方之和.这个看起来很难解决.如果转化为求方案数的有序对的个数.那么就相当于求A和B同时取,最后序列一样的种数. 令dp[i][j][k]表示A在上管道取了i个,下管道取了j个,B在上管道取 ...

  8. 【题解】NOI2009管道取珠

    又是艰难想题的一晚,又是做不出来的一题 (:д:) 好想哭啊…… 这题最关键的一点还是提供一种全新的想法.看到平方和这种东西,真的不好dp.然而我一直陷在化式子的泥潭中出不来.平方能够联想到什么?原本 ...

  9. BZOJ1566 [NOI2009]管道取珠 【dp】

    题目 输入格式 第一行包含两个整数n, m,分别表示上下两个管道中球的数目. 第二行为一个AB字符串,长度为n,表示上管道中从左到右球的类型.其中A表示浅色球,B表示深色球. 第三行为一个AB字符串, ...

随机推荐

  1. iOS SSL Pinning 保护你的 API

    随着互联网的发展,网站全面 https 化已经越来越被重视,做为 App 开发人员,从一开始就让 API 都走 SSL 也是十分必要的.但是光这样就足够了吗? SSL 可以保护线上 API 数据不被篡 ...

  2. 生鲜水果商城PC手机微信完整版源码2018版(免费)

    采用php+mysql架构,含有PC.手机.微信三端,只需要修改一下数据库配置,并恢复一下数据即可使用,还有微信.支付宝等接口,如有问题请在文章下面留言一下,我看到会协助一下的,下载包里面含有详细的安 ...

  3. 「日常训练」 Genghis Khan the Conqueror(HDU-4126)

    题意 给定\(n\)个点和\(m\)条无向边(\(n\le 3000\)),需要将这\(n\)个点连通.但是有\(Q\)次(\(Q\le 10^4\))等概率的破坏,每次破坏会把\(m\)条边中的某条 ...

  4. 第二章 IP协议详解

    第二章 IP协议详解 2.1 IP服务的特点 它为上层协议提供了无状态,无连接,不可靠的服务 名称 简介 优点 缺点 对付缺点的方法 无状态 IP通信双方不同步传输数据的状态信息 无须为保持通信的状态 ...

  5. JavaScript基本概念(1)-声明提升

    声明提升: function > var > other var提升的时候,只是声明提升,但是赋值还是会在原来的位置. Javascript Hoisting:In javascript, ...

  6. 【WXS数据类型】Object

    Object 是一种无序的键值对. 属性: 名称 值类型 说明 [Object].constructor [String] 返回值为“Object”,表示类型的结构字符串 方法: 原型:[Object ...

  7. Apache--Override参数详解

    1  AuthConfig  允许使用所有的权限指令,他们包括AuthDBMGroupFile AuthDBMUserFile  AuthGroupFile  AuthName AuthTypeAut ...

  8. 浅谈PCA

    最近在回顾PCA方面的知识,发现对于之前的很多东西有了新的理解,下面和大家分享下我的一些个人的理解 1.我们为什么要用PCA,它能解决我什么问题? PCA(Principal Component An ...

  9. HDU 2491 Priest John's Busiest Day(贪心)(2008 Asia Regional Beijing)

    Description John is the only priest in his town. October 26th is the John's busiest day in a year be ...

  10. java设计模式简介

    设计模式简介: 设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用.设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案.这些解决方案是众多 ...