Noip2001 提高组 T3
T3
题目描述
给出一个长度不超过200的由小写英文字母组成的字母串(约定;该字串以每行20个字母的方式输入,且保证每行一定为20个)。要求将此字母串分成k份(1<k<=40),且每份中包含的单词个数加起来总数最大(每份中包含的单词可以部分重叠。当选用一个单词之后,其第一个字母不能再用。例如字符串this中可包含this和is,选用this之后就不能包含th)。
单词在给出的一个不超过6个单词的字典中。
要求输出最大的个数。
输入输出格式
输入格式:
每组的第一行有二个正整数(p,k)
p表示字串的行数;
k表示分为k个部分。
接下来的p行,每行均有20个字符。
再接下来有一个正整数s,表示字典中单词个数。(1<=s<=6)
接下来的s行,每行均有一个单词。
输出格式:
一个整数,分别对应每组测试数据的相应结果。
输入输出样例
1 3
thisisabookyouareaoh
4
is
a
ok
sab
7
说明
this/isabookyoua/reaoh
思路:
我们可以用DP+KMP来处理这道题
首先需要知道题目要求是:
不能以同一个开头生成两个单词
千万别理解错意思、、、
由于在拆分字符串的过程中,如果以某位置为首某个较短单词被截断,那么以该位置为首的较长单词必然也会被截断。
也就是说,对于各个位置来说我们选取较短的单词总不会比选取较长的单词所形成的单词少。
那么题目就转化成了在每一个位置上选取能够形成的最短的单词,如果不能形成则为0,直接continue即可
DP转移方程也是很好想的:
if(dp[x][j-1])
dp[i][j]=max(dp[i][j],dp[x][j-1]+KMP(x+1,i));//其中x是我们枚举的划分位置的前一个字符
坑点:
注意题目要求的读入方式
上代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int L1 = ;
const int L2 = ;
int len;
char a[L1];
void init(int T) {
char c;
while(T--) {
for(int i=; i<; i++) {
scanf(" %c",&c);
a[len++]=c;
}
}
} char d[L2][L1]; //存字典
int sum,cnt;
//sum:统计一段单词中可以匹配的总个数
//cnt:字典单词总个数
int nxt[L2][L1],nowx[L2],dl[L2];
//nxt[]:开两维的原因是因为----不止一个模式串要跟文本串进行匹配
//nowx[]:每个单词当前在KMP函数中匹配的位数
//dl[]:字典中每个单词的长度
bool vis[L1]; //判重
int KMP(int l,int r) {
sum=;
memset(vis,false,sizeof(vis));
memset(nowx,,sizeof(nowx));
for(int i=l; i<=r; i++)
for(int j=; j<cnt; j++) {
while(nowx[j] && a[i]!=d[j][nowx[j]])
nowx[j]=nxt[j][nowx[j]];
if(a[i]==d[j][nowx[j]]) nowx[j]++;
if(nowx[j]==dl[j] && !vis[i-dl[j]+]) { //匹配成功
vis[i-dl[j]+]=true; //标记开头
sum++;
}
}
return sum;
} int dp[L1][L1];
int main() {
int p,k,x;
scanf("%d%d",&p,&k);
init(p);
scanf("%d",&cnt);
for(int i=; i<cnt; i++) {
scanf("%s",d[i]);
dl[i]=strlen(d[i]);
}
for(int i=; i<cnt; i++) {
x=;
nxt[i][x]=nxt[i][x+]=;
for(int j=; j<dl[i]; j++) {
while(x && d[i][x]!=d[i][j]) x=nxt[i][x];
nxt[i][j+]=d[i][x] == d[i][j] ? ++x : ;
}
}
for(int i=; i<len; i++) {
for(int j=; j<=k; j++) {
if(j==) dp[i][j]=KMP(,i);
else {
for(x=; x<i; x++)
if(dp[x][j-])
dp[i][j]=max(dp[i][j],dp[x][j-]+KMP(x+,i));
}
}
}
printf("%d",dp[len-][k]);
return ;
}
Noip2001 提高组 T3的更多相关文章
- 洛谷 P1025 & [NOIP2001提高组] 数的划分(搜索剪枝)
		题目链接 https://www.luogu.org/problemnew/show/P1025 解题思路 一道简单的dfs题,但是需要剪枝,否则会TLE. 我们用dfs(a,u,num)来表示上一个 ... 
- JZOJ2020年8月11日提高组T3 页
		JZOJ2020年8月11日提高组T3 页 题目 Description 战神阿瑞斯听说2008年在中华大地上,将举行一届规模盛大的奥林匹克运动会,心中顿觉异常兴奋,他想让天马在广阔的天空上,举行一场 ... 
- 【GDKOI2014】JZOJ2020年8月13日提高组T3 壕壕的寒假作业
		[GDKOI2014]JZOJ2020年8月13日提高组T3 壕壕的寒假作业 题目 Description Input Output 输出n行.第i行输出两个整数,分别表示第i份作业最早完成的时刻以及 ... 
- JZOJ2020年8月10日提高组T3 玩诈欺的小杉
		JZOJ2020年8月10日提高组T3 玩诈欺的小杉 题目 Description 是这样的,在小杉的面前有一个N行M列的棋盘,棋盘上有\(N*M\)个有黑白棋的棋子(一面为黑,一面为白),一开始都是 ... 
- 【佛山市选2013】JZOJ2020年8月7日提高组T3 海明距离
		[佛山市选2013]JZOJ2020年8月7日提高组T3 海明距离 题目 描述 对于二进制串a,b,他们之间的海明距离是指两个串异或之后串中1的个数.异或的规则为: 0 XOR 0 = 0 1 XOR ... 
- 【NOIP2015模拟11.5】JZOJ8月5日提高组T3 旅行
		[NOIP2015模拟11.5]JZOJ8月5日提高组T3 旅行 题目 若不存在第\(k\)短路径时,输出"Stupid Mike" 题解 题意 给出一个有\(n\)个点的树 问这 ... 
- [NOIP2001提高组]CODEVS1014 Car的旅行路线(最短路)
		最短路,这个不难想,但是要为它加边就有点麻烦..还好写完就过了(虽然WA了一次,因为我调试用的输出没删了..),不然实在是觉得挺难调的.. ------------------------------ ... 
- 【NOIP2013提高组T3】加分二叉树
		题目描述 设一个n个节点的二叉树tree的中序遍历为(1,2,3,…,n),其中数字1,2,3,…,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di,tree及它的每个子树都 ... 
- noip1998   提高组t3   挖地雷
		题目背景 NOIp1996提高组第三题 题目描述 在一个地图上有N个地窖(N<=20),每个地窖中埋有一定数量的地雷.同时,给出地窖之间的连接路径.当地窖及其连接的数据给出之后,某人可以从任一处 ... 
随机推荐
- 通过命令行新建qt项目,并编译打包
			ref: https://blog.csdn.net/weixin_42837024/article/details/81945656 平台 :Ubuntu 18.04 QT版本 :5.9.1 ( ... 
- (九) spring 使用自定义限定符注解
			案例一 定义接口 CD.java package interfacepackage; public interface CD { void play(); } 定义接口 player .java p ... 
- java封装数据类型——Long
			Long 是长整型 long 的封装数据类型.我们知道 long 相对于 int 的差异就是数据表示的范围扩大了,其它大部分特性都是一样的.所以 Long 跟 Integer 大部分方法都是相同的. ... 
- js之拖拽事件
			js之拖拽事件 api:https://www.runoob.com/jsref/event-ondrag.html 拖拽事件是js原生的事件,使用时在div上添加 draggable="t ... 
- Java 面向对象(七)多态
			一.多态概述(Polymorphism) 1.引入 多态是继封装.继承之后,面向对象的第三大特性. 通过不同的事物,体现出来的不同的形态.多态,描述的就是这样的状态.如跑的动作,每个动物的跑的动作就是 ... 
- liunx mkisofs 命令的使用(制作iso)
			参考的博客 http://www.cnblogs.com/darkknightzh/p/8564483.html 有很多时候需要在liunx 环境中将文件打成 iso 所有很多时候就会用到这个命令(m ... 
- vue——store全局存储
			业务场景:刷新页面时,首次拉取所有配置,存储到store状态管理用于全局调用: import Vue from 'vue' import Vuex from 'vuex' import userInf ... 
- html5+css3 background-clip 技巧
			<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ... 
- MVC方式显示数据(数据库)
			新建实体数据模型 选择ADO.NET实体数据模型,名称改为数据库名 因为使用现有数据库,所以选择来自数据库的EF设计器,只演示所以只选择一个表,空模型可后期增加表 选择从数据库更新模型 新建数据库连接 ... 
- java - day008 -  接口,内部类
			接口 作用: 结构设计工具,用来解耦合,需要有子类,隔离具体实现 接口是一个极端的抽象类 用 interface 代替 class 用 implements 代替 extends // 接口中所有东西 ... 
