题面

ARC058D Iroha Loves Strings

给定 \(n\) 个字符串,从中选出若干个按给出顺序连接起来,总长等于 \(m\),求字典序最小的,保证有解。

数据范围:\(1\le n\le 2000\),\(1\le k\le 10^4\),字符串总长 \(S\le 10^6\)。


题解

atcoder 的题就是好啊,非常巧妙,毫不毒瘤。

这篇题解要抨击 Z-functionycx,造福人民。

有一个非常显然的思路:每次选当前选了的串右边没选且能选的中最小的。

这里的能选的可以用个后缀背包判断,可以用 bitset 维护。

注意到对于每个长度 \(i\) 能不能由后缀拼成有个分界点,设为 \(r_i\)。

这个做法有个明显的问题:如果有一个串 geo 和一个串 geoafo,怎么知道哪个好呢?

如果选了 geo,如果剩下的串只有 boomzero,那么就不如 geoafo 了。

如果选了 geoafo,如果剩下的串有个 aba,那么就 geoaba 更优了。

所以想到用 dp,来避免不同长度的字符串的比较。

设 \(f_i\) 表示拼成的长度为 \(i\) 且剩下字符串可以拼成 \(m-i\) 的字符串中字典序最小的(注意 \(f_i\) 是个字符串!)。

然后 \(g_i\) 是拼成 \(f_i\) 的最右字符串的编号,要最左。

转移就是从小到大枚举每个 \(i\)(前提是有这样的 \(f_i\)),然后枚举加上的字符串长度 \(j\)。

要找到 \(g_i\) 到 \(r_{m-i-j}\) 中长度为 \(j\) 的字典序最小字符串,这个可以用 st 表预处理 \(\Theta(1)\) 完成。

然后把这个串加到 \(f_i\) 后面暴力和 \(f_{i+j}\) 比较即可。

关于时间复杂度,总感觉是 \(\Theta(n^2)\) 的,但是不会证,就当 \(\Theta(n^3)\) 的吧(可是它明明跑得很快 /kk)。


代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef double db;
#define x first
#define y second
#define bg begin()
#define ed end()
#define pb push_back
#define mp make_pair
#define sz(a) int((a).size())
#define R(i,n) for(int i(0);i<(n);++i)
#define L(i,n) for(int i((n)-1);i>=0;--i)
const int iinf=0x3f3f3f3f;
const ll linf=0x3f3f3f3f3f3f3f3f; //Data
const int N=2000+1,M=1e4+1,D=12;
int n,m,o[N],we[N],ri[M],g[M];
vector<int> id[M],st[M][D];
string s[N],f[M];
bitset<M> h[N]; //Functions
int get(int j,int l,int r){
l=lower_bound(id[j].bg,id[j].ed,l)-id[j].bg;
r=lower_bound(id[j].bg,id[j].ed,r)-id[j].bg;
if(r-l<=0) return -1;
int t=log2(r-l),&a=st[j][t][l],&b=st[j][t][r-(1<<t)];
return we[a]>we[b]?b:a;
} //Main
int main(){
ios::sync_with_stdio(0);
cin.tie(0),cout.tie(0);
cin>>n>>m,h[n].set(0);
R(i,n) cin>>s[o[i]=i];
stable_sort(o,o+n,[&](int i,int j){return s[i]<s[j];});
R(i,n) we[o[i]]=i,id[sz(s[i])].pb(i);
R(i,m+1){
g[i]=-1,ri[i]=0;
vector<int> &c=id[i],*a=st[i];
if(!sz(c)) continue;
R(i,D) a[i].resize(sz(c));
R(i,sz(c)) a[0][i]=c[i];
R(t,D-1)R(i,sz(c)+1-(2<<t)){
int &x=a[t][i],&y=a[t][i+(1<<t)];
a[t+1][i]=we[x]>we[y]?y:x;
}
}
L(i,n){
h[i]=h[i+1]|h[i+1]<<sz(s[i]);
R(j,m+1)if(h[i][j]&&!h[i+1][j]) ri[j]=i;
}
f[0]="",g[0]=0,ri[0]=n;
R(j,m)if(~g[j])L(i,m+1-j){
int t=get(i,g[j],ri[m-j-i]);
// cout<<"j="<<j<<"->j+i="<<j+i<<'\n';
// cout<<"get("<<i<<","<<g[j]<<","<<ri[m-j-i]<<")="<<t<<'\n';
if(~t&&(!~g[j+i]||f[j+i]>f[j]+s[t]))
f[j+i]=f[j]+s[t],g[j+i]=t+1;
}
// R(i,m+1) cout<<g[i]<<" "<<f[i]<<'\n';
cout<<f[m]<<'\n';
return 0;
}

祝大家学习愉快!

题解-ARC058D Iroha Loves Strings的更多相关文章

  1. 文字列大好きいろはちゃんイージー / Iroha Loves Strings (ABC Edition) (优先队列)

    题目链接:http://abc042.contest.atcoder.jp/tasks/abc042_b Time limit : 2sec / Memory limit : 256MB Score ...

  2. Atcoder Regular Contest 058 D - 文字列大好きいろはちゃん / Iroha Loves Strings(单调栈+Z 函数)

    洛谷题面传送门 & Atcoder 题面传送门 神仙题. mol 一发现场(bushi)独立切掉此题的 ycx %%%%%%% 首先咱们可以想到一个非常 naive 的 DP,\(dp_{i, ...

  3. Codeforces Round #254 (Div. 1) D - DZY Loves Strings

    D - DZY Loves Strings 思路:感觉这种把询问按大小分成两类解决的问题都很不好想.. https://codeforces.com/blog/entry/12959 题解说得很清楚啦 ...

  4. Codeforces Round #254 (Div. 1) D. DZY Loves Strings hash 暴力

    D. DZY Loves Strings 题目连接: http://codeforces.com/contest/444/problem/D Description DZY loves strings ...

  5. [CodeForces - 447B] B - DZY Loves Strings

    B - DZY Loves Strings DZY loves collecting special strings which only contain lowercase letters. For ...

  6. 【题解】DZY Loves Chinese

    [题解]DZY Loves Chinese II 不吐槽这题面了... 考虑如何维护图的连通性,如果把图的变成一颗的\(dfs\)生成树,那么如果把一个节点的父边和他接下来所有的返祖边删除,那么我们就 ...

  7. Codeforces Round #FF (Div. 2):B. DZY Loves Strings

    B. DZY Loves Strings time limit per test 1 second memory limit per test 256 megabytes input standard ...

  8. CF447B DZY Loves Strings 题解

    Content 有一个长度为 \(n\) 的仅含小写字母的字符串 \(s\) 以及 26 个英文小写字母的价值 \(W_\texttt{a},W_\texttt{b},...,W_\texttt{z} ...

  9. CF447B DZY Loves Strings 贪心

    DZY loves collecting special strings which only contain lowercase letters. For each lowercase letter ...

随机推荐

  1. 小谢第58问:nuxt搭建企业官网

    最近公司要重构公司官网,jq+bootstrap 改为了vue,刚开始我以为用vue不是挺好的嘛,后来才发现,有于vue单页面的特性,不利于搜索引擎的抓取,因此在seo方面需要另外想办法,于是乎,就找 ...

  2. nginx&http 第六章 http 协议学习 1

    1.HTTP方法 GET :获取资源 ,GET 方法用来请求访问已被 URI 识别的资源 POST:传输(上传和下载)实体主体 ,POST 方法用来传输实体的主体.虽然用 GET 方法也可以传输实体的 ...

  3. linux中?*tee|\各类引号和-n-e\t\n

    1.通配符:?和*  ?   --匹配任意字符单次. *    --匹配任意字符任意次. [root@localhost test]# rm -fr *     2.管道符: |  将前面命令的结果传 ...

  4. 修改ssh的默认22端口,并使用scp的方法

    修改默认的22的ssh端口只需要修改 /etc/ssh/sshd_config 中的 port 字段为你想要的端口就可以了 以后用其他机器ssh登录这台机器只需要: ssh -p (port) (ip ...

  5. bWAPP----HTML Injection - Reflected (URL)

    HTML Injection - Reflected (URL) 核心代码 1 <div id="main"> 2 3 <h1>HTML Injection ...

  6. 已安装的nginx添加其他模块

    总体操作就是添加新模块并重新编译源码,然后把编译后的nginx可执行文件覆盖原来的那个即可.1 查看已安装的参数nginx -V拷贝那些巴拉巴拉的参数,后面编译的时候使用 2 下载相同版本号的源码,解 ...

  7. Markdown特点

    前言 以前经常在 gitHub 中看到 .md 格式的文件,一直没有注意,也不明白为什么文本文档的后缀不是 .txt ,后来无意中看到了 Markdown,看到了用这个东西写得一些web界面等特别的规 ...

  8. ubuntu安装php的 mongodb扩展

    wget https://pecl.php.net/get/mongodb-1.2.6.tgztar -zxvf mongodb-1.2.6.tgz cd mongodb-1.2.6 /usr/bin ...

  9. Shamir秘密共享方案 (Python)

    Shamir's Secret Sharing scheme is an important cryptographic algorithm that allows private informati ...

  10. Nginx搭建文件共享服务器

    前言 Nginx除了做正反向代理和负载均衡,还能做动静分离服务器,如此便可以当作文件共享服务器使用. 环境 WIN 10 Vmware Workstation 15 Player CentOS Lin ...