题面在这里

description

输入两个长度分别为\(n\)和\(m\)的颜色序列,要求按顺序合并成同一个序列,即每次可以把一个序列开头的颜色放到新序列的尾部。

对于每个颜色\(c\)来说,其跨度\(L(c)\)等于最大位置和最小位置之差。

你的任务是找一种合并方式,使得所有\(L(c)\)的总和最小。

——摘自《算法竞赛入门经典(第2版)》,刘汝佳 著

data range

\[n,m\le 5000
\]

solution

序列合并问题,使用\(f[i][j]\)表示第一个序列前\(i\)个颜色和第二个序列前\(j\)个颜色合并之后的最小贡献

转移时每次往后添加两个序列首端的其中一个字符;

虽然通过这种状态无法得知每种颜色在这些方案中的始末位置,但我们仍然可以计算答案——未来费用的动态规划问题,提前计算贡献!

即每次往序列尾端添上一个颜色的时候,我们可以提前加上仍未终结的颜色的贡献

我们使用辅助数组\(g[i][j]\)表示第一个序列前\(i\)个颜色和第二个序列前\(j\)个颜色合并之后还未结束的颜色数量,于是可以这样转移:

\[f[i][j]=min\{f[i-1][j]+g[i-1][j],f[i][j-1]+g[i][j-1]\}
\]

答案存储在\(f[n][m]\),做完啦

code

#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<iomanip>
#include<cstring>
#include<complex>
#include<vector>
#include<cstdio>
#include<string>
#include<bitset>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<set>
//#define TEST
#define FILE "a"
#define mp make_pair
#define pb push_back
#define RG register
#define il inline
using namespace std;
typedef unsigned long long ull;
typedef vector<int>VI;
typedef long long ll;
typedef double dd;
const int inf=1e9+7;
const int mod2=998244353;
const int rev2=332748118;
const int mod1=1e9+7;
const int N=5010;
const dd eps=1e-10;
const ll INF=1e18;
il ll read(){
RG ll data=0,w=1;RG char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
if(ch=='-')w=-1,ch=getchar();
while(ch<='9'&&ch>='0')data=data*10+ch-48,ch=getchar();
return data*w;
} il void file(){
freopen(FILE".in","r",stdin);
freopen(FILE".out","w",stdout);
} int n,m,g[N][N],f[N][N],tot[400],cnt[400];
char a[N],b[N];
il int solve(){
memset(tot,0,sizeof(tot));
for(RG int i=1;i<=n;i++)tot[a[i]]++;
for(RG int i=1;i<=m;i++)tot[b[i]]++;
for(RG int i=0;i<=n;i++){
memset(cnt,0,sizeof(cnt));
g[i][0]=0;
for(RG int j=1;j<=i;j++){
if(!cnt[a[j]])g[i][0]++;
cnt[a[j]]++;
if(cnt[a[j]]==tot[a[j]])g[i][0]--;
}
for(RG int j=1;j<=m;j++){
g[i][j]=g[i][j-1];
if(!cnt[b[j]])g[i][j]++;
cnt[b[j]]++;
if(cnt[b[j]]==tot[b[j]])g[i][j]--;
}
}
f[0][0]=0;
for(RG int i=0;i<=n;i++)
for(RG int j=0;j<=m;j++){
if(i||j)f[i][j]=2147484647/2;
if(i)f[i][j]=min(f[i][j],f[i-1][j]+g[i-1][j]);
if(j)f[i][j]=min(f[i][j],f[i][j-1]+g[i][j-1]);
}
return f[n][m];
} int main()
{
RG int T=read();
while(T--){
scanf("%s%s",a+1,b+1);n=strlen(a+1);m=strlen(b+1);
printf("%d\n",solve());
}
return 0;
}

[UVA1625]Color Length的更多相关文章

  1. [水题日常]UVA1625 Color Length

    来整理一下思路- 一句话题意:给两个大写字母的序列,每次取出其中一个数列的第一个元素放到新序列里面,对每个字母\(c\)记它的跨度\(L(c)\)为这个字母最后出现的位置-第一次出现的位置,求新序列所 ...

  2. UVA - 1625 Color Length[序列DP 代价计算技巧]

    UVA - 1625 Color Length   白书 很明显f[i][j]表示第一个取到i第二个取到j的代价 问题在于代价的计算,并不知道每种颜色的开始和结束   和模拟赛那道环形DP很想,计算这 ...

  3. UVA - 1625 Color Length[序列DP 提前计算代价]

    UVA - 1625 Color Length   白书 很明显f[i][j]表示第一个取到i第二个取到j的代价 问题在于代价的计算,并不知道每种颜色的开始和结束   和模拟赛那道环形DP很想,计算这 ...

  4. UVa 1625 Color Length

    思路还算明白,不过要落实到代码上还真敲不出来. 题意: 有两个由大写字母组成的颜色序列,将它们合并成一个序列:每次可以把其中一个序列开头的颜色放到新序列的尾部. 对于每种颜色,其跨度定义为合并后的序列 ...

  5. Color Length

    题意: 给出两个字符串,求把两字符串组成一个字符串使的字符串中的相同字母的最远距离的和最小. 分析: 本题关键在于怎么计算距离和的方法上.dp[i][j]表示处理到长度i的a串,长度j的b串还需要的计 ...

  6. 动态规划(模型转换):uvaoj 1625 Color Length

    [PDF Link]题目点这里 这道题一眼就是动态规划,然而貌似并不好做. 如果不转换模型,状态是难以处理的. 巧妙地转化:不直接求一种字母头尾距离,而是拆开放到状态中. #include <i ...

  7. [UVa-437] Color Length

    无法用复杂状态进行转移时改变计算方式:巧妙的整体考虑:压缩空间优化时间 传送门:$>here<$ 题意 给出两个字符串a,b,可以将他们穿插起来(相对位置不变).要求最小化ΣL(c),其中 ...

  8. UVa 1625 - Color Length(线性DP + 滚动数组)

    链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  9. UVa 1625 Color Length (DP)

    题意:给定两个序列,让你组成一个新的序列,让两个相同字符的位置最大差之和最小.组成方式只能从一个序列前部拿出一个字符放到新序列中. 析:这个题状态表示和转移很容易想到,主要是在处理上面,dp[i][j ...

随机推荐

  1. JetBrains PyCharm 2017.3注册码

    JetBrains PyCharm 2017.3注册码 (1)在激活界面的License server输入:http://idea.liyang.io:或者:点击help→Register→Licen ...

  2. 如何导入XML数据 (python3.6.6区别于python2 环境)

    1.在python2中 代码如下图: 放在python3 环境下执行,将出现如下错误: 原因: python2中形如myTree.keys()[0]这样的写法是没有问题的,因为myTree.keys( ...

  3. Spark知识点

    1.Spark架构 分布式spark应用中的组件 在分布式环境下,Spark集群采用的是主/从结构.在一个Spark集群中,有一个节点负责中央协调,调度各个分布式工作节点.这个中央协调节点被称为驱动器 ...

  4. Python的matplotlib模块的使用-Github仓库

    import matplotlib.pyplot as plt import numpy as np import requests url='https://api.github.com/searc ...

  5. python数据类型及其特有方法

    一.运算符 in方法 "hello" in "abcdefghijklmnopqrstuvwxyz" "li" in ["gg&q ...

  6. Ubuntu装完后要做的几件事

    Ubuntu装完后要做的几件事 改hosts 无论哪里,改hosts都是第一件事,没hosts咋google.没google咋活.在终端输入命令 sudo gedit /etc/hosts在# The ...

  7. 按升序打印X,Y,Z的整数值

    #include <stdio.h> #define TRUE 1 #define FALSE 0 int main() { int x,y,z; printf("x: &quo ...

  8. The Road to learn React书籍学习笔记(第四章)

    高级React组件 本章将重点介绍高级 React 组件的实现.我们将了解什么是高阶组件以及如何实现它们.此外,我们还将深入探讨 React 中更高级的主题,并用它实现复杂的交互功能. 引用 DOM ...

  9. HTML布局的元素

    header 定义文档或节的页眉 nav 定义导航链接的容器 section 定义文档中的节 article 定义独立的自包含文章 aside 定义内容之外的内容(比如侧栏) footer 定义文档或 ...

  10. Android ImageSwitcher 配合Picasso解决内存溢出(OOM)问题

    最近项目中用到了 ImageSwitcher 来实现图片切换,使用起来很简单,但发现当图片比较大(超过了3M)时,程序出现了内存溢出(OOM)问题而崩溃了. 原因就是图片太大了,显示到 ImageVi ...