题目

题目大意

给你一个字符串,判断这个字符串是否为“阶乘字符串”。

就是子序列包含字符集的全排列的字符串。

n≤26n\leq 26n≤26 ∣S∣≤450|S|\leq 450∣S∣≤450


思考历程

首先,设toi,chto_{i,ch}toi,ch​表示iii后面遇到的第一个chchch(字符)的位置。

这个东西很好预处理。搞出之后可以看成一张图。

那么我们就可以搞出最裸的暴力:枚举全排列,在图上跑,找一找是否有这样的路径。

显然,当前选的字母一定是越早越好,这样后面就能有更多的选择。所以这个方法是正确的。

从000开始在图上跑,每次走没有走过的字母,直至走出一条长度为nnn的路径。显然图上的每一条这样的路径都是唯一的。

于是我就试着求出这些路径条数,即为子序列包含的全排列字符串的个数。

状压DP:设fi,sf_{i,s}fi,s​表示到iii,出现过的字符的集合为sss的方案。

这样跑一跑显然是会超时的。所以我就试着折半搜索,大大减少了时间复杂度。

但是还是有点大……其中sss还比较难存,所以我打了map

然而最终……30分……

和人家暴力的分数一模一样……

应该是常数过大吧……


正解

也是状压DP,设fsf_{s}fs​表示sss集合中的字符的全排列全部出现时的最前的位置。

转移的时候枚举k∈Sk\in Sk∈S,然后从fs−{k}f_{s-\{k\}}fs−{k}​转移过来,方程显然。

预处理的那个数组还是要的。

但是2262^{26}226很大……

有个很强大的结论:如果n>21n>21n>21,则450450450的长度以内没有答案。

具体证明……我不会,网上也没有……

我只知道最短长度是O(n2)O(n^2)O(n2)级别的,但是不知道怎么求……

所以应该预估一下时间复杂度,发现跑不过去,大胆猜想,从而得出这个结论


代码

using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cassert>
#define INF 0x7f7f7f7f
int n,len;
char lis[453];
int to[453][26];
int f[1<<21|1];
int main(){
freopen("in.txt","r",stdin);
int T;
scanf("%d",&T);
while (T--){
scanf("%d%s",&n,lis+1);
if (n>21){
printf("NO\n");
continue;
}
len=strlen(lis+1);
memset(to,0,sizeof to);
for (int i=0;i<=len;++i){
for (int j=i+1,k=0;j<=len && k<n;++j)
if (!to[i][lis[j]-'a'])
to[i][lis[j]-'a']=j,k++;
for (int j=0;j<n;++j)
if (!to[i][j])
to[i][j]=INF;
}
memset(f,0,sizeof f);
f[0]=0;
for (int i=1;i<(1<<n);++i)
for (int j=0;j<n;++j)
if (i>>j&1)
f[i]=(f[i^1<<j]==INF?INF:max(f[i],to[f[i^1<<j]][j]));
if (f[(1<<n)-1]==INF)
printf("NO\n");
else
printf("YES\n");
}
return 0;
}

总结

论大胆猜想的重要性……

[JZOJ3293] 【SHTSC2013】阶乘字符串的更多相关文章

  1. 【JZOJ3293】【BZOJ4416】【luoguP3989】阶乘字符串

    description 给定一个由前n个小写字母组成的串S. 串S是阶乘字符串当且仅当前n个小写字母的全排列(共n!种)都作为S的子序列(可以不连续)出现. 由这个定义出发,可以得到一个简单的枚举法去 ...

  2. 洛谷 P3989 [SHOI2013]阶乘字符串 解题报告

    P3989 [SHOI2013]阶乘字符串 题目描述 给定一个由前\(n(\le 26)\)个小写字母组成的串\(S(|S|\le 450)\).串\(S\)是阶乘字符串当且仅当前 \(n\) 个小写 ...

  3. BZOJ 4416 【SHOI2013】 阶乘字符串

    题目链接:阶乘字符串 又是一道不会做的题……看了题解后我被吓傻了…… 首先我们可以有一个显然的\(O(2^nn)\)的做法.我们先预处理出\(g_{i,j}\)表示字符串中\(i\)号位置开始第一个\ ...

  4. [SHOI2013]阶乘字符串

    题目描述 给定一个由前\(n\)个小写字母组成的串\(S\). 串\(S\)是阶乘字符串当且仅当前\(n\)个小写字母的全排列(共\(n!\)种)都作为\(S\)的子序列(可以不连续)出现. 由这个定 ...

  5. BZOJ4416: [Shoi2013]阶乘字符串

    可以大胆猜想n>21时无解,至于依据,不开O2,1s,n<=21刚好能卡过去= = 并不会证= = #include<cstdio> void up(int& a,in ...

  6. BZOJ4416 [Shoi2013]阶乘字符串 【序列自动机 + 状压dp】

    题目链接 BZOJ4416 题解 建立序列自动机,即预处理数组\(nxt[i][j]\)表示\(i\)位置之后下一个\(j\)出现的位置 设\(f[i]\)表示合法字符集合为\(i\)的最短前缀,枚举 ...

  7. [BZOJ4416][SHOI2013]阶乘字符串(子集DP)

    怎么也没想到是子集DP,想到了应该就没什么难度了. 首先n>21时必定为NO. g[i][j]表示位置i后的第一个字母j在哪个位置,n*21求出. f[S]表示S的所有全排列子序列出现的最后末尾 ...

  8. BZOJ4416 SHOI2013阶乘字符串(状压dp)

    当n大到一定程度(>21)时一定无解,并不会证. 如果要取出一个排列,显然应该让每一位在序列中的位置尽量靠前.于是设f[S]表示存在S子集中这些字母所组成的所有排列的最短前缀的长度,枚举当前排列 ...

  9. [暑假的bzoj刷水记录]

    (这篇我就不信有网站来扣) 这个暑假打算刷刷题啥的 但是写博客好累啊  堆一起算了 隔一段更新一下.  7月27号之前刷的的就不写了 , 写的累 代码不贴了,可以找我要啊.. 2017.8.27upd ...

随机推荐

  1. 3、发送第一个api请求

    接口地址:https://www.v2ex.com/api/topics/latest.json Method: GET Authentication: None 我们打开postman,方法选择ge ...

  2. ssrf对redis未授权访问写webshell

    docker建立redis镜像 docker run -d -p 9999:6379 redis 将redis的6379端口映射到物理机的9999端口 使用工具生成攻击代码 攻击 进入容器查看

  3. Vue的项目搭建及请求生命周期

    目录 Vue的项目搭建及请求生命周期 Vue-CLI的项目搭建 环境搭建 项目创建 pycharm运行Vue项目 Vue项目的大体结构 Vue的请求生命周期 两个小用法 Vue的项目搭建及请求生命周期 ...

  4. 2016CCPC杭州现场赛 B-Bomb /// tarjan缩点

    题目大意: 给定n个爆破点的信息 x y r w 表示爆破点位置为 (x,y) 爆破范围是以位置为圆心 半径为r的圆 引爆这个点的代价为w 引爆某个点时 其他位置在该爆破范围内的爆破点也会被引爆 求引 ...

  5. idea中项目发布到svn服务器

    1. 在svn上创建仓库 https://5VY9MN7OCQV7B9N/svn/KH75-myhouse 2. 将idea的项目上传至svn服务器 项目变红 上传完成项目变成绿色 查看svn服务有文 ...

  6. Vue项目如何关闭Eslint检测

    找到build/webpack.base.config.js文件,修改如下 将图中第二个红色框的内容 "createLintingRule()" 清空,然后保存重新启动项目即可.

  7. vue echart例——柱状图及高度自适应

    1.安装 npm install echarts -s 2.例——点击tab切换时柱状图重绘,高度根据返回数据设置. <template> <div> <ul id=&q ...

  8. 将sparkStreaming的结果保存到S3

    将spark解析的结果保存到S3 这个和保存到本地的区别在于,你需要配置aws的key和密码,以及它的region,代码如下 package com.alo7.spark import java.ut ...

  9. winsock 服务器代码(不建议win服务器listen防火墙会禁止外部访问的)

    int SessionBase::ServerSock() { /* 4 * WSADATA是个结构体,在WSAStartup中被填充. 5 * WSAStartup为调用WinSock准备初始化的工 ...

  10. 如何优雅的在 vue 中添加权限控制

    前言 在一个项目中,一些功能会涉及到重要的数据管理,为了确保数据的安全,我们会在项目中加入权限来限制每个用户的操作.作为前端,我们要做的是配合后端给到的权限数据,做页面上的各种各样的限制. 需求 因为 ...