\(\mathcal{Description}\)

  Link.

  指定一棵大小为 \(n\),以 \(1\) 为根的有根树的 \(m\) 对邻接关系与 \(q\) 组 \(\text{LCA}\) 关系,求合法树的个数。

  \(0\le m<n\le13\),\(q\le100\)。

\(\mathcal{Solution}\)

  巧妙的状压 owo。不考虑限制,自然地有状态 \(f(u,S)\) 表示用 \(S\) 中的结点构成以 \(u\) 为根的树的方案数。转移相当于划分出一棵子树,有:

\[f(u,S)=\sum_{v\in T\subseteq(S\setminus u)}f(v,T)f(u,S-T)
\]

  不过这样显然会算重复。考虑任意固定子树 \(T\) 内的某个点,设 \(p\not=u\) 且 \(p\in T\),钦定 \(p\in T\) 就避免了重复,则:

\[f(u,S)=\sum_{v,p\in T\subseteq(S\setminus u)}f(v,T)f(u,S-T)
\]

  注意 \(p\) 在求和过程中是常量


  接下来着手处理限制:

  • 限制 \(\text{LCA}\),设当前状态 \(f(w,S)\),枚举到子集 \(T\):

    • 若 \((u\in T)\land(v\in T)\Leftrightarrow\mathrm T\)(注意最后这个罗马字体的 \(\mathrm T\) 表示逻辑运算为真),\(u\) 和 \(v\) 的 \(\text{LCA}\) 必然在 \(T\) 中,所以必然不是 \(w\),矛盾。
    • 若 \(q,r\) 的 \(\text{LCA}\) 指定为 \(p\),且 \(p\in T\land(q\not\in T\lor q\not\in T)\Leftrightarrow \mathrm T\),即两点的 \(\text{LCA}\) 深于其中至少一个点,显然不满足。
  • 限制邻接点,同样地设当前状态 \(f(w,S)\),枚举到子集 \(T\):
    • 若 \(u,v\not=r\) 邻接,且 \((u\in T)\leftrightarrow(v\in T)\Leftrightarrow\mathrm F\),即有且仅有其中一点属于 \(T\),矛盾。
    • 若 \(u,v\) 均与 \(w\) 邻接,且 \((u\in T)\land(v\in T)\Leftrightarrow\mathrm T\),即 \(w\) 向子树 \(T\) 内的至少两个点连边,矛盾。

  转移的时候判一下这四种情况就行啦。

  复杂度 \(\mathcal O(3^nn(n+m+q))\),不过跑不满。为什么我交一发当场最优解 rank1 呢 www?

\(\mathcal{Code}\)

#include <cstdio>
#include <vector>
#include <cstring> #define bel( x, S ) ( ( S >> x ) & 1 ) typedef long long LL;
typedef std::pair<int, int> pii; const int MAXN = 13;
int n, m, q;
LL f[MAXN + 5][1 << MAXN];
std::vector<int> adj[MAXN + 5];
std::vector<pii> dif[MAXN + 5]; inline bool check ( const int r, const int T ) {
for ( pii p: dif[r] ) { // LCA情况1.
if ( bel ( p.first, T ) && bel ( p.second, T ) ) {
return false;
}
}
for ( int u = 0; u < n; ++ u ) { // LCA情况2.
if ( ! ( ( T >> u ) & 1 ) ) continue;
for ( pii p: dif[u] ) {
if ( ! bel ( p.first, T ) || ! bel ( p.second, T ) ) {
return false;
}
}
}
for ( int u = 0; u < n; ++ u ) { // 邻接情况1.
if ( u == r ) continue;
for ( int v: adj[u] ) {
if ( v ^ r && bel ( u, T ) ^ bel ( v, T ) ) {
return false;
}
}
}
int cnt = 0;
for ( int u: adj[r] ) cnt += bel ( u, T ); // 邻接情况2.
return cnt <= 1;
} inline LL solve ( const int r, int S ) {
LL& ret = f[r][S];
if ( ~ ret ) return ret;
ret = 0, S ^= 1 << r; // 这里注意去除根节点.
int p;
for ( p = 0; p < n && ! ( ( S >> p ) & 1 ); ++ p ); // 钦定一点p.
for ( int T = S; T; T = ( T - 1 ) & S ) { // 枚举子集.
if ( ! ( ( T >> p ) & 1 ) || ! check ( r, T ) ) continue;
for ( int u = 0; u < n; ++ u ) {
if ( ! bel ( u, T ) ) continue;
ret += solve ( u, T ) * solve ( r, S ^ T ^ ( 1 << r ) );
}
}
return ret;
} int main () {
scanf ( "%d %d %d", &n, &m, &q );
for ( int i = 1, u, v; i <= m; ++ i ) {
scanf ( "%d %d", &u, &v ), -- u, -- v;
adj[u].push_back ( v ), adj[v].push_back ( u );
}
for ( int i = 1, u, v, w; i <= q; ++ i ) {
scanf ( "%d %d %d", &u, &v, &w ), -- u, -- v, -- w;
dif[w].push_back ( { u, v } );
}
memset ( f, 0xff, sizeof f );
for ( int u = 0; u < n; ++ u ) f[u][1 << u] = 1; // 单点,一种方案.
printf ( "%lld\n", solve ( 0, ( 1 << n ) - 1 ) );
return 0;
}

Solution -「CF 599E」Sandy and Nuts的更多相关文章

  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 487E」Tourists

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

随机推荐

  1. spring security 登出操作 详细说明

    1.前言 这里专门 做 spring security 登出操作 的  详细记录 2.操作 (1)目录结构 (2)在security 拦截规则配置文件添加退出登录支持 源码 package com.e ...

  2. tomcat启动卡在了 At least one JAR was scanned for TLDs yet contained no TLDs 的根本原因与解决办法

    1.前言 有时候服务器开启时启动不了,卡在了 org.apache.jasper.servlet.TldScanner.scanJars At least one JAR was scanned fo ...

  3. spring cloud bus 消息总线 动态刷新配置文件 【actuator 与 RabbitMQ配合完成】

    1.前言 单机刷新配置文件,使用actuator就足够了 ,但是 分布式微服务 不可能是单机 ,将会有很多很多的工程 ,无法手动一个一个的发送刷新请求, 因此引入了消息中间件 ,常用的 消息中间件 是 ...

  4. Go语言读取各种配置文件

    配置文件结构体 config.go package config type System struct { Mode string `mapstructure:"mode" jso ...

  5. Centos安装DenyHosts

    一.背景 个人申请的腾讯云主机被扫描端口,数百次登录失败的尝试 解决方法:安装工具,记录并屏蔽恶意访问. 二.检查环境 系统安装的sshd是否支持tcp_wrappers(默认都支持) ldd /us ...

  6. 《剑指offer》面试题27. 二叉树的镜像

    问题描述 请完成一个函数,输入一个二叉树,该函数输出它的镜像. 例如输入:      4    /   \   2     7  / \   / \ 1   3 6   9 镜像输出:      4 ...

  7. mysql数据库优化1

    目录 数据库结构的设计优化 1.数据库结构的设计 2.针对大型的数据量提前进行分库和分表 3.分库分表带来的问题 4.表结构设计注意的问题 查询优化 1.查询语句的注意事项 2.应尽量避免在 wher ...

  8. zip方式安装MySQL提示找不到 MSVCP120.dll的解决方法

    电脑重装系统后,用zip的方式安装MySQL,在执行mysqld --initialize的时候提示 由于找不到 MSVCP120.dll,无法继续执行代码.重新安装程序可能会解决此问题. 解决的方法 ...

  9. 【解决了一个小问题】golang go.mod中多了一个斜杠导致replace无效

    replace github.com/sxxx/common_lib/src/ => ../../common_lib/src 修改成 replace github.com/sxxx/commo ...

  10. 加深对AQS原理的理解示例二:自己设计一个同步工具,同一时刻最多只有两个线程能访问,超过线程将被阻塞

    /** *@Desc 设计一个同步工具,同一时刻最多只有两个线程能访问,超过线程将被阻塞<br> * 思路分析: * 1.共享锁 两个线程及以内能成功获取到锁 * 2. *@Author ...