\(\mathscr{Description}\)

  Link.

  给定 \(n\) 个字符串 \(S_{1..n}\),选出其一个最大子集 \(T\),使得 \(T\) 中的字符串两两不存在包含关系。要求构造答案。

  \(n\le750\),\(\sum|S|\le10^7\)。

\(\mathscr{Solution}\)

  \(\subseteq\) 在 \(\{S_n\}\) 中是偏序关系,那么最长反链 \(T\) 的大小就是最小链覆盖大小。先建出 AC 自动机,通过 fail 树的相邻祖孙关系(每个点与其最深祖先的关系)建立偏序,再通过 bitset 维护传递闭包。利用经典二分图最大匹配模型就能求出 \(|T|\)。

  难点在于 \(T\) 的构造。事实上,这就是普遍的构造最长反链的方法。推荐 C202044zxy 的博客。由于还没有想到 motivated 的讲解,所以直接搬前人的轮子了。(

  提炼一点结论性的过程方便复习:

  • 先求最大匹配。
  • 最大匹配 \(\rightarrow\) 最小点覆盖,二分图上 DFS。
  • 最小点覆盖 \(\rightarrow\) 最大独立集,直接取补集。
  • 实点和虚点都在独立集里的点选入反链。

  实现上,\(\mathcal O(n^3)\) Hungary 然后 DFS 标记一遍就行了,不需要想证明一样分很多步。(所以我还是觉得这个证明写得不自然。

\(\mathscr{Code}\)

/*+Rainybunny+*/

#include <bits/stdc++.h>

#define rep(i, l, r) for (int i = l, rep##i = r; i <= rep##i; ++i)
#define per(i, r, l) for (int i = r, per##i = l; i >= per##i; --i) const int MAXN = 750, MAXL = 1e7, IINF = 0x3f3f3f3f;
int n, mtc[MAXN * 2 + 5];
bool vis[MAXN * 2 + 5];
std::string str[MAXN + 5];
std::bitset<MAXN + 5> adj[MAXN + 5]; struct AhoCorasickAutomaton {
int node, ch[MAXL + 5][2], len[MAXL + 5][2], fail[MAXL + 5], bel[MAXL + 5];
int que[MAXL + 5], hd, tl;
AhoCorasickAutomaton(): node(1) {} inline int insert(const std::string& str, const int id) {
int u = 1;
for (int c: str) {
if (!ch[u][c -= 'a']) ch[u][c] = ++node;
u = ch[u][c];
}
return bel[u] = id, u;
} inline void build() {
hd = 1, tl = 0;
if (ch[1][0]) fail[que[++tl] = ch[1][0]] = 1;
else ch[1][0] = 1;
if (ch[1][1]) fail[que[++tl] = ch[1][1]] = 1;
else ch[1][1] = 1;
while (hd <= tl) {
int u = que[hd++];
if (ch[u][0]) fail[que[++tl] = ch[u][0]] = ch[fail[u]][0];
else ch[u][0] = ch[fail[u]][0];
if (ch[u][1]) fail[que[++tl] = ch[u][1]] = ch[fail[u]][1];
else ch[u][1] = ch[fail[u]][1];
}
}
} acam; struct DSU {
int fa[MAXL + 5];
inline int find(const int x) {
return x == fa[x] ? x : fa[x] = find(fa[x]);
}
} dsu; inline bool augment(const int u) {
rep (v, 1, n) if (adj[u].test(v) && !vis[v]) {
vis[v] = true;
if (!mtc[v + n] || augment(mtc[v + n])) {
mtc[v + n] = u, mtc[u] = v + n;
return true;
}
}
return false;
} inline void mark(const int u) {
vis[u + n] = true;
rep (v, 1, n) if (adj[v].test(u) && !vis[v]) {
vis[v] = true, mark(mtc[v] - n);
}
} int main() {
// freopen("a.in", "r", stdin);
// freopen("a.out", "w", stdout);
std::ios::sync_with_stdio(false), std::cin.tie(0), std::cout.tie(0); std::cin >> n;
rep (i, 1, n) std::cin >> str[i], acam.insert(str[i], i);
acam.build(); rep (i, 1, acam.node) {
if (i != 1 && !acam.bel[i]) dsu.fa[i] = acam.fail[i];
else dsu.fa[i] = i;
} rep (i, 1, n) {
int u = 1;
for (int c: str[i]) {
u = acam.ch[u][c - 'a'];
if (acam.bel[dsu.find(u)]) adj[i].set(acam.bel[dsu.find(u)]);
}
u = acam.fail[u]; // When bel[u]=i, there's some thing wrong. Fix it.
if (acam.bel[dsu.find(u)]) adj[i].set(acam.bel[dsu.find(u)]);
}
rep (k, 1, n) rep (i, 1, n) if (adj[i].test(k)) adj[i] |= adj[k];
rep (i, 1, n) adj[i].reset(i); int ans = n;
rep (i, 1, n) ans -= augment(i), memset(vis, false, sizeof vis);
rep (i, 1, n) if (!mtc[i + n]) mark(i); std::cout << ans << '\n';
rep (i, 1, n) if (!vis[i] && vis[i + n]) std::cout << i << ' ';
std::cout << '\n';
return 0;
}

Solution -「CF 590E」Birthday的更多相关文章

  1. Solution -「CF 1342E」Placing Rooks

    \(\mathcal{Description}\)   Link.   在一个 \(n\times n\) 的国际象棋棋盘上摆 \(n\) 个车,求满足: 所有格子都可以被攻击到. 恰好存在 \(k\ ...

  2. Solution -「CF 1622F」Quadratic Set

    \(\mathscr{Description}\)   Link.   求 \(S\subseteq\{1,2,\dots,n\}\),使得 \(\prod_{i\in S}i\) 是完全平方数,并最 ...

  3. Solution -「CF 923F」Public Service

    \(\mathscr{Description}\)   Link.   给定两棵含 \(n\) 个结点的树 \(T_1=(V_1,E_1),T_2=(V_2,E_2)\),求一个双射 \(\varph ...

  4. Solution -「CF 923E」Perpetual Subtraction

    \(\mathcal{Description}\)   Link.   有一个整数 \(x\in[0,n]\),初始时以 \(p_i\) 的概率取值 \(i\).进行 \(m\) 轮变换,每次均匀随机 ...

  5. Solution -「CF 1586F」Defender of Childhood Dreams

    \(\mathcal{Description}\)   Link.   定义有向图 \(G=(V,E)\),\(|V|=n\),\(\lang u,v\rang \in E \Leftrightarr ...

  6. Solution -「CF 1237E」Balanced Binary Search Trees

    \(\mathcal{Description}\)   Link.   定义棵点权为 \(1\sim n\) 的二叉搜索树 \(T\) 是 好树,当且仅当: 除去最深的所有叶子后,\(T\) 是满的: ...

  7. Solution -「CF 623E」Transforming Sequence

    题目 题意简述   link.   有一个 \(n\) 个元素的集合,你需要进行 \(m\) 次操作.每次操作选择集合的一个非空子集,要求该集合不是已选集合的并的子集.求操作的方案数,对 \(10^9 ...

  8. Solution -「CF 1023F」Mobile Phone Network

    \(\mathcal{Description}\)   Link.   有一个 \(n\) 个结点的图,并给定 \(m_1\) 条无向带权黑边,\(m_2\) 条无向无权白边.你需要为每条白边指定边权 ...

  9. Solution -「CF 599E」Sandy and Nuts

    \(\mathcal{Description}\)   Link.   指定一棵大小为 \(n\),以 \(1\) 为根的有根树的 \(m\) 对邻接关系与 \(q\) 组 \(\text{LCA}\ ...

  10. Solution -「CF 487E」Tourists

    \(\mathcal{Description}\)   Link.   维护一个 \(n\) 个点 \(m\) 条边的简单无向连通图,点有点权.\(q\) 次操作: 修改单点点权. 询问两点所有可能路 ...

随机推荐

  1. LookupViT:类似SE的token压缩方案,加速还能丰富特征 | ECCV'24

    视觉变换器(ViT)已成为众多工业级视觉解决方案的事实标准选择.但由于每一层都计算自注意力,这导致其推理成本对许多场景而言是不可接受的,因为自注意力在标记数量上具有平方的计算复杂度.另一方面,图像中的 ...

  2. Redhat7重置root管理员密码

    如果要重置Red Hat Enterprise Linux Server release 7.0 的root常见有2种办法(均测试有效) rd.break方法 1.重启Linux系统主机并出现引导界面 ...

  3. nginx从入门到实战

    概述 异步非阻塞的高性能HTTP和反向代理服务器 nginx的运行架构 1.Master进程 启动检查nginx.conf是否正确 根据配置文件创建.监控worker进程的数量和状态 监听socket ...

  4. PHP mysql 大量批量insert或update数据出错问题

    UPDATE users SET age = 30 WHERE name = 'Alice'; UPDATE users SET age = 25 WHERE name = 'Bob'; UPDATE ...

  5. Java 面试用什么项目?全是商场秒杀 RPC,我吐了

    看了几百份简历,真的超过 90% 的小伙伴的项目是商城.RPC.秒杀.论坛.外卖.点评等等烂大街的项目,人人都知道这些项目烂大街了,但大部分同学还是得硬着头皮做,没办法,网络上能找到的.教程比较完善的 ...

  6. 通过wget命令扒站仿站

    在Linux下,通过一个命令就可以把整个站相关的文件全部下载下来. wget -r -p -k -np [网址] 参数说明: -r : 递归下载 -p : 下载所有用于显示 HTML 页面的图片之类的 ...

  7. byte,关于127+1等于多少

    public class Main { public static void main(String[] args) { Integer i1 = 100; Integer i2 = 100; Int ...

  8. Liunx-Shell脚本

    shell可以理解为对命令行的一个解释器,命令行输入命令,shell执行,linux系统输出结果 1. shell脚本格式 开头: #!/bin/bash #!告诉系统其后路径所指定的程序即是解释此脚 ...

  9. PHP扩展之Yaconf

    这个是继鸟哥出品的yaf,yar 之后的又一个好用的工具.  Yaconf配置管理工具 具体可以看鸟哥的文档: https://www.laruence.com/2015/06/12/3051.htm ...

  10. 21stUESTC

    数矩形 平面上有 \(n\) 个点,这 \(n\) 个点两两不重合.问这 \(n\) 个点可以组成多少个矩形 请注意:矩形的边不必平行于坐标轴. \(4 ≤ n ≤ 1000\) 保证这些点两两不重合 ...