NOI2006 最大获利 洛谷P4174
洛谷题目传送门!
题目描述
新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战。THU 集团旗下的 CS&T 通讯公司在新一代通讯技术血战的前夜,需要做太多的准备工作,仅就站址选择一项,就需要完成前期市场研究、站址勘测、最优化等项目。
在前期市场调查和站址勘测之后,公司得到了一共 N 个可以作为通讯信号中转站的地址,而由于这些地址的地理位置差异,在不同的地方建造通讯中转站需要投入的成本也是不一样的,所幸在前期调查之后这些都是已知数据:建立第 i个通讯中转站需要的成本为PiP_iP
另外公司调查得出了所有期望中的用户群,一共 M 个。关于第 i 个用户群的信息概括为AiA_iABiB_iBCiC_iCCiC_iCAiA_iABiB_iB
THU 集团的 CS&T 公司可以有选择的建立一些中转站(投入成本),为一些用户提供服务并获得收益(获益之和)。那么如何选择最终建立的中转站才能让公司的净获利最大呢?(净获利 = 获益之和 – 投入成本之和)
输入输出格式
输入格式:
输入文件中第一行有两个正整数 N 和 M 。
第二行中有 N 个整数描述每一个通讯中转站的建立成本,依次为P1,P2,…,PNP_1 , P_2 , …,P_NP
以下 M 行,第(i + 2)行的三个数Ai,BiA_i , B_iACiC_iC
所有变量的含义可以参见题目描述。
输出格式:
你的程序只要向输出文件输出一个整数,表示公司可以得到的最大净获利。
输入输出样例
说明
样例:选择建立 1、2、3 号中转站,则需要投入成本 6,获利为 10,因此得到最大收益 4。
很明显,直接用网络流模板。
首先考虑如何建模型:

先建立超级源点(废话), 然后连向每一个中转站,流量为成本 cost。 然后再从中转站连向使用它的用户,流量为INF (选择不影响价格,因此流量无限大)。最后从用户流向汇点,
流量为获利gain。
接着,算出所有用户gain的总值,减掉最小割(最大流)即可。 So, Why? 首先先看我们需要计算的是什么: 总获利 - 总成本。
对于亏本用户,其gain 小于 cost, 从其流出的流量肯定 <= gain。因此我们将这部分剪掉,相当于收益为0.
对于赚钱用户,其gain 大于 cost, 从其流出的流量肯定 <= cost。因此我们将这部分剪掉,相当于剪掉了cost,即为实际利润。 因此,建图完成。当前弧优化(不能忘,否则T掉)+ Dicnic 板子。 (当然,用SPFA的最小费用最大流版本也能跑,
设单位费用为1即可,但是速度好像更慢,内存更大)。那为什么还用,雾。
#include <bits/stdc++.h>
using namespace std;
#define N 500010
#define ll long long
#define INF (~0u>>1)
#define isdigit(c) ((c)>='0'&&(c)<='9') // https://www.luogu.com.cn/problem/P4174 inline int read(){
ll x = , s = ;
char c = getchar();
while(!isdigit(c)){
if(c == '-')s = -;
c = getchar();
}
while(isdigit(c)){
x = x * + c - '';
c = getchar();
}
return x * s;
} struct node{
int u, v, w;
int next = -;
}t[N];
int f[N];
int ht, s;
int deth[N], cur[N];
int n, m; int bian = -;//全部从-1开始
void addedge(ll u, ll v,ll w){
bian++;
t[bian].u = u;
t[bian].v = v;
t[bian].w = w;
t[bian].next = f[u];
f[u] = bian;
return ;
} inline void add(ll u, ll v,ll w){ //注意要加反边。这里就如此写了
addedge(u, v, w);
addedge(v, u, );
return ;
}
queue <int> q;
bool bfs(int s, int ht){
memset(deth, , sizeof(deth));
while(!q.empty())q.pop();//进行初始化
q.push(s);
deth[s] = ;//起点记得设为-1
while(!q.empty()){
int now = q.front();q.pop();
for(int i = f[now]; ~i; i = t[i].next){
int v = t[i].v, u = t[i].u, w = t[i].w;
if(!deth[v] && w > ){
deth[v] = deth[u] + ; // 分层操作
q.push(v);
}
}
}
return deth[ht] != ;
} ll dfs(int now,int dist){
if(now == ht)return dist;
for(int& i = cur[now]; ~i; i = t[i].next){ // 记得当前弧优化
int w = t[i].w, v = t[i].v, u = t[i].u;
if(deth[v] == deth[u] + && w){
int di = dfs(v, min(dist, w));
if(di > ){
t[i].w -= di;
t[i^].w += di;
return di;
}
}
}
return ;
}
ll Dicnic(){ // 经典Dicnic 操作,就是板子
ll ans = ;
while(bfs(s, ht)){
memcpy(cur, f, sizeof(cur));
while(ll temp = dfs(s,INF))
ans += temp;
}
return ans;
} int main(){
// freopen("P4174_9.in", "r", stdin);
memset(f, -, sizeof(f));
n = read(), m = read();
for(int i = ;i <= n; i++){
int temp = read();
add(, i, temp); // 从源点连向中转站
}
ll ans = ;
int to1, to2, temp;
for(int i = ;i <= m; i++){
int to1 = read(), to2 = read(), temp = read();
add(to1, i + n, INF);
add(to2, i + n, INF);
add(i + n, + m + n, temp);//从用户连向汇点
ans += temp;
}
s = , ht = n + m + ;
ans -= Dicnic();//减掉届不到的用户和届到的中转站
printf("%lld\n", ans);
return ;
}
典型的脑子题,动手简单动脑难
NOI2006 最大获利 洛谷P4174的更多相关文章
- 洛谷 P4174 [NOI2006]最大获利 && 洛谷 P2762 太空飞行计划问题 (最大权闭合子图 && 最小割输出任意一组方案)
https://www.luogu.org/problemnew/show/P4174 最大权闭合子图的模板 每个通讯站建一个点,点权为-Pi:每个用户建一个点,点权为Ci,分别向Ai和Bi对应的点连 ...
- 洛谷 P4174 [NOI2006]最大获利 解题报告
P4174 [NOI2006]最大获利 题目描述 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战.THU 集团旗下的 CS&T 通讯公司在新一代通讯技术血战的前夜,需要 ...
- 洛谷P4174 [NOI2006]最大获利(最大流)
题目描述 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战.THU 集团旗下的 CS&T 通讯公司在新一代通讯技术血战的前夜,需要做太多的准备工作,仅就站址选择一项,就需 ...
- [洛谷P4174][NOI2006]最大获利
题目大意:同Petya and Graph,数据范围改成$n\leqslant5\times10^3,m\leqslant5\times10^4$ 题解:同上 卡点:无 C++ Code: #incl ...
- P4174 [NOI2006]最大获利(网络流)
P4174 [NOI2006]最大获利 还是最大权闭合子图的题 对于每个中转站$k$:$link(k,T,P_k)$ 对于每个用户$i$.中转站$A_i,B_i$.贡献$C_i$ $link(S,i, ...
- P4174 [NOI2006]最大获利 (最大权闭合子图)
P4174 [NOI2006]最大获利 (最大权闭合子图) 题目链接 题意 建\(i\)站台需要\(p_i\)的花费,当\(A_i,B_i\)都建立时获得\(C_i\)的利润,求最大的利润 思路 最大 ...
- 洛谷 P2949 [USACO09OPEN]工作调度Work Scheduling 题解
P2949 [USACO09OPEN]工作调度Work Scheduling 题目描述 Farmer John has so very many jobs to do! In order to run ...
- [NOI2006] 最大获利
[NOI2006] 最大获利 ★★★☆ 输入文件:profit.in 输出文件:profit.out 简单对比时间限制:2 s 内存限制:512 MB [问题描述] 新的技术正冲击着手 ...
- 洛谷1640 bzoj1854游戏 匈牙利就是又短又快
bzoj炸了,靠离线版题目做了两道(过过样例什么的还是轻松的)但是交不了,正巧洛谷有个"大牛分站",就转回洛谷做题了 水题先行,一道傻逼匈牙利 其实本来的思路是搜索然后发现写出来类 ...
随机推荐
- postman(介绍)
Postman 界面介绍 一. 安装后首次打开 postman,会提示你是否需要登录,登录的话可以云端保存你的收藏及历史记录,不登陆不影响使用. 二. 进入后就是如下图所示的界面了.看到这么多按钮 ...
- 一个简单的wed服务器SHTTPD(9)————main函数文件,Makefile,头文件
主函数: #include "lcw_shttpd.h" //初始化时服务器的默认配置 extern struct conf_opts conf_para= { "/us ...
- 面试中的volatile关键字
在Java的面试当中,面试官最爱问的就是volatile关键字相关的内容.经过多次面试之后,你是否思考过,为什么他们那么爱问volatile关键字相关的问题?而对于你,如果作为面试官,是否也会考虑采用 ...
- java的Timer定时器任务
在项目开发中,经常会遇到需要实现一些定时操作的任务,写过很多遍了,然而每次写的时候,总是会对一些细节有所遗忘,后来想想可能是没有总结的缘故,所以今天小编就打算总结一下可能会被遗忘的小点: 1. pub ...
- Java——理解面向对象
1.程序设计的三种基本结构 顺序结构 顺序结构表示程序中的各操作是按照它们在源代码中的排列顺序依次执行的 选择结构 选择结构表示程序的处理需要根据某个特定的条件选择其中的一个分支执行.选择结构有单选择 ...
- Java种sleep和wait的区别
1,sleep方法是Thread类的静态方法,wait()是Object超类的成员方法 2,sleep()方法导致了程序暂停执行指定的时间,让出cpu该其他线程,但是他的监控状态依然保持者,当指定的时 ...
- Python学习之字符串中的下标和切片以及逆序
python中的下标从0开始 从后往前取 注意:后面的2代表步长,先看2:-1取出来的数值 [起始位置:终止位置:步长] 逆序 但是发现如果[-1:0:-1]发现是取不到第一个元素的,那么怎么办? 此 ...
- 第一行Kotlin系列(三)Intent 向上一页返回数据onActivityResult的使用
1.MainActivity.kt跳转处理 声明全局的按钮对象 private lateinit var button8: Button 实例化按钮对象 button8 = findViewById( ...
- [hdu5416 CRB and Tree]树上路径异或和,dfs
题意:给一棵树,每条边有一个权值,求满足u到v的路径上的异或和为s的(u,v)点对数 思路:计a到b的异或和为f(a,b),则f(a,b)=f(a,root)^f(b,root).考虑dfs,一边计算 ...
- Redis学习笔记(九) AOF持久化
除了RDB持久化功能之外,Redis还提供了AOF持久化功能.与RDB持久化通过保存数据库中的键值对来记录数据库状态不同,AOF持久化是通过保存Redis服务器所执行的写命令来记录数据库状态的. 服务 ...