题面

这是一道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都不会,请各路大神指教。

 #include <map>
#include <set>
#include <cmath>
#include <queue>
#include <stack>
#include <cstdio>
#include <string>
#include <vector>
#include <cstring>
#include <complex>
#include <cstdlib>
#include <iostream>
#include <algorithm>
#define rg register
#define ll long long
using namespace std; inline int gi()
{
rg int r = ; rg bool b = ; rg char c = getchar();
while (c < '' || c > '') { if (c == '-') b = ; c = getchar(); }
while (c >= '' && c <= '') { r = r * + c - '', c = getchar(); }
if (b) return r; return -r;
} const int inf = , N = , MOD = ;
int n,m,f[N][N][N];
char S[N],X[N]; inline void input()
{
freopen ("!.in", "r", stdin);
n=gi(), m=gi();
scanf("%s%s",S+,X+);
} inline void output()
{
freopen ("!.out", "w", stdout);
printf("%d\n",f[n][m][n]);
} inline void cal(int &t,int d) { t+=d; if (t >= MOD) t-=MOD; } inline void solve()
{
int i,j,k,l,tmp;
f[][][]=;
for (i=; i<=n; i++)
for (j=; j<=m; j++)
for (k=; k<=n; k++)
{
tmp=f[i][j][k], l=i+j-k;
if (!tmp || !l || l > m) continue;
if (S[i+] == S[k+])
cal(f[i+][j][k+],tmp);
if (X[j+] == S[k+])
cal(f[i][j+][k+],tmp);
if (S[i+] == X[l+])
cal(f[i+][j][k],tmp);
if (X[j+] == X[l+])
cal(f[i][j+][k],tmp);
}
} int main()
{
input();
solve();
output();
return ;
}

Update

博主终于会把空间优化到n^2辣!!!

PS:记得要清零!!!

 #include <bits/stdc++.h>
#define rg register
#define ll long long
using namespace std; inline int gi()
{
rg int r = ; rg bool b = ; rg char c = getchar();
while (c < '' || c > '') { if (c == '-') b = ; c = getchar(); }
while (c >= '' && c <= '') { r = r * + c - '', c = getchar(); }
if (b) return r; return -r;
} const int inf = , N = , MOD = ;
int n,m,f[][N][N];
char S[N],X[N]; inline void input()
{
n=gi(), m=gi();
scanf("%s%s",S,X);
} inline void cal(rg int &t,rg int d)
{
t+=d;
if (t >= MOD)
t-=MOD;
} inline void solve()
{
rg int i,j,k,p,q,l,r,now,lst;
f[][][]=, now=;
for (k=; k<n+m; ++k) //枚举一共选了多少个,因为每次更新都会多选一个,所以只需枚举到 n+m-1
{
l=max(k-m,), r=min(k,n); //计算 S 管道取珠的数量范围
lst=now, now^=;
for (i=l; i<=r; ++i) //分别枚举序列 x,y
for (j=l; j<=r; ++j)
{
p=k-i, q=k-j; //i,j 表示 S 管道取的数量,p,q表示 X 管道的数量
if (!f[lst][i][j])
continue;
if (S[i] == S[j])
cal(f[now][i+][j+],f[lst][i][j]);
if (S[i] == X[q])
cal(f[now][i+][j],f[lst][i][j]);
if (X[p] == S[j])
cal(f[now][i][j+],f[lst][i][j]);
if (X[p] == X[q])
cal(f[now][i][j],f[lst][i][j]);
f[lst][i][j]=; //每次更新后要记得清零
}
}
printf("%d\n",f[now][n][n]);
} int main()
{
input();
solve();
return ;
}

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

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

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

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

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

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

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

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

    来自FallDream的博客,未经允许,请勿转载,谢谢. n<=500 神题...... 发现这个平方可以看作两个序列相同的对数  然后就可以表示状态了. f[i][j][k]表示两个序列各选了 ...

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

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

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

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

  7. NOI2009 管道取珠 神仙DP

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

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

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

  9. 【题解】NOI2009管道取珠

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

  10. BZOJ1566:[NOI2009]管道取珠——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=1566 https://www.luogu.org/problemnew/show/P1758 题目 ...

随机推荐

  1. TF-IDF学习笔记

    计算文本的权重向量,有个很有效的权重方案:TF-IDF权重策略.TF-IDF含义是词频逆文档频率,指的是,如果某个词或短语在一篇文章中出现的频率高,并且在其他文章中很少出现,则认为此词或短语具有很好的 ...

  2. fastscript增加公共函数

    fastscript增加公共函数 unit fs_BsCommFuncs; interface{$i fs.inc}uses SysUtils, Classes, fs_iclassesrtti, f ...

  3. 渗透测试思路 | Linux下自动化搭建FakeAP,劫持用户在Portal认证下的所有流量

    如何在linux下搭建一个fakeap,使得portal认证下的用户无法发现连接你的假AP,并且能够正常上网.先说一下portal认证.无线WIFI认证方式主要有wpa2 和 open两种,而port ...

  4. 在Intellij上面导入项目 & AOP示例项目 & AspectJ学习 & Spring AoP学习

    为了学习这篇文章里面下载的代码:http://www.cnblogs.com/charlesblc/p/6083687.html 需要用Intellij导入一个已有工程.源文件原始内容也可见:link ...

  5. POJ 1017 Packets(积累)

    [题意简述]:这个是别人的博客,有清晰的题意描写叙述.和解题思路,借助他的想法,能够非常好的解决问题! [分析]:贪心?模拟?见代码 //216K 16Ms #include<iostream& ...

  6. Android自己定义ViewGroup打造各种风格的SlidingMenu

    看鸿洋大大的QQ5.0側滑菜单的视频课程,对于側滑的时的动画效果的实现有了新的认识,似乎打通了任督二脉.眼下能够实现随意效果的側滑菜单了.感谢鸿洋大大!! 鸿洋大大用的是HorizontalScrol ...

  7. 百度地图SDK调试SDKInitializer.initialize(getApplicationContext())错误

    首先描写叙述下问题出现的原因.開始的时候写了一个百度地图SDK的demo来试功能,由于最開始用的是Eclipse自带的AVD来调试,一切正常. 都能够正常验证,可是由于受不了重复的重新启动AVD设备, ...

  8. LINUX下GDB反汇编和调试

    Linux下的汇编与Windows汇编最大的不同就是第一个操作数是原操作数,第二个是目的操作数.而Windows下却是相反. 1. 基本操作指令 简单的操作数类型说明.一般有三种. (1)马上数操作数 ...

  9. iOS学习笔记12-网络(一)NSURLConnection

    一.网络请求 在网络开发中.须要了解一些经常使用的请求方法: GET请求:get是获取数据的意思,数据以明文在URL中传递,受限于URL长度,所以数据传输量比較小. POST请求:post是向serv ...

  10. python(24)- 面向对象进阶

    面向对象基础知识: 1.面向对象是一种编程方式,此编程方式的实现是基于对类和对象的使用: 2.类是一个模板,模板中包装了多个‘函数’供使用(可以将多函数中公用的变量封装到对象中): 3.对象,根据模板 ...