又是经典模型的好题目

题目描述

人类智慧之神 zhangzj 最近有点胖,所以要减肥,他买了 NN 种减肥药,发现每种减肥药使用了若干种药材,总共正好有 NN 种不同的药材。

经过他的人脑实验,他发现如果他吃下去了 KK(0≤K≤N0≤K≤N)种减肥药,而这 KK 种减肥药使用的药材并集大小也为 KK,这 KK 种才会有效果,否则无效。
第 ii 种减肥药在产生效果的时候会使 zhangzj 的体重增加 PiPi​ 斤,显然 PiPi​ 可以小于 00。

他想知道,一次吃药最好情况下体重变化量是多少,当然可以一种药也不吃,此时体重不变。 由于某些奥妙重重的情况,我们可以让这 NN 种减肥药每一种对应一个其使用的药材,且 NN 种减肥药对应的药材互不相同(即有完美匹配)。

输入格式

第一行一个整数 NN。

接下来 NN 行,每行描述一种减肥药,对于一种减肥药,第一个数读入使用的药材个数 tt,接下来 tt 个整数表示使用的药材编号,一个药材编号在一行只会出现一次。
最后一行 NN 个整数,第 ii 个整数 PiPi​ 表示第 ii 种减肥药产生效果时的体重变化量。

数据范围与提示

对于 30%30% 的数据,N≤20N≤20;
对于另外 10%10% 的数据,Pi<0Pi​<0;
对于 100%100% 的数据,1≤N≤300,∣Pi∣≤10000001≤N≤300,∣Pi​∣≤1000000。


题目分析

这道题最主要的麻烦在于药材和药的数目要相同,这意味着最优情况下,可能需要选取一些收益为负的药来达到合法状态。因此wqs二分药材权值负增量的做法就会挂得很彻底。

再观察题目的特殊性质,发现给出的一张图是恰好两侧点数相同,且具有完美匹配的二分图。这个条件说明在左侧任意选取$k$个点,右侧对应的$k'$必定有$k \le k'$。这是一个相当重要的条件。在这个前提下,我们可以把每种减肥药的权值改为$BASE-P_i$;药材的权值设成$BASE$,再以此做最大权闭合子图。关于这个做法的解释如下所示:

这里是第一个例子,最优方案是三个全选,收益为20。考虑一下“表面权值和最优”(但是不合法)的选前两个的方案,由于我们在右侧每个药材都连出了$BASE$的边,那么当左边选取$k$个、右边选取$k\le k'$个时,进行的操作相当于是选了右侧$k'\times BASE$的割,然而很明显选左侧$k\times BASE-\sum P_i$的割更优。但在最大权闭合子图的方面看,割去$(S,i)$也即意味着点$i$就是不选了,这与前一假设矛盾。因此,右边边权的$BASE$增量能够致使$k\ge k'$,那么综上所述保证了$k=k'$。

以下是第二个例子。在这个例子中,为了保证合法,我们不得不选取增肥100的减肥药。同样是考虑选取前两个元素的方案,会发现仍然由于右侧有三个$BASE,所以导致只有在加入第三个元素时才得到最大值。

 #include<bits/stdc++.h>
const int maxn = ;
const int maxm = ;
const int INF = 2e9;
const int base = 5e6; struct Edge
{
int u,v,f,c;
Edge(int a=, int b=, int c=, int d=):u(a),v(b),f(c),c(d) {}
}edges[maxm];
int edgeTot,head[maxn],nxt[maxm],lv[maxn];
int n,ans,tot,S,T; void addedge(int u, int v, int c)
{
edges[edgeTot] = Edge(u, v, , c), nxt[edgeTot] = head[u], head[u] = edgeTot, ++edgeTot;
edges[edgeTot] = Edge(v, u, , ), nxt[edgeTot] = head[v], head[v] = edgeTot, ++edgeTot;
}
bool buildLevel()
{
std::queue<int> q;
memset(lv, , sizeof lv);
q.push(S), lv[S] = ;
for (int tmp; q.size(); )
{
tmp = q.front(), q.pop();
for (int i=head[tmp]; i!=-; i=nxt[i])
{
int v = edges[i].v;
if (!lv[v]&&edges[i].f < edges[i].c){
lv[v] = lv[tmp]+, q.push(v);
if (v==T) return true;
}
}
}
return false;
}
int fndPath(int x, int lim)
{
int sum = ;
if (x==T||!lim) return lim;
for (int i=head[x]; i!=-&&sum < lim; i=nxt[i])
{
int v = edges[i].v, val;
if (lv[v]==lv[x]+&&edges[i].f < edges[i].c){
if ((val = fndPath(v, std::min(lim-sum, edges[i].c-edges[i].f)))){
edges[i].f += val, edges[i^].f -= val, sum += val;
}else lv[v] = -;
}
}
return sum;
}
int dinic()
{
int ret = , val;
while ((buildLevel()))
while ((val = fndPath(S, INF))) ret += val;
return ret;
}
int main()
{
freopen("loj6045.in","r",stdin);
freopen("loj6045.out","w",stdout);
memset(head, -, sizeof head);
scanf("%d",&n), S = , T = *n+;
for (int i=,k; i<=n; i++)
{
scanf("%d",&k);
for (int u; k; --k)
scanf("%d",&u), addedge(i, u+n, INF);
}
for (int i=,s; i<=n; i++)
{
scanf("%d",&s);
addedge(S, i, base-s);
addedge(i+n, T, base);
tot += base-s;
}
printf("%d\n",-tot+dinic());
return ;
}

END

【思维题 最大权闭合子图】loj#6045. 「雅礼集训 2017 Day8」价的更多相关文章

  1. LOJ#6045. 「雅礼集训 2017 Day8」价(最小割)

    题面 传送门 题解 首先先把所有权值取个相反数来求最大收益,因为最小收益很奇怪 然后建图如下:\(S\to\)药,容量\(\inf+p_i\),药\(\to\)药材,容量\(\inf\),药材\(\t ...

  2. 【LYOI 212】「雅礼集训 2017 Day8」价(二分匹配+最大权闭合子图)

    「雅礼集训 2017 Day8」价 内存限制: 512 MiB时间限制: 1000 ms 输入文件: z.in输出文件: z.out   [分析] 蛤?一开始看错题了,但是也没有改,因为不会做. 一开 ...

  3. loj #6046. 「雅礼集训 2017 Day8」爷

    #6046. 「雅礼集训 2017 Day8」爷 题目描述 如果你对山口丁和 G&P 没有兴趣,可以无视题目背景,因为你估计看不懂 …… 在第 63 回战车道全国高中生大赛中,军神西住美穗带领 ...

  4. LOJ#6046. 「雅礼集训 2017 Day8」爷(分块)

    题面 传送门 题解 转化为\(dfs\)序之后就变成一个区间加,区间查询\(k\)小值的问题了,这显然只能分块了 然而我们分块之后需要在块内排序,然后二分\(k\)小值并在块内二分小于它的元素--一个 ...

  5. LOJ#6044. 「雅礼集训 2017 Day8」共(Prufer序列)

    题面 传送门 题解 答案就是\(S(n-k,k)\times {n-1\choose k-1}\) 其中\(S(n,m)\)表示左边\(n\)个点,右边\(m\)个点的完全二分图的生成树个数,它的值为 ...

  6. LOJ #6044 -「雅礼集训 2017 Day8」共(矩阵树定理+手推行列式)

    题面传送门 一道代码让你觉得它是道给初学者做的题,然鹅我竟没想到? 首先考虑做一步转化,我们考虑将整棵树按深度奇偶性转化为一张二分图,即将深度为奇数的点视作二分图的左部,深度为偶数的点视作二分图的右部 ...

  7. [LOJ#6044]. 「雅礼集训 2017 Day8」共[二分图、prufer序列]

    题意 题目链接 分析 钦定 \(k\) 个点作为深度为奇数的点,有 \(\binom{n-1}{k-1}\) 种方案. 将树黑白染色,这张完全二分图的生成树的个数就是我们钦定 \(k\) 个点之后合法 ...

  8. [LOJ 6031]「雅礼集训 2017 Day1」字符串

    [LOJ 6031] 「雅礼集训 2017 Day1」字符串 题意 给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询 ...

  9. [LOJ 6030]「雅礼集训 2017 Day1」矩阵

    [LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...

随机推荐

  1. POJ1845 Sumdiv 数学?逆元?

    当初写过一篇分治的 题意:求A^B的所有因子之和,并对其取模 9901再输出 对于数A=p1^c1+p2^c2+...+pn*cn,它的所有约数之和为(1+p1+p1^2+p1^3+...+p1^(c ...

  2. 使用 dbutils 的结果集包装类 StringTrimmedResultSet

    1.功能 StringTrimmedResultSet 的功能是去掉结果集中数据的前后空格,这个方法是在取结果的时候处理. 2.使用 一般在新建 QueryRunner 对象的时候使用: QueryR ...

  3. Unity www动态加载网上图片

    一. 1.新建一个UGUI的Button,删掉它的Image组件,添加一个Raw Image组件.如图: 由于删除了Image组件,所以画圈的位置是空的,运行后会自动把Raw Image添加到那里. ...

  4. 【密码学】Https握手协议以及证书认证

    1. 什么是https Https = http + 加密 + 认证 https是对http的安全强化,在http的基础上引入了加密和认证过程.通过加密和认证构建一条安全的传输通道.所以https可以 ...

  5. 5 - 参考函数-API

    5.1 鼠标管理 a). MouseClick 点击鼠标 MouseClick ( "button" [, x, y [, clicks = 1 [, speed = 10]]] ...

  6. 一般的linux系统默认安装的vim是精简版

    一般的linux系统默认安装的vim是精简版(vim-tiny),所以不能配置语法检查等属性或获取在线帮助.需要安装vim-x:x.x.x,vim-common,vim-runtime. :synta ...

  7. Centos 6.5 添加PHP5.6-7.1的源

    centOS6.5 安装后 自带的源中php是5.3版本的,对与php一些常用的框架而言 ,已经不能满足需求了: 使用下面的源 就可以更新到php7.1版本了. # rpm -Uvh http://r ...

  8. React搭建脚手架

    npm install -g create-react-app    //安装 create-react-app react-demo    // react-demo 项目的名称 cd react- ...

  9. 使用visio创建数据库模型

    使用软件:visio 2007 下载地址:https://pan.baidu.com/s/1i4LG1Id 主要参考:http://blog.csdn.net/zhang_xinxiu/article ...

  10. 【Shell脚本学习25】Shell文件包含

    像其他语言一样,Shell 也可以包含外部脚本,将外部脚本的内容合并到当前脚本. Shell 中包含脚本可以使用: . filename 或 source filename 两种方式的效果相同,简单起 ...