Codeforces Round #523 (Div. 2) E. Politics(最小费+思维建图)
https://codeforces.com/contest/1061/problem/E
题意
有n个点(<=500),标记第i个点的代价a[i],然后分别在这n个点建两棵树,对于每颗树的每个点来说,都有一个p[i],代表他的子树中被标记的点的个数需要等于p[i],请你选择需要标记的点,使得可以满足上述两棵树的要求,并且使得代价最大,输出最大的代价或者-1(不存在)
思路
- 这道题难在建图
- 首先考虑子树问题,是否选择每个节点对他的子孙节点没有影响,但是对他自身及其祖先节点有影响,注意p[i]=0代表i的子树没有限制
假设v为u的子孙节点且p[u]>0(有限制),sum=p[v1]+p[v2]+..+p[vn],
sum>p[u],则对于本棵树来说不存在满足条件的选点情况
sum<=p[u],则剩下p[u]-sum的空间给本棵树和另外一颗树选择 - 设col[u]为最靠近u的有限制的祖先节点(包括u),那么选择u就等于填充了两棵树的col[u]节点
- 所以可以这样建图,建超源超汇S,T,将第一颗树的点和源点相连,cap=p[u]-sum,cost=0,将第二颗树的点和汇点相连,cap=p[u]-sum,cost=0,将两棵树之间的点col1[u]和col2[u]相连,cap=1,cost=-a[u],跑一个mcmf
- 还需要判一下左边容量等于右边容量并且最大流等于容量,才能存在满足条件的选点情况
#include<bits/stdc++.h>
#define N 2005
#define M 4000000
#define inf 0x3f3f3f3f
using namespace std;
int n,rt1,rt2,ent,S,T,i,u,v,q,x,hd[N],p[N],col[N],f[N],d[N],inq[N],pre[N],a[N];
int flow,cost,ans,in,out;
struct edge{
int u,v,nt,c,w;
edge(int u=0,int v=0,int nt=0,int c=0,int w=0):u(u),v(v),nt(nt),c(c),w(w){}
}E[M];
void fail(){
cout<<-1;exit(0);
}
void add(int u,int v,int c,int w){
E[++ent]=edge(u,v,hd[u],c,w);hd[u]=ent;
E[++ent]=edge(v,u,hd[v],0,-w);hd[v]=ent;
}
int dfs(int u,int fa,int tar){
if(p[u])tar=u;
col[u]=tar;
int cnt=0,tp=p[u];
for(int i=hd[u];i;i=E[i].nt){
int v=E[i].v;
if(v==fa)continue;
cnt+=dfs(v,u,tar);
}
if(p[u]){if(p[u]<cnt)fail();p[u]-=cnt;cnt=tp;}
return cnt;
}
int bf(){
queue<int>Q;
for(i=0;i<=T;i++)d[i]=inf;
memset(inq,0,sizeof(inq));
d[S]=0;inq[S]=1;pre[S]=0;f[S]=inf;Q.push(S);
while(!Q.empty()){
int u=Q.front();Q.pop();
inq[u]=0;
for(int i=hd[u];i;i=E[i].nt){
edge &e=E[i];
if(e.c&&d[e.v]>d[u]+e.w){
d[e.v]=d[u]+e.w;
pre[e.v]=i;
f[e.v]=min(f[u],e.c);
if(!inq[e.v]){inq[e.v]=1;Q.push(e.v);}
}
}
}
return d[T]<inf;
}
void mcmf(){
if(in!=out)fail();
flow=0;cost=0;
while(bf()){
flow+=f[T];cost-=d[T]*f[T];
for(int i=T;i!=S;i=E[pre[i]].u){
E[pre[i]].c-=f[T];
E[pre[i]^1].c+=f[T];
}
}
if(flow!=in)fail();
cout<<cost<<endl;
}
int main(){
cin>>n>>rt1>>rt2;rt2+=n;
ent=1;S=2*n+1;T=2*n+2;
for(i=1;i<=n;i++)scanf("%d",&a[i]);
for(i=0;i<n-1;i++){scanf("%d%d",&u,&v);add(u,v,0,0);}
for(i=0;i<n-1;i++){scanf("%d%d",&u,&v);u+=n;v+=n;add(u,v,0,0);}
cin>>q;
for(i=0;i<q;i++){scanf("%d%d",&u,&x);p[u]=x;}
cin>>q;
for(i=0;i<q;i++){scanf("%d%d",&u,&x);p[u+n]=x;}
dfs(rt1,0,rt1);dfs(rt2,0,rt2);
memset(hd,0,sizeof(hd));ent=1;
in=out=0;
for(i=1;i<=n;i++){in+=p[i];if(p[i])add(S,i,p[i],0);}
for(i=1;i<=n;i++){out+=p[i+n];if(p[i+n])add(i+n,T,p[i+n],0);}
for(i=1;i<=n;i++)add(col[i],col[i+n],1,-a[i]);
mcmf();
}
Codeforces Round #523 (Div. 2) E. Politics(最小费+思维建图)的更多相关文章
- Codeforces Round #580 (Div. 2)-D. Shortest Cycle(思维建图+dfs找最小环)
You are given nn integer numbers a1,a2,…,ana1,a2,…,an. Consider graph on nn nodes, in which nodes ii ...
- Codeforces Round #406 (Div. 2) D. Legacy (线段树建图dij)
D. Legacy time limit per test 2 seconds memory limit per test 256 megabytes input standard input out ...
- Codeforces Round #523 (Div. 2)
Codeforces Round #523 (Div. 2) 题目一览表 来源 考察知识点 完成时间 A Coins cf 贪心(签到题) 2018.11.23 B Views Matter cf 思 ...
- Codeforces Round #521 (Div. 3) E. Thematic Contests(思维)
Codeforces Round #521 (Div. 3) E. Thematic Contests 题目传送门 题意: 现在有n个题目,每种题目有自己的类型要举办一次考试,考试的原则是每天只有一 ...
- Codeforces Round #523 (Div. 2) F. Katya and Segments Sets (交互题+思维)
https://codeforces.com/contest/1061/problem/F 题意 假设存在一颗完全k叉树(n<=1e5),允许你进行最多(n*60)次询问,然后输出这棵树的根,每 ...
- Codeforces Round #523 (Div. 2) Solution
A. Coins Water. #include <bits/stdc++.h> using namespace std; int n, s; int main() { while (sc ...
- Codeforces Round #523 (Div. 2) Cdp
题:https://codeforces.com/contest/1061/problem/C 题意:给你一个序列,我们求他们子序列的个数,这个子序列有个限制就是每一个子序列上的值都必须是能整除他的下 ...
- Codeforces Round #523 (Div. 2) D. TV Shows 模拟(多重集 先把所有区间加入多重集合)+贪心+二分
题意:给出n个电视节目的起始和结束时间 并且租一台电视需要x +y*(b-a) [a,b]为时段 问完整看完电视节目的最小花费是多少 思路:贪心的思想 情况1 如果新租一台电视的花费<=在空 ...
- Codeforces Round #523 (Div. 2) D. TV Shows
传送门 https://www.cnblogs.com/violet-acmer/p/10005351.html 题意: 有n个节目,每个节目都有个开始时间和结束时间. 定义x,y分别为租电视需要的花 ...
随机推荐
- Framework7框架结构
以2017年2月10日发布的Framework7 1.5.3为例. (function(){ 'use strict'; window.Framework7 = function(){ var app ...
- mvc下添加 EntityFramework的引用
首先 打开工具-扩展和更新-联机-Visual Studio库,找到NuGet Package Manager 检查是否 安装,如果没有安装 先安装插件 安装成功后,右键点击‘引用’,如下图 然后 ...
- swift - view的指定位置切圆角
1. extension UIView{ func addCorner(conrners: UIRectCorner , radius: CGFloat) { let maskPath = UIBez ...
- java 线程Thread 技术--线程创建源码解释
永远不要忘记最基础的东西,只有把最基础的知识打牢靠,才能够使你走的更远,我将从今天开始,进行线程知识的回顾,一些常用知识点,以及java1.5 引入的并发库,进行详细的讲解与总结 创建线程的目的是为了 ...
- linux命令学习之:sed
sed:Stream Editor文本流编辑,sed是一个“非交互式的”面向字符流的编辑器.能同时处理多个文件多行的内容,可以不对原文件改动,把整个文件输入到屏幕,可以把只匹配到模式的内容输入到屏幕上 ...
- MYSLQ数据库 day 1
啥是SQL? 据库的组成部分,其中数据库管理系统可以接收一些命令,对数据文件进行添加.删除.修改.查询等操作.那么这些命令就是 SQL . SQL:(Structured Query Language ...
- 集合 day8
一,集合. 集合是无序的,不重复的数据集合,它里面的元素是可哈希的(不可变类型),但是集合本身是不可哈希(所以集合做不了字典的键)的.以下是集合最重要的两点: 去重,把一个列表变成集合,就自动去重了. ...
- 8P - 钱币兑换问题
在一个国家仅有1分,2分,3分硬币,将钱N兑换成硬币有很多种兑法.请你编程序计算出共有多少种兑法. Input 每行只有一个正整数N,N小于32768. Output 对应每个输入,输出兑换方法数. ...
- UI设计课程教程分享:Banner的设计和技巧
Banner是一个网站的中心主题,可以从banner看出网站的内容.所以一个好的banner对网站的影响很大. 提高banner的制作从几点深入了解:文字排版.选择适合的图片及背景.颜色的用法. 一. ...
- Python之路(第三篇):Python基本数据类型字符串(二)
一.基本数据类型1.字符串 str字符串方法介绍(二)a --expandtabs( ) expandtabs( ) 把字符串中的 tab 符号('\t')转为空格参数默认为8,注意字符串原有的空格也 ...