这个题大概就是每一个位置都有一个能填字符的限制(一个点集),给出已有的$n$个字符,问能填出的最小字典序的字符串。

总体思路是贪心,每一位尽量选最小的字符。

关键在于判断在某位选了一个字符后,接下来的位置能否满足限制。

考虑怎么判断有解,这里有一种网络流的思路:

  1. 有$6$个点,代表了$a - f$$6$个字符,有源点向这些点连边,流量为该字符的个数。
  2. 另有$2^{6}$个点,代表了各个点集,这些点向汇点连边,流量为该点集在所有限制中出现的次数。
  3. 如果某个点在一个点集中,则由该点向该点集连流量为$INF$的边。

易得,当流量满流时,有合法解。

当然,用网络流这个模型虽然很容易理解,但如果每一次检验都跑一边的话效率不行,出题人有一种基于调整法的网络流做法,在已知流网络上进行修改,但是我不是很懂,代码也较长,在此不做累述。

此处所用的做法十分简洁。有以下结论:

  1. 接下来仅考虑某位置选了某个字符后,判断剩下的一个后缀是否有解。
  2. 定义$cnt_{i}$表示字符$i$的剩余数量,$num_{s}$表示与集合$s$有交的位置(该位置上限制的字符集与$s$有交)。
  3. 枚举$2^{6}$个集合,当所有的集合都准合法时才有解。
  4. 准合法的定义是:$\sum_{i = a}^{f} (其中 i \in s) <= num_{s}$。

显然,如果$|s|=1$,即$s$中只有一个字符,那$s$准合法就是$s$合法(存在合法解)。

这里用于说明$s$是合法的当且仅当所有$s$的子集是合法的,且$s$是准合法的。

由于我们从小到大枚举所有集合,可以保证所有之前已经枚举过的集合已经合法了,即它的子集都合法,故只要判断该集合是否准合法。

关于这个条件的必要性:

  1. 如果$s$不是准合法的,显然$s$不会是合法的,因为没有足够位置放。
  2. 如果$s$的某一个子集$s_{i}$不合法,也就是$s_{i}$的有关位置$num_{s_{i}}$不够调计,那在$s$的有关位置$num_{s}$中并不会有更多的供$s_{i}$中的字符放置的位置,那$s$也将是不合法的。

关于这个条件的充分性:

  1. 如果所有它的子集都合法了,并且$s$准合法,就不会存在某一个集合中的元素占用了过多的公共位置,因为那样会导致另一个子集不合法,即产生矛盾。

这样的话就能在$O(nA2^{A})$解决问题了,其中$A$是字符集大小。

$\bigodot$技巧与套路:

  • 利用网络流的模型
  • 集合的枚举以及使用
    1. #include <cstdio>
    2. #include <cstring>
    3. #include <algorithm>
    4.  
    5. const int N = , ST = ;
    6.  
    7. int n, m, bit[N], ans[N], cnt[], num[ST + ][N];
    8. char s[N], ssr[];
    9.  
    10. inline int Check(int x) {
    11. for (int st = ; st <= ST; ++st) {
    12. int cnum = ;
    13. for (int i = ; i < ; ++i) {
    14. if ((st >> i) & ) cnum += cnt[i];
    15. }
    16. if (num[st][n] - num[st][x] < cnum) return ;
    17. }
    18. return ;
    19. }
    20.  
    21. int main() {
    22. scanf("%s%d", s + , &m);
    23. n = strlen(s + );
    24. for (int i = ; i <= n; ++i) {
    25. bit[i] = ST; ans[i] = -;
    26. ++cnt[s[i] - 'a'];
    27. }
    28. for (int i = , x; i <= m; ++i) {
    29. scanf("%d%s", &x, ssr);
    30. int le = strlen(ssr), st = ;
    31. for (int j = ; j < le; ++j) {
    32. st |= << (ssr[j] - 'a');
    33. }
    34. bit[x] &= st;
    35. }
    36.  
    37. for (int st = ; st <= ST; ++st) {
    38. for (int i = ; i <= n; ++i) {
    39. num[st][i] = num[st][i - ] + (bool)(bit[i] & st);
    40. }
    41. }
    42.  
    43. for (int i = ; i <= n; ++i) {
    44. for (int j = ; j < ; ++j) {
    45. --cnt[j];
    46. if (((bit[i] >> j) & ) && Check(i)) {
    47. ans[i] = j; break;
    48. }
    49. ++cnt[j];
    50. }
    51. if (ans[i] == -) {
    52. puts("Impossible");
    53. return ;
    54. }
    55. }
    56. for (int i = ; i <= n; ++i) {
    57. putchar(ans[i] + 'a');
    58. }
    59.  
    60. return ;
    61. }

【Cf Edu #47 G】Allowed Letters的更多相关文章

  1. 【Cf Edu #47 F】Dominant Indices(长链剖分)

    要求每个点子树中节点最多的层数,一个通常的思路是树上启发式合并,对于每一个点,保留它的重儿子的贡献,暴力扫轻儿子将他们的贡献合并到重儿子里来. 参考重链剖分,由于一个点向上最多只有$log$条轻边,故 ...

  2. 【cf contest 1119 G】Get Ready for the Battle

    题目 你有\(n\)个士兵,需要将他们分成\(m\)组,每组可以为0: 现在这些士兵要去攻打\(m\)个敌人,每个敌人的生命值为\(hp_i\) : 一轮游戏中一组士兵选定一个攻打的敌人,敌人生命值- ...

  3. B. Lost Number【CF交互题 暴力】

    B. Lost Number[CF交互题 暴力] This is an interactive problem. Remember to flush your output while communi ...

  4. 【CF edu 27 G. Shortest Path Problem?】

    time limit per test 3 seconds memory limit per test 512 megabytes input standard input output standa ...

  5. 3.26-3.31【cf补题+其他】

      计蒜客)翻硬币 //暴力匹配 #include<cstdio> #include<cstring> #define CLR(a, b) memset((a), (b), s ...

  6. 【郑轻邀请赛 G】密室逃脱

    [题目链接]:https://acm.zzuli.edu.cn/zzuliacm/problem.php?id=2133 [题意] [题解] 考虑每一个二进制数的最高位->第i位; 肯定是1(这 ...

  7. 【cf补题记录】Codeforces Round #608 (Div. 2)

    比赛传送门 再次改下写博客的格式,以锻炼自己码字能力 A. Suits 题意:有四种材料,第一套西装需要 \(a\).\(d\) 各一件,卖 \(e\) 块:第二套西装需要 \(b\).\(c\).\ ...

  8. 【CF 549G Happy Line】排序

    题目链接:http://codeforces.com/problemset/problem/549/G 题意:给定一个n个元素的整数序列a[], 任意时刻对于任一对相邻元素a[i-1]. a[i],若 ...

  9. 【CF 675D Tree Construction】BST

    题目链接:http://codeforces.com/problemset/problem/675/D 题意:给一个由n个互异整数组成的序列a[],模拟BST的插入过程,依次输出每插入一个元素a[i] ...

随机推荐

  1. 前端开发利器 livereload -- 从此告别浏览器F5键

    各位从事前端开发的童鞋们,大家每天coding && coding,然后F5 && F5,今天推荐一个静态文件在浏览器中自动更新的扩展 livereload,不同手动刷 ...

  2. Hyperledger Fabric chaincode 开发(疑难解答)

    Q&A Q1: 使用fabric release 1.2 进行golang chaincode开发时报错: ..\..\hyperledger\fabric\vendor\github.com ...

  3. hadoop之定制自己的sort过程

    Key排序 1. 继承WritableComparator 在hadoop之Shuffle和Sort中,可以看到mapper的输出文件spill文件需要在内存中排序,并且在输入reducer之前,不同 ...

  4. nginx 在ubuntu上使用笔记(绑定域名)

    1. 重启nginx的两个语句: sudo service nginx restart sudo nginx -s reload 2. nginx配置文件路径: etc/nginx/ 尤其是 site ...

  5. Java时间格式的使用,bug难时真是坑

    很简单的问题,尤其是新手开发,要多自己动手写代码,都说程序猿大都是程序的搬用工,其实不然,好的写手,和差的写手,区别就在是不是会花时间读读代码,并且自己动手实践一下,其实一个程序范这样的错误,绝对是低 ...

  6. Kubernetes探索学习002--Kubernetes的基本使用

    Kubernetes 的基本使用方法 原则:使用YAML文件描述你要部署的API对象! 以部署nginx静态站点为例,具体操作及内容如下 1.编写YAML文件 [root@kubernetes01 ~ ...

  7. 多种方法实现左右固定,中间自适应的CSS布局

    布局是面试中常问的问题,尤其是这类的题目,怎么答才好呢? 大多数人的第一个方法是浮动,没错,浮动.第二个方法呢?你回答定位,没错.第三个方法呢?.... 第四个方法呢?第五个方法呢?.... 其实能想 ...

  8. mysql和oracle查询出的一条结果中的多个字段拼接

    1,mysql concat('a','b','c')和concat_ws('a','b','c')的区别:前者如果有某个值为空,结果为空;后者如果有某个值为空,可以忽略这个控制 SELECT con ...

  9. 进阶系列(4)—— C#文件与流

    一. 驱动器 在Windows操作系统中,存储介质统称为驱动器,硬盘由于可以划分为多个区域,每一个区域称为一个驱动器..NET Framew   ork提供DriveInfo类和 DriveType枚 ...

  10. 结对作业(web)

    作业源代码地址:https://git.coding.net/mal123/arithmetic.git 网页版测试地址:http://47.93.197.5:8080/mal_war_explode ...