传送门

注意到 $m$ 只有 $20$ ,考虑一下状压 $dp$

设 $f[S]$ 表示当前确定的字符集合为 $S$ ,那么转移就考虑从最右边加入的下一个字符 $c$

那么问题来了,代价如何计算

考虑每次加入一个字符以后对于所有字符间的移动$(c_i,c_{i+1})$产生的代价

那么显然只有当 $c_i \in S$ ,$c_{i+1} \notin S$ 时,移动 $(c_i,c_{i+1})$ 会多经过当前加入的字符的位置,那么需要的时间就会增加 $1$

所以我们可以维护一个 $cnt[i][j]$ 表示 $c_{i}=i,c_{i+1}=j$ 的移动数量

那么每次转移代价可以直接 $m^2$ 枚举所有 $c_{i} \in S$ ,$c_{i+1} \notin S$ 的移动求出

这样复杂度是 $m^2 \cdot 2^m$ 算一下发现达到了 $4 \cdot 10^8$ 级别,时间限制 $1s$,显然很有问题

那么现在有两种选择,优化算代价的速度,或者用信仰直接交复杂度不对的代码

然后你发现信仰是对的,卡着时限是可以过的....比如这位神仙:https://codeforces.com/contest/1238/submission/62149786

四个点 $997ms$ 一个点 $998ms$ $\text{Orz}$

但是我没有信仰

来考虑一下如何快速计算代价,设 $h[S]$ 表示所有 $c_i \in S$ ,$c_{i+1} \notin S$ 的移动的数量

再设 $g[p][S],p \notin S$ 表示 $c_i=p , c_{i+1} \in S$ 的移动的数量

那么有 $h[S] = \sum_{p \in S} g[p][U\text{^}S]$ ,其中 $U$ 是全集

现在考虑计算 $g$ ,如果对于每个 $p,S$ 都枚举所有 $p' \notin S$ 再累加显然太慢了

注意到 $S$ 的子集 $S'$ 的代价 $g[p][S']$ 一定会加入到 $g[p][S]$ 中,那么对于 $S$ 直接枚举某一位选择的 $p'$

设 $S'|(1<<p')=S$,有 $g[p][S]=g[p][S']+cnt[p][p']$

然后现在又有一个小问题,要对所有 $S$ 求出某一个为 $1$ 的位置

考虑树状数组的操作, $x&-x$ 就是 $x$ 的第一位 $1$ 的值

最后维护 $pos[x]$ 表示值为 $x=(1<<p)$ 时的 $p$

然后这一题就解决了,复杂度 $O(m \cdot 2^m)$

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<vector>
using namespace std;
typedef long long ll;
inline int read()
{
int x=,f=; char ch=getchar();
while(ch<''||ch>'') { if(ch=='-') f=-; ch=getchar(); }
while(ch>=''&&ch<='') { x=(x<<)+(x<<)+(ch^); ch=getchar(); }
return x*f;
}
const int N=1e5+,M=;
int n,m,cnt[M][M];
int pos[(<<M)+],f[(<<M)+],g[M][(<<M)+],h[(<<M)+];
char s[N];
int main()
{
n=read(),m=read();
scanf("%s",s+);
for(int i=;i<n;i++)
cnt[s[i]-'a'][s[i+]-'a']++,
cnt[s[i+]-'a'][s[i]-'a']++;
for(int i=;i<m;i++) pos[<<i]=i;
memset(f,0x3f,sizeof(f));
int mx=(<<m)-; f[]=;
for(int p=;p<m;p++)
for(int o=;o<mx;o++)
{
if(o&(<<p)) continue;
g[p][o]=g[p][o-(o&-o)]+cnt[p][pos[o&-o]];
}
for(int o=;o<mx;o++)
for(int p=;p<m;p++)
if(o&(<<p)) h[o]+=g[p][mx^o];
for(int o=;o<mx;o++)
for(int p=;p<m;p++)
{
if(o&(<<p)) continue;
f[o^(<<p)]=min(f[o^(<<p)],f[o]+h[o]);
}
printf("%d\n",f[mx]);
return ;
}

Codeforces 1238E. Keyboard Purchase的更多相关文章

  1. Keyboard Purchase CodeForces - 1238E (状压)

    大意: 给定串$s$, 字符集为字母表前$m$个字符, 求一个$m$排列$pos$, 使得$\sum\limits_{i=2}^n|{pos}_{s_{i-1}}-{pos}_{s_{i}}|$最小. ...

  2. codeforces 831B. Keyboard Layouts 解题报告

    题目链接:http://codeforces.com/contest/831/problem/B 题目意思:给出两个长度为26,由小写字母组成的字符串s1和s2,对于给出的第三个字符串s3,写出对应s ...

  3. CF1238E.Keyboard Purchase 题解 状压/子集划分DP

    作者:zifeiy 标签:状压DP,子集划分DP 题目链接:https://codeforces.com/contest/1238/problem/E 题目大意: 给你一个长度为 \(n(n \le ...

  4. Codeforces 474A Keyboard (水

    题目链接:点击打开链接 键盘移位了,问输出相应的字母 #include <cstdio> #include <cstring> char a[105]; char b[3][1 ...

  5. CodeForces 474A Keyboard (水题)

    题意:给定一个键盘,然后一行字母,和一个字符,代表把那一行字母在键盘上左移还是右移一位. 析:没什么好说的,直接暴力就好. 代码如下: #include<bits/stdc++.h> us ...

  6. Codeforces1238E. Keyboard Purchase(状压dp + 计算贡献)

    题目链接:传送门 思路: 题目中的m为20,而不是26,显然在疯狂暗示要用状压来做. 考虑状压字母集合.如果想要保存字母集合中的各字母的顺序,那就和经典的n!的状态的状压没什么区别了,时间复杂度为O( ...

  7. 【CF1238E】Keyboard Purchase(状压DP,贡献)

    题意:有m种小写字符,给定一个长为n的序列,定义编辑距离为序列中相邻两个字母位置差的绝对值之和,其中字母位置是一个1到m的排列 安排一种方案,求编辑距离最小 n<=1e5,m<=20 思路 ...

  8. CF-diary

    (做题方式:瞟题解然后码) 1238E. Keyboard Purchase \(\texttt{Difficulty:2200}\) 题意 给你一个长度为 \(n\) 的由前 \(m\) 个小写字母 ...

  9. Educational Codeforces Round 74

    目录 Contest Info Solutions A. Prime Subtraction B. Kill 'Em All C. Standard Free2play D. AB-string E. ...

随机推荐

  1. Excel2016 保存\复制 卡死问题解决

    遇到的问题: 工作中经常碰到一些Excel表, 复制一行, 再粘贴要等5s以上才能显示成功. 保存一下文档, 也会出现页面白屏卡死的情况, 经过网上多个帖子进行操作依旧无解, 最后找到了自己的方法得以 ...

  2. MACD中短线交易系统

    1.MA5.MA10金叉,且股价收盘站稳5日均线 2.MACD金叉 3.MACD红绿柱 a.MACD红柱发散,表示多头力量增强,此时买入或加仓 b.MACD红柱收缩,表示多头力量减弱,此时卖出或减仓 ...

  3. 怎样修改 VS Code 主题?

    方法1. 点击左上角 File > Preferences > Color Theme. 方法2. 使用快捷键: Ctrl + K , Ctrl + T  PS: 查询各种操作的快捷键可以 ...

  4. 使用X.509数字证书加密解密实务(三)-- 使用RSA证书结合对称加密技术加密长数据

    一.  使用证书结合对称加密算法加.解密长数据 上一章节讨论了如何使用RSA证书加密数据,文中提到:“Dotnet的RSA实现有个特点,它必须要在明文中添加一些随机数,所以明文不能把128字节占满,实 ...

  5. Javascript中new的作用

    关于js中new关键字的理解,先来看个例子:像这样创建实例时使用new与不使用new有什么区别????function ParasiticPerson(name, age, job) {    var ...

  6. Ubuntu12.04 root登陆方法【保证有效】

    su -取得root权限后,gedit /etc/lightdm/lightdm.conf ,里面的内容全部改为 [SeatDefaults] greeter-session=unity-greete ...

  7. C++ STL 之 容器的深拷贝和浅拷贝

    如果我们没有提供拷贝构造函数,没有重载=操作符,vector 对我们的 mc 对象进行的简单的浅拷贝,将拷贝的对象插入到容器中,导致我们的 mc 对象的 data 指针和容器中mc 对象的拷贝对象中的 ...

  8. 基于Vue实现拖拽效果

    参考地址:基于Vue实现拖拽效果 参考链接中讲的比较详细,我只使用了其中自定义指令的方法.整体代码如下: <template> <!-- 卡片 --> <div clas ...

  9. BBPlus团队ALPHA冲刺博客(肖文恒)

    ALPHA冲刺博客 第一天:https://www.cnblogs.com/bbplus/p/11931039.html 第二天:https://www.cnblogs.com/bbplus/p/11 ...

  10. impala 下的SQL操作

    1.修改字段中文名称 ALTER TABLE tablename CHANGE doc_rev_ind  doc_rev_ind varchar(40) comment '收取要求' 2.增加一列 A ...