题面在这里

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. java程序执行系统命令

    String cmd="orakill orcl 1233";//解锁数据库表 Process proc = Runtime.getRuntime().exec(cmd);

  2. 使用百度定位Api获取当前用户登录地址

    最近在做一个商城项目,客户想把网站做成类似于美团的效果,切换地区时,内容也跟随变化.这就要首先解决根据用户id获得地址的问题,最终决定使用百度定位(不适用于搭建反向代理的项目) String url ...

  3. APSchedule的练习使用

    1 简介 APScheduler的全称是Advanced Python Scheduler.它是一个轻量级的 Python 定时任务调度框架.APScheduler 支持三种调度任务:固定时间间隔,固 ...

  4. php+高德地图webapi 高德jsapi 实现 当前位置与目标位置距离 并按照距离排序(坐标逆转换)

    <script type="text/javascript" src="https://api.map.baidu.com/api?v=2.0&ak='自己 ...

  5. Hadoop(4)--Hbase

    Hadoop 其它组成角色介绍--Hbase 在apache的官方网站上,对于Hbase的定义是他是Hadoop的第一个分布式.可扩展的大数据存储的数据库,他的目标是将非常大的表托管到一个集群中进行相 ...

  6. python函数(2017-8-2)

    1. def 函数名(形式参数) 函数体 return "123" 函数执行了return之后就不再执行下面的代码 2. 默认形参实参的位置一一对应 如果要调整位置,指定形参名字 ...

  7. YUM工具使用

    一.yum命令概述: 1.简介: yum命令时在Fedora和RedHat以及SUSE中基于rpm的软件包管理器,它可以使系统管理人员交互和自动化地更细与管理RPM软件包,能够从指定的服务器自动下载R ...

  8. BootCDNApi使用记录

    通过API获取BootCDN所加速的所有前端开源库的基本信息和文件列表 API 将一下API链接中的.min字样去掉后,获取到的JSON格式的返回信息是经过良好的格式化的,便于查看. 所有开源库简要信 ...

  9. go学习笔记-函数

    函数 定义 格式 func function_name( [parameter list] ) [return_types] { 函数体 } 解析 func:函数由 func 开始声明 functio ...

  10. 异步消息处理(Message, Handler, MessageQueue, Looper)

    AsyncTask 适用于单线程任务处理,多任务处理还是 Message/Handler 处理方便一些 主要使用方式: 1,创建子类继承自 Handler 类,覆盖 handleMessage(Mes ...