POJ 3276 The Cow Lexicon DP 难度: 0
题目
http://poj.org/problem?id=3267
题意
有长为L的字符串(L<=600),现在问最少去掉多少个字符,使得字符串由字典中的词不重叠地构成。
字典中含有k(k<=300)个词,每个词长度最多为25
思路
最粗暴的想法:假设当前已经匹配到i为第k个单词的结尾,向前找到第k个单词最早可以开始匹配的位置j,令dp为记载从字符串开始到当前位置完成字典匹配后去掉字符数目的最小值,那么明显dp[i] =dp[j] + i - j - words[k].size()
感想
- TLE,原因是复杂度为O(300 * 300 * 600 * 25),所以后来直接用match从后往前匹配,去掉一重循环
- WA: 原因是忽略了字母匹配之后要向下走一个,恰好实例中单词又没有重复的字母。
Code
Mem:712K Time:1610MS
#include <iostream>
#include <cstring>
#include <queue>
using namespace std;
int w, l;
string words[601];
char msg[301];
int dp[301];
int start_inds[601];
int match(int end_ind, string word) {
int k = end_ind - 1;
for (int i = (int)word.size() - 1; i >= 0; i--, k--) {
while(k >= 0 && msg[k] != word[i]) k--;
if (i == 0)return k;
}
return -1;
}
int main() {
cin >> w >> l;
cin >> msg;
for (int i = 0; i <= l; i++) {
dp[i] = i;
}
for (int i = 0; i < w; i++) {
cin >> words[i];
}
for (int i = 0; i <= l; i++) {
for (int j = 0; j < i; j++) {
dp[i] = min(dp[i], dp[j] + i - j);
}
for (int k = 0; k < w; k++) {
if (words[k].size() > i)continue;
int largest_start_ind = match(i, words[k]);
start_inds[k] = largest_start_ind;
if (largest_start_ind >= 0 && dp[i] > dp[largest_start_ind] + i - largest_start_ind - (int)words[k].size()) {
dp[i] = dp[largest_start_ind] + i - largest_start_ind - (int)words[k].size();
}
}
}
cout << dp[l] << endl;
return 0;
}
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET 9 new features-C#13新的锁类型和语义
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
· 你所不知道的 C/C++ 宏知识
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· 不到万不得已,千万不要去外包
· C# WebAPI 插件热插拔(持续更新中)
· 会议真的有必要吗?我们产品开发9年了,但从来没开过会
· 如何打造一个高并发系统?
· 【译】我们最喜欢的2024年的 Visual Studio 新功能
2017-02-03 POJ 3436 ACM Computer Factory 最大流,拆点 难度:1