字符串类dp的题目总结
熟练掌握回文串吧,大致有dp或者模拟类的吧
①dp+预处理,懂得如何枚举回文串(一)
②dp匹配类型的题目(二)
③dp+预处理 子串类型 (三)
④字符串的组合数(四)
一:划分成回文串 UVA11584 紫书275 dp+预处理
题目大意:输入一个字符串,把他划分成尽量少的回文串,能划分成几个?
思路:定义dp[i]表示i之间最少能弄成几个回文串,然后枚举1~j(j < i),然后在枚举j的时候判断目前是否为回文串,这样复杂度为n^3.因此我们提前用n^2判断好是否为回文串就行了。回文串的方法就是枚举中心,但是枚举中心也是有技巧的!
lrj老师的代码太牛了!!!第一次见到这么写的!!!具体看代码上的注释吧!!!
// UVa11584 Partitioning by Palindromes
// Rujia Liu
// This code is slightly different from the book.
// It uses memoization to judge whether s[i..j] is a palindrome.
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std; const int maxn = + ;
int n, kase, vis[maxn][maxn], p[maxn][maxn], d[maxn];
char s[maxn]; int is_palindrome(int i, int j) {
if(i >= j) return ;
if(s[i] != s[j]) return ;
//p[i][j]定义的是i到j是否为回文串
if(vis[i][j] == kase) return p[i][j];
vis[i][j] = kase;//我去,还能这么玩,在dp的过程中预处理,然后用kase区分
p[i][j] = is_palindrome(i+, j-);
return p[i][j];
} int main() {
int T;
scanf("%d", &T);
memset(vis, , sizeof(vis));
for(kase = ; kase <= T; kase++) {
scanf("%s", s+);
n = strlen(s+);
d[] = ;
for(int i = ; i <= n; i++) {
d[i] = i+;
for(int j = ; j < i; j++)
if(is_palindrome(j+, i)) d[i] = min(d[i], d[j] + );
}
printf("%d\n", d[n]);
}
return ;
}
然后我的for循环和lrj老师的不一样
//看看会不会爆int!数组会不会少了一维!
//取物问题一定要小心先手胜利的条件
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define ALL(a) a.begin(), a.end()
#define pb push_back
#define mk make_pair
#define fi first
#define se second
const int maxn = + ;
char atlas[maxn];
int dp[maxn];
bool p[maxn][maxn], vis[maxn][maxn]; bool dfs(int i, int j){
if (i == j || j < i) return true;
if (atlas[i] != atlas[j]) return false;
if (vis[i][j]) return p[i][j];
vis[i][j] = true;
p[i][j] = dfs(i + , j - );
return p[i][j];
} int main(){
int t; cin >> t;
while (t--){
memset(vis, , sizeof(vis));
scanf("%s", atlas + );
int len = strlen(atlas + );
for (int i = ; i <= len; i++){
for (int j = ; j < i; j++){
p[i][j] = dfs(j, i);
}
}
memset(dp, 0x3f, sizeof(dp));
dp[] = ;
for (int i = ; i <= len; i++){
dp[i] = dp[i - ] + ;//我定义目前这个字符单独组成回文串
for (int j = ; j <= i; j++){
if (p[i][j]){
dp[i] = min(dp[i], dp[j - ] + );
}
}
}
printf("%d\n", dp[len]);
}
return ;
}
学习之处:回文串有两种,一种是abba,另一种是cbabc。然后如何枚举这两种本来是应该要分类讨论的,我的想法是枚举i-j,然后用n^2的想法
二:括号序列 UVA1626 紫书278
题目大意:给你几个合法的串的定义,只包含()[]这四个符号,加上多少的(、)、[、]能使给定的串变成一个合法的串
思路:定义dp[i][j]表示从i~j成功匹配所需要添加的最少的符号的个数使之变成合法的串。
首先我们根据dfs来,可以发现,如果是dfs(i, j),我们要尝试在每一种i~j中进行分割,然后如果i和j是一个合法的串,那么就直接dfs(i-1, j-1),不然就进行分割,然后记得,当最后只有一个符号的时候就假定要再加一个字符即可。
//看看会不会爆int!数组会不会少了一维!
//取物问题一定要小心先手胜利的条件
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define ALL(a) a.begin(), a.end()
#define pb push_back
#define mk make_pair
#define fi first
#define se second
const int inf = 0x3f3f3f3f;
const int maxn = + ;
char s[maxn];
int dp[maxn][maxn];//从最右边i开始,到j匹配成功的最小的数目 bool match(int i, int j){
if (s[i] == '(' && s[j] == ')') return true;
if (s[i] == '[' && s[j] == ']') return true;
return false;
} void display(int i, int j){
if (i == j) {
if (s[i] == '(' || s[i] == ')') printf("()");
else printf("[]");
return ;
}
int ans = dp[i][j];
if (match(i, j) && ans == dp[i + ][j - ]){
printf("%c", s[i]);
display(i + , j - );
printf("%c", s[j]);
return ;
}
for (int k = i; k <= j - ; k++){
if (ans == dp[i][k] + dp[k + ][j]){
display(i, k); display(k + , j);
return ;
}
}
} void readline(char* S) {
fgets(S, maxn, stdin);
} int main(){
int T;
readline(s); sscanf(s, "%d", &T); readline(s);
while (T--){
readline(s);
int n = strlen(s) - ;
for (int i = ; i <= n; i++){
dp[i][i] = ;
}
for (int i = n - ; i >= ; i--){
for (int j = i + ; j <= n; j++){
dp[i][j] = inf;
if (match(i, j)) dp[i][j] = min(dp[i][j], dp[i + ][j - ]);
for (int k = i; k <= j - ; k++){//如果从k=i+1开始枚举的话,就需要我下面的两句话,但是弊端就是输出的时候要多很多的条件。不过从i开始枚举的话就不需要了
//if (match(i, k)) dp[i][j] = min(dp[i][j], dp[i + 1][k - 1] + dp[k + 1][j]);
//if (match(k, j)) dp[i][j] = min(dp[i][j], dp[i][k - 1] + dp[k + 1][j - 1]);
///上面两句话有没有无所谓,因为你的j是会枚举到你这些列举的情况的
dp[i][j] = min(dp[i][j], dp[i][k] + dp[k + ][j]);
}
}
}
display(, n);
printf("\n");
if(T) printf("\n");
readline(s);
//printf("%d\n", dp[0][n]);
}
return ;
}
学习之处:定义的学习,dp边界的找寻通过dfs来进行的,如何路径输出
三:http://codeforces.com/contest/477/problem/C codeforces 272 div1 C
题目大意:给你字符串s和p,从s删除0~|s|个字符,最多能形成多少个不重叠的字串p
思路:定义dp[i][j]表示前i个字符删除j个,定义cal(i)表示从第i个开始往前数,删除几个才能组合成一个子串p。
//看看会不会爆int!数组会不会少了一维!
//取物问题一定要小心先手胜利的条件
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define ALL(a) a.begin(), a.end()
#define pb push_back
#define mk make_pair
#define fi first
#define se second
const int maxn = + ;
char s[maxn], p[maxn];
int lens, lenp, pos;
int dp[maxn][maxn];//对当前位置i,删除j个能产生的最大匹配数 int cal(int ps){
int lp = lenp;
int cnt = ;
for (int i = ps; i >= ; i--){
if (s[i] == p[lp]){
lp--;
if (lp == ){
pos = i;
return cnt;
}
}
else cnt++;
}
} int main(){
scanf("%s", s + ); lens = strlen(s + );
scanf("%s", p + ); lenp = strlen(p + ); for (int i = lenp; i <= lens; i++){
int cnt = cal(i);
for (int j = ; j < i; j++){
dp[i][j] = max(dp[i][j], dp[i - ][j]);
if (j >= cnt && pos - >= j - cnt){
dp[i][j] = max(dp[i][j], dp[pos - ][j - cnt] + );
}
}
}
for (int i = ; i <= lens; i++){
printf("%d%c", dp[lens][i], i == lens ? '\n' : ' ');
}
return ;
}
四:http://codeforces.com/contest/476/problem/B codeforces 272 div2 B
题目大意:给你两个串,串1是由+或-组合而成,串二是由+、-、?组合而成的。?可以随便改变成+或-,问有多少的概率能使得两个串的+和-数目相等
思路:先求出串1串二中的+-?的数目,然后让串一的+-减去串二的+-,如果小于0就直接概率为0,反之进行dp即可。定义dp[i][j],表示一共有i+j个?,生成i个+,j个-。
//看看会不会爆int!数组会不会少了一维!
//取物问题一定要小心先手胜利的条件
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define ALL(a) a.begin(), a.end()
#define pb push_back
#define mk make_pair
#define fi first
#define se second
const int maxn = + ;
char s1[maxn], s2[maxn];
double dp[maxn][maxn]; int id(char c){
if (c == '+') return ;
if (c == '-') return ;
if (c == '?') return ;
} int main(){
scanf("%s%s", s1, s2);
int len = strlen(s1);
for (int i = ; i < len; i++){
s1[i] = id(s1[i]);
s2[i] = id(s2[i]);
}
sort(s1, s1 + len);
sort(s2, s2 + len);
int t10 = upper_bound(s1, s1 + len, ) - lower_bound(s1, s1 + len, );
int t11 = upper_bound(s1, s1 + len, ) - lower_bound(s1, s1 + len, ); int t20 = upper_bound(s2, s2 + len, ) - lower_bound(s2, s2 + len, );
int t21 = upper_bound(s2, s2 + len, ) - lower_bound(s2, s2 + len, );
int t22 = upper_bound(s2, s2 + len, ) - lower_bound(s2, s2 + len, ); t10 -= t20, t11 -= t21;
if (t10 < || t11 < ){
printf("0.000000000000\n");
return ;
}
double ans = ;
dp[][] = 1.0;
for (int i = ; i <= t22; i++){//有i个
for (int j = ; j <= i; j++){//有j个+
int l = i - j;
if (j == ) dp[j][l] = dp[j][l - ] * 0.5;
else if (l == ) dp[j][l] = dp[j - ][l] * 0.5;
else dp[j][l] = max(dp[j - ][l] * 0.5 + dp[j][l - ] * 0.5, dp[j][l]);
}
}
printf("%.12f\n", dp[t10][t11]);
return ;
}
五:
六:
七:
字符串类dp的题目总结的更多相关文章
- Core Java 总结(字符和字符串类问题)
所有代码均在本地编译运行测试,环境为 Windows7 32位机器 + eclipse Mars.2 Release (4.5.2) 2016-10-17 整理 字符,字符串类问题 正则表达式问题 J ...
- 编码实现字符串类CNString实现运算符重载
题目描述: 编码实现字符串类CNString,该类有默认构造函数.类的拷贝函数.类的析构函数及运算符重载,需实现以下"="运算符."+"运算."[]& ...
- 动态规划——区间DP,计数类DP,数位统计DP
本博客部分内容参考:<算法竞赛进阶指南> 一.区间DP 划重点: 以前所学过的线性DP一般从初始状态开始,沿着阶段的扩张向某个方向递推,直至计算出目标状态. 区间DP也属于线性DP的一种, ...
- SDOI2010代码拍卖会 (计数类DP)
P2481 SDOI2010代码拍卖会 $ solution: $ 这道题调了好久好久,久到都要放弃了.洛谷的第五个点是真的强,简简单单一个1,调了快4个小时! 这道题第一眼怎么都是数位DP,奈何数据 ...
- CH5E26 扑克牌 (计数类DP)
$ CH~5E26~\times ~ $ 扑克牌: (计数类DP) $ solution: $ 唉,计数类DP总是这么有套路,就是想不到. 这道题我们首先可以发现牌的花色没有价值,只需要知道每种牌有 ...
- 迷宫类dp整合
这是迷宫类dp我自己取的名字,通常比较简单,上货 简单模型 数字三角形 状态表示:f[i][j]表示起点第\(i\)行第\(j\)个数最短路径的长度 状态转移:\(f[i][j] = max(f[i ...
- Java:字符串类String的功能介绍
在java中,字符串是一个比较常用的类,因为代码中基本上处理的很多数据都是字符串类型的,因此,掌握字符串类的具体用法显得很重要了. 它的主要功能有如下几种:获取.判断.转换.替换.切割.字串的获取.大 ...
- Qt入门-字符串类QString
原地址:http://blog.csdn.net/xgbing/article/details/7770854 QString是Unicode字符的集合,它是Qt API中使用的字符串类. QStri ...
- 可变字符串类 StringBuilder
string类创建的字符串是不可变的(同一内存中),每更改一次,就会新开辟内存,不利于高效频繁操作. 当频繁操作同一字符串变量时,建议使用StringBuilder. 可变字符串类StringBuil ...
随机推荐
- [C++] OOP - Virtual Functions and Abstract Base Classes
Ordinarily, if we do not use a function, we do not need to supply a definition of the function. Howe ...
- UVALive 3668 A Funny Stone Game(博弈)
Description The funny stone game is coming. There are n piles of stones, numbered with 0, 1, 2,..., ...
- 《C++面试知识点》
[动态内存] 1. 由内置指针管理的动态内存(即new和delete管理动态内存),直到被显式释放之前它都是存在的.假设该指针变量被销毁,那该内存将不会自动释放(即所谓的“内存泄漏”). 2. 可以用 ...
- RabbitMQ安装与初始配置【转载】
Erlang安装 rabbitmq依赖于Erlang,需先安装,推荐安装rabbitmq/erlang-rpm: #clone源码 git clone https://github.com/rabbi ...
- vue-cli2使用cdn方式引入cytoscape
1. index.html头部引用 <script src="https://cdnjs.cloudflare.com/ajax/libs/cytoscape/3.2.19/cytos ...
- 【python】Python: Enum枚举的实现
从C#系语言过来用Python,好不容易适应了写代码不打花括号,突然有一天发现它居然木有枚举……于是stackoverflow了一把,发现神人的枚举(enum)实现到处都是,于是汉化总结过来. 如果是 ...
- 了解Solr
Solr 简 介 采用Java开发,基于Lucene的全文搜索服务器.同时对其进行了扩展,提供了比Lucene更为丰富的查询语言,同时实现了可配置.可扩展并对查询性能进行了优化,并且提供了一个完善的功 ...
- JVM启动参数详解 (转)
非标准参数 非标准参数又称为扩展参数,其列表如下: -Xint 设置jvm以解释模式运行,所有的字节码将被直接执行,而不会编译成本地码. -Xbatch 关闭后台代码编译,强制在前台编译,编译 ...
- 如何提升集群资源利用率? 阿里容器调度系统Sigma 深入解析
阿里妹导读:为了保证系统的在线交易服务顺利运转,最初几年,阿里都是在双11大促来临之前大量采购机器储备计算资源,导致了双11之后资源大量闲置点现象.是否能把计算任务与在线服务进行混合部署,在现有弹性资 ...
- 洛谷3805:【模板】manacher算法——题解
https://www.luogu.org/problemnew/show/P3805 给出一个只由小写英文字符a,b,c...y,z组成的字符串S,求S中最长回文串的长度. 字符串长度为n 板子题, ...