hdu5745--La Vie en rose (DP+bitset)
好题,学到新姿势!
题意:给两个字符串 a 和 b ,b可以进行变换,规则是可以任意交换相邻两个字符的位置,但是不可以有交叉(例如3和4交换,5和6交换 互不影响,但是2和3,3和4就不可以)。求a中每一个位置能不能匹配b或b变换得到的子串。
题解:考虑dp。dp[i][j][k]表示a[i]和b[j]匹配,k为1表示j未做交换,k=0表示j和j-1进行交换,k=2表示j和j+1进行交换。
直接DP会爆内存。可以想到使用滚动数组,因为递推时只与前一个状态有关。
但是加了滚动数组还是会T。
其实O(N*M)的复杂度,T了倒也很正常,但是据说各种姿势的暴力都能过,为什么这就过不去>_<
bitset优化dp
---
学习一下bitset。
可以当作一个bool型数组考虑,bitset<N> bs; 可以考虑成一个数组bool bs[N]。
相关操作:
bs.set(); 全部置1,bs.reset()全部置0;
bs.set(pos);等价于bs[pos]=1,bs.reset(pos)等价于bs[pos]=0;
最重点的来了,bitset<N> a, b;
a和b可以直接进行操作,
!a //按位取反
a^b //按位异或
a|b //按位或
a&b //按位与
a=b<<3 //整体移位
a.count(); //a中1的个数
bitset有什么用呢(也许还有其他的用处,这里讲的是本题用到的)
如果有一个bool数组 a[N] 和b[N] 把每一个位异或的话,一定是
for (int i = 0; i < N; ++i) c[i] = a[i] ^ b[i];
但是如果用bitset直接a^b的话,只需要O(N/机器字节数)
这样可以实现常数优化。
---
这和这道题有什么关系呢?
观察dp方程,
dp[i][j][] = ((dp[i-][j-][] || dp[i-][j-][]) && a[i] == b[j]);
先考虑一个,其余两个同理。
可以发现dp[i]只与dp[i-1]有关。
代码大概是这个样子的
for (int j = ; j < lb; ++j) {
for (int i = ; i < la; ++i) {
dp[i][j][] = ((dp[i-][j-][] || dp[i-][j-][]);
}
}
那么把第一位压缩,dp数组定义为bitset<N> dp[M][3] ,就可以把dp方程写成dp[j][1] = ((dp[j-1][1] | dp[j-1][0]) << 1);
这里左移是因为这里i-1求出的是i。
但是a[i]==b[j]要怎么处理呢?
因为前面是用一个bitset处理的,我们希望把a[i]==b[j]也化成一个bitset的形式,也就是对于每一个j,有一个bitset<N> bs,bs[i]其中来表示a[i]==b[j]的值,但是这样又是N*M,明显不可能的,其实不需要对于每一个j都有一个bitset,因为一共有26个字母,于是只需对每一个字母求出,即bitset<N>26,然后使用b[j]那个的字母bitset就可以了。
至于滚动数组就很好实现了,
这题正解反而卡常数卡的厉害,姿势不对很容易超时,没人性!! T^T
AC代码(3166MS)
#include <cstdio>
#include <bitset>
#include <iostream>
#include <cstring>
using namespace std;
const int N = ;
const int M = ;
char a[N];
char b[M];
//bool dp[N][M][3]; 0和前面的交换 1不交换 2和后面的交换
bitset<N> dp[][];
bitset<N> w[]; // 记录对于每个字母 p[i]是否为这个字母
int main()
{
//freopen("in", "r", stdin);
int T;
int la, lb;
scanf("%d", &T);
while (T--) {
scanf("%d%d", &la, &lb);
scanf("%s%s", a, b); for (int i = ; i < ; ++i) w[i].reset();
for (int i = ; i < la; ++i) w[a[i]-'a'][i] = ;
for (int i = ; i < ; ++i) for (int j = ; j < ; ++j) dp[i][j].reset();
dp[][] = w[b[]-'a'];
if (lb > ) dp[][] = w[b[]-'a']; int now = ;
for (int j = ; j < lb; ++j) {
now ^= ;
dp[now][] = ((dp[now^][]) << ) & w[b[j-]-'a'];
dp[now][] = ((dp[now^][] | dp[now^][]) << ) & w[b[j]-'a'];
if (j < lb - ) dp[now][] = ((dp[now^][] | dp[now^][]) << ) & w[b[j+]-'a'];
} for (int i = ; i < la; ++i) {
if (dp[now][][i+lb-] || dp[now][][i+lb-]) printf("");
else printf("");
}
puts(""); }
return ;
}
hdu5745--La Vie en rose (DP+bitset)的更多相关文章
- hdu 5745 La Vie en rose DP + bitset优化
http://acm.hdu.edu.cn/showproblem.php?pid=5745 这题好劲爆啊.dp容易想,但是要bitset优化,就想不到了. 先放一个tle的dp.复杂度O(n * m ...
- hdu5745 La Vie en rose 巧妙地dp+bitset优化+滚动数组减少内存
/** 题目:hdu5745 La Vie en rose 链接:http://acm.hdu.edu.cn/showproblem.php?pid=5745 题意:题目给出的变换规则其实就是交换相邻 ...
- HDU 5745 La Vie en rose (DP||模拟) 2016杭电多校联合第二场
题目:传送门. 这是一道阅读理解题,正解是DP,实际上模拟就能做.pij+1 指的是 (pij)+1不是 pi(j+1),判断能否交换输出即可. #include <iostream> # ...
- HDU 5745 La Vie en rose 暴力
La Vie en rose 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5745 Description Professor Zhang woul ...
- HDU 5745 La Vie en rose
La Vie en rose Time Limit: 14000/7000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)T ...
- hdu 5745 La Vie en rose(2016多校第二场)
La Vie en rose Time Limit: 14000/7000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)T ...
- La Vie en rose (模拟)
#include<bits/stdc++.h> using namespace std; ; ; int T, n, m; char str1[maxm], str2[maxn]; int ...
- HDU5716, HDU5745【dp+bitset】
DP+bitset HDU5716 dp[i][j] = dp[i-1][j-1] && (s[i] in set[j]); 第二维压bitset #include <bits ...
- 字符串匹配dp+bitset,滚动数组优化——hdu5745(经典)
bitset的经典优化,即把可行性01数组的转移代价降低 bitset的适用情况,当内层状态只和外层状态的上一个状态相关,并且内层状态的相关距离是一个固定的数,可用bitset,换言之,能用滚动数组是 ...
随机推荐
- POSTMAN 数据关联
概述 在使用postman测试接口是,我们可能需要先获取一个token,然后再将这个token发送到第二个请求.这个需要做postman的关联,一次性完成这两个测试. 实现方法 1.编写两个控制器方法 ...
- JDK线程池的使用
转载自:https://my.oschina.net/hosee/blog/614319: 摘要: 本系列基于炼数成金课程,为了更好的学习,做了系列的记录. 本文主要介绍: 1. 线程池的基本使用 2 ...
- 富文本粘贴word文档内容图片处理
公司做的项目要用到文本上传功能. 网上找了很久,大部分都有一些不成熟的问题,终于让我找到了一个成熟的项目. 下面就来看看: 1.打开工程: 对于文档的上传我们需要知道这个项目是否符合我们的初衷. 运行 ...
- Gitolite 权限控制
官网 http://gitolite.com/gitolite/index.html 安装配置 http://gitolite.com/gitolite/install/ 傻瓜安装教程 http:// ...
- ibatis.net之我的调整:Update语句的动态set字段
动态Update语句 如果Map中指定多个字段 <update id="UpdateStaff"> update Staff set Name=#Name#,InDat ...
- Android学习指南之三十八:Android手势操作编程[转]
手势操作在我们使用智能设备的过程中奉献了不一样的体验.Android开发中必然会进行手势操作方面的编程.那么它的原理是怎样的呢?我们如何进行手势操作编程呢? 手势操作原理 首先,在Android系统中 ...
- noip第5课作业
1. 计算税收 [问题描述] 对某产品征收税金,在产值1万元以上收税5%:在1万元以下但在5000元或者以上的征收税3%:在5000元以下但在1000元或以上征收税2%:1000元以下的免收税 ...
- webView自适应及缩放
WebView wv=(WebView) findViewById(R.id.webView); wv.setVisibility(WebView.VISIBLE); WebSettings ws = ...
- libtool 创建库的工具
libtool 创建库的工具 1. 背景 在不同的系统中建立动态链接库的方法有很大的差别,这主要是因为每个系统对动态链接库的用法和实现并不相同,以及编译器对动态链接库支持的选项也不太一样. 对于开发人 ...
- hdu 1695 GCD 【莫比乌斯函数】
题目大意:给你 a , b , c , d , k 五个值 (题目说明了 你可以认为 a=c=1) x 属于 [1,b] ,y属于[1,d] 让你求有多少对这样的 (x,y)满足gcd(x,y)= ...