[HEOI2012] 朋友圈(最大团 + 结论 + 二分图 + 网络流)

problem

luogu-P2423

solution

本题即求无向图最大团问题。这是个 NP hard\text{NP hard}NP hard 问题,所以必须从图的特殊性质出发,否则只能暴搜。

异或运算等价于二进制下不进位的加法运算。

observation1:\text{observation1}:observation1: AAA 国人之间做朋友的条件等价于两人的 aia_iai 奇偶性不同。

observation2:\text{observation2}:observation2: 基于上一条,我们可以得出最大朋友圈中 AAA 国人数只可能包含 0/1/20/1/20/1/2 人。

observation3:\text{observation3}:observation3: BBB 国人的限制有两种,且是各自独立的。

observation4:\text{observation4}:observation4:BBB 国人按照 bib_ibi 的奇偶性分类,则两类之间的人一定都是两两可以做朋友的。

observation5:\text{observation5}:observation5: 而第二种限制的朋友关系则是一个在奇数类一个在偶数类。

BBB 国分类后的情况形似二分图。

只不过二分图是,左右内部两两无边,然后有连接左右的边。

这里是左右内部两两有边,然后有连接左右的边。

observation6:\text{observation6}:observation6: 原图的补图是个二分图,且原图的最大团等于补图的最大独立集。

最大团是两两有边,最大独立集是两两无边,感觉上原图的最大团就应该等于补图的最大独立集。

严谨证明可自行百度。

而二分图的最大独立集又等于二分图的最小点覆盖。

而二分图的最小点覆盖又等于总点数减去最大匹配数。

所以问题就变成了求补图(二分图)的最大匹配数,网络流即可。

至于 AAA 国的人,直接枚举是选 0/1/20/1/20/1/2 个人,然后每次重新建图跑网络流即可。

因为有 AAA 国人的存在,所以只用考虑与枚举的 AAA 国人是朋友的 BBB 国人跑网络流即可。

具体可以看代码实现。

code

#include <bits/stdc++.h>
using namespace std;
#define maxn 3005
int T, A, B, M, s, t, cnt, ans, tot;
queue < int > q;
vector < int > G[maxn];
pair < int, int > e[maxn * maxn];
struct node { int to, nxt, flow; }E[maxn * maxn];
int head[maxn], cur[maxn], dep[maxn], vis[maxn], a[maxn], b[maxn];
int g[2][maxn];void addedge( int u, int v ) {E[++ cnt] = { v, head[u], 1 }, head[u] = cnt;E[++ cnt] = { u, head[v], 0 }, head[v] = cnt;
}bool bfs() {memset( dep, 0, sizeof( dep ) );memcpy( cur, head, sizeof( head ) );dep[s] = 1; q.push( s );while( ! q.empty() ) {int u = q.front(); q.pop();for( int i = head[u];~ i;i = E[i].nxt ) {int v = E[i].to;if( ! dep[v] and E[i].flow ) dep[v] = dep[u] + 1, q.push( v );}}return dep[t];
}int dfs( int u, int cap ) {if( u == t or ! cap ) return cap;int flow = 0;for( int i = cur[u];~ i;i = E[i].nxt ) {int v = E[i].to; cur[u] = i;if( dep[v] == dep[u] + 1 ) {int w = dfs( v, min( cap, E[i].flow ) );if( ! w ) continue;E[i ^ 1].flow += w;E[i].flow -= w;flow += w;cap -= w;if( ! cap ) break;}}return flow;
}void dinic( int n ) {while( bfs() ) {n -= dfs( s, 1e9 );//总点数-最大匹配 才是要求的补图的最小点覆盖 即原图的最大团if( n <= ans ) return;}ans = n;
}void build() {cnt = -1, memset( head, -1, sizeof( head ) );for( int i = 1;i <= tot;i ++ )if( vis[e[i].first] and vis[e[i].second] )addedge( e[i].first, e[i].second );for( int i = 1;i <= g[0][0];i ++ ) if( vis[g[0][i]] ) addedge( s, g[0][i] );for( int i = 1;i <= g[1][0];i ++ ) if( vis[g[1][i]] ) addedge( g[1][i], t );
}int main() {scanf( "%d", &T );while( T -- ) {scanf( "%d %d %d", &A, &B, &M );for( int i = 1;i <= A;i ++ ) G[i].clear();s = 0, t = B + 1, ans = tot = g[0][0] = g[1][0] = 0;for( int i = 1;i <= A;i ++ ) scanf( "%d", &a[i] );for( int i = 1;i <= B;i ++ ) scanf( "%d", &b[i] );for( int i = 1, x, y;i <= M;i ++ ) {scanf( "%d %d", &x, &y );G[x].push_back( y );}for( int i = 1;i <= B;i ++ ) g[b[i]&1][++g[b[i]&1][0]] = i;for( int i = 1;i <= g[0][0];i ++ )for( int j = 1;j <= g[1][0];j ++ )if( __builtin_popcount( b[g[0][i]] | b[g[1][j]] ) & 1 )continue;elsee[++ tot] = make_pair( g[0][i], g[1][j] );for( int i = 1;i <= B;i ++ ) vis[i] = 1;build(), dinic( B ); //一个A类人都不选for( int i = 1;i <= A;i ++ ) { //枚举选的一个A类人memset( vis, 0, sizeof( vis ) );for( int x : G[i] ) vis[x] ++;int n = 1;//n当前情况的总点数for( int j = 1;j <= B;j ++ ) n += vis[j];if( n <= ans ) continue;else build(), dinic( n );}for( int i = 1;i <= A;i ++ )for( int j = i + 1;j <= A;j ++ )if( ( a[i] ^ a[j] ) & 1 ) {//枚举选择两个A类人memset( vis, 0, sizeof( vis ) );for( int x : G[i] ) vis[x] ++;for( int x : G[j] ) vis[x] ++;int n = 2;for( int k = 1;k <= B;k ++ )if( vis[k] < 2 ) vis[k] = 0;else vis[k] = 1, n ++;if( n <= ans ) continue;else build(), dinic( n );}printf( "%d\n", ans );}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/316337.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

cf1555 E. Boring Segments

cf1555 E. Boring Segments 题意&#xff1a; 给你n个线段&#xff0c;最大点是m&#xff0c;每一个线段有一个权值w&#xff0c;你能选择线段来覆盖1-m这个区间的&#xff0c;选择的代价为最大权值和最小权值的差。问你最小的的代价是多少。 题解&#xff1a; 尺取线段树 …

C#并行编程(3):并行循环

初识并行循环并行循环主要用来处理数据并行的&#xff0c;如&#xff0c;同时对数组或列表中的多个数据执行相同的操作。在C#编程中&#xff0c;我们使用并行类System.Threading.Tasks.Parallel提供的静态方法Parallel.For和Parallel.ForEach来实现并行循环。从方法名可以看出&…

[NOI Online 2022 提高组] 丹钓战(单调栈 + 树状数组 / 主席树)

problem luogu-P8251 solution 按照题意模拟单调栈。 求出对于 iii 而言&#xff0c;当时单调栈的栈顶元素记为 pip_ipi​。 如果到 iii 时&#xff0c;栈顶已经为 pip_ipi​ 了&#xff0c;意味着这中间的所有元素要么是被 iii 弹出&#xff0c;要么就是被 iii 前面的某些…

Acwing 252. 树

Acwing 252. 树 题意&#xff1a; 给定一个有 N 个点&#xff08;编号 0,1,…,N−1&#xff09;的树&#xff0c;每条边都有一个权值&#xff08;不超过 1000&#xff09;。 树上两个节点 x 与 y 之间的路径长度就是路径上各条边的权值之和。 求长度不超过 K 的路径有多少条…

.net 4.5部署到docker容器

.NET FX 应用程序也是可以容器化的&#xff0c;容器化的选项有两个&#xff1a;部署到windows容器部署到linux容器部署到windows容器由于.net本身就是运行在windows平台的&#xff0c;所以它与windows容器也是更加适合&#xff0c;你可以以iis镜像为基础&#xff0c;去编写你的…

[NOI Online 2022 提高组] 讨论(巧妙的切入方式)

problem luogu-P8252 solution 本题最难处理的就是两个人会做的题目集合是包含关系的限制。 将所有人按会做的题数从大到小排序。 然后枚举 iii&#xff0c;只要这个人和之前某个人存在有至少一道公共的题目&#xff0c;并且保证这个人有新的题目&#xff0c;那么这两个人…

P4149 [IOI2011]Race

P4149 [IOI2011]Race 题意&#xff1a; 给一棵树&#xff0c;每条边有权。求一条简单路径&#xff0c;权值和等于 k&#xff0c;且边的数量最小。 题解&#xff1a; 用t[i]:长度为i的路径包含的最少边数 按照子树顺序&#xff0c;依次用dep[u]t[K-d[u]]更新ans&#xff0c;…

将传统 WPF 程序迁移到 DotNetCore 3.0

介绍由于历史原因&#xff0c;基于 Windows 平台存在着大量的基于 .NetFramework 开发的 WPF 和 WinForm 相关程序&#xff0c;如果将这些程序全部基于 DotNetCore 3.0 重写一遍显然是不现实的&#xff0c;但是 DotNetCore 是未来发展的趋势。所以本文通过以 WPF 为例&#xff…

[CodeForces 1603C] Extreme Extension(贪心 + 数论分块优化dp)

problem CodeForces solution observation1:\text{observation1}:observation1: 对于一个非空子段 [l,r][l,r][l,r]&#xff0c;最后一个元素 ara_rar​ 一定不会被操作。 observation2:\text{observation2}:observation2: 基于上一条进一步地有&#xff0c;对于一个非空子段…

.NET Core 时代已经到了,你准备好了吗

今天很多人都收到了阿里云函数计算支持.NET Core的短信了。通过访问 https://help.aliyun.com/document_detail/112379.html 你可以看到最新的说明。现在和过去的两年不同&#xff0c;因为最恶劣的时期已经过去&#xff0c;经历过了最黑暗的时刻&#xff0c;我们正在走向光明的…

cf600 E. Lomsat gelral

cf600 E. Lomsat gelral 题意&#xff1a; 给出一个树&#xff0c;求出每个节点的子树中出现次数最多的颜色的编号和 题解&#xff1a; 树上启发式合并 树上启发式合并讲解 其实就是&#xff1a;递归轻儿子&#xff0c;消除影响&#xff0c;递归重儿子&#xff0c;不消除影…

ASP.NET Core appsettings.json文件(9)《从零开始学ASP.NET CORE MVC》:

本文出自《从零开始学ASP.NET CORE MVC》推荐文章&#xff1a;ASP.NET Core launchsettings.json文件ASP.NET Core appsettings.json文件在本视频中&#xff0c;我们将讨论ASP.NET Core 项目中appsettings.json文件的重要性。在以前的ASP.NET版本中&#xff0c;我们将应用程序配…

[CQOI2018] 交错序列(矩阵加速优化dp)

problem luogu-P4456 solution 预处理阶乘和阶乘的逆元&#xff0c;枚举 111 出现次数 iii&#xff0c;∑(n−i1i)(n−i)aib\sum\binom{n-i1}{i}(n-i)^ai^b∑(in−i1​)(n−i)aib。 (n−i1i)\binom{n-i1}{i}(in−i1​) 如何推出来? 从 nnn 个中选 iii 个 (ni)\binom ni(in​…

P4245 【模板】任意模数多项式乘法(NTT)

题意&#xff1a; P4245 【模板】任意模数多项式乘法 题解&#xff1a; NTT模板&#xff0c;记录一下 代码&#xff1a; #include <bits/stdc.h>using namespace std;#define REP(i, a, b) for (int i (a), _end_ (b); i < _end_; i) #define debug(...) fprintf…

在Windows上使用Docker运行.NetCore

今天我们来说下如何在windows下使用docker运行.net core&#xff0c;既然是docker&#xff0c;那么我们首先得在windows上安装docker。在Windows安装 docker 有两种选择 &#xff1a;1、docker for windows2、docker toolbox 区别&#xff1a;docker for windows-64位Windows 1…

[AtCoder Educational DP Contest] J - Sushi(期望dp)

problem luogu 现有N(1≤N≤300)N(1 ≤ N ≤ 300)N(1≤N≤300) 个盘子&#xff0c;编号为1,2,3,…,N1,2,3,…,N1,2,3,…,N。 第 iii个盘中放有 ai(1≤ai≤3)a_i(1≤a_i ≤3)ai​(1≤ai​≤3)个寿司。 接下来每次执行以下操作&#xff0c;直至吃完所有的寿司。 从第 1,2,3,…

cf570 D. Tree Requests

cf570 D. Tree Requests 题意&#xff1a; 给定一个以 1 为根的 n 个结点的树&#xff0c;每个点上有一个字母&#xff08;a-z&#xff09;。每次询问 a, b 查询以 a 为根的子树内深度为 b 的结点上的字母重新排列之后是否能构成回文串。 题解&#xff1a; 回文串形成条件&…

浅谈C#在网络波动时防重复提交

前几天&#xff0c;公司数据库出现了两条相同的数据&#xff0c;而且时间相同&#xff08;毫秒也相同&#xff09;。排查原因&#xff0c;发现是网络波动造成了重复提交。由于网络波动而重复提交的例子也比较多&#xff1a;网络上&#xff0c;防重复提交的方法也很多&#xff0…

[AtCoder Educational DP Contest] V - Subtree(树形dp + 前缀积/后缀积)

problem luogu 给一棵树&#xff0c;对每一个节点染成黑色或白色。 对于每一个节点&#xff0c;求强制把这个节点染成黑色的情况下&#xff0c;所有的黑色节点组成一个联通块的染色方案数&#xff0c;答案对 MMM 取模。 1≤n≤1e5,2≤M≤1e91\le n\le 1e5,2\le M\le 1e91≤n…

P2634 [国家集训队]聪聪可可(点分治做法)

P2634 [国家集训队]聪聪可可 题意&#xff1a; 一颗n个点的树&#xff0c;问其中两点之间的边上数的和加起来是3的倍数的点对有多少个&#xff1f; 输出这样的点对所占比例 题解&#xff1a; 因为是求三的倍数&#xff0c;我们num来记录%30&#xff0c;1&#xff0c;2的数量…