Codeforces Round #378 (Div. 2)F - Drivers Dissatisfaction GNU
http://codeforces.com/contest/733/problem/F
题意:给你一些城市和一些路,每条路有不满意程度和每减少一点不满意程度的花费,给出最大花费,要求找出花费小于s的最小生成树中最小的不满意程度
题解:首先明确的是肯定只删除一条路然后其他的找最小生成树即可,但是直接搞复杂度O(m*mlogm)所以先prim扣出一颗最小生成树,然后枚举所有边删这条边,如果在最小生成树上,直接减到贡献即可,如果不在最小生成树上,那么加上这条边,树肯定有环了,我们用lca暴力的扣出这个环最大的不满意程度的那条边,删掉即可,然后算贡献。总的复杂度是O(mlogm+m*logn)
//#pragma comment(linker, "/stack:200000000")
//#pragma GCC optimize("Ofast,no-stack-protector")
//#pragma GCC target("sse,sse2,sse3,ssse3,sse4,popcnt,abm,mmx,avx,tune=native")
//#pragma GCC optimize("unroll-loops")
#include<bits/stdc++.h>
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define pi acos(-1.0)
#define ll long long
#define vi vector<int>
#define mod 1000000009
#define C 0.5772156649
#define ls l,m,rt<<1
#define rs m+1,r,rt<<1|1
#define pil pair<int,ll>
#define pli pair<ll,int>
#define pii pair<int,int>
#define cd complex<double>
#define ull unsigned long long
#define base 1000000000000000000
#define fio ios::sync_with_stdio(false);cin.tie(0) using namespace std; const double g=10.0,eps=1e-;
const int N=+,maxn=+,inf=0x3f3f3f3f,INF=0x3f3f3f3f3f3f3f3f; struct edge{
int u,v,id;ll w,c;
bool operator <(const edge&rhs)const{
return w<rhs.w;
}
}e[N];
int father[N],fa[][N],id[][N],n,m,dep[N];
ll ma[][N],s;
bool vis[N];
vector<edge>v[N];
int Find(int x)
{
return father[x]==x?x:father[x]=Find(father[x]);
}
bool cmp(edge a,edge b)
{
return a.id<b.id;
}
ll prim()
{
for(int i=;i<=m;i++)scanf("%lld",&e[i].w);
for(int i=;i<=m;i++)scanf("%lld",&e[i].c);
for(int i=;i<=m;i++)
{
e[i].id=i;
scanf("%d%d",&e[i].u,&e[i].v);
}
for(int i=;i<=n;i++)father[i]=i;
sort(e+,e++m);
ll sum=;
for(int i=;i<=m;i++)
{
int x=e[i].u,y=e[i].v;
int fx=Find(x),fy=Find(y);
if(fx!=fy)
{
father[fx]=fy;
vis[e[i].id]=;
sum+=e[i].w;
}
}
sort(e+,e++m,cmp);
for(int i=;i<=m;i++)
{
if(vis[i])
{
v[e[i].u].pb(e[i]);
swap(e[i].u,e[i].v);
v[e[i].u].pb(e[i]);
}
}
return sum;
}
void dfs(int u,int f,int d)
{
dep[u]=d;
for(int i=;i<v[u].size();i++)
{
int x=v[u][i].v;
if(x!=f)
{
fa[][x]=u;
ma[][x]=v[u][i].w;
id[][x]=v[u][i].id;
dfs(x,u,d+);
}
}
}
void init()
{
dfs(,-,);
// for(int i=1;i<=n;i++)printf("%d %lld %d \n",fa[0][i],ma[0][i],id[0][i]);
for(int i=;i<;i++)
{
for(int j=;j<=n;j++)
{
fa[i][j]=fa[i-][fa[i-][j]];
if(ma[i-][fa[i-][j]]>ma[i-][j])
{
ma[i][j]=ma[i-][fa[i-][j]];
id[i][j]=id[i-][fa[i-][j]];
}
else
{
ma[i][j]=ma[i-][j];
id[i][j]=id[i-][j];
}
}
}
}
int lca(int x,int y)
{
if(dep[x]>dep[y])swap(x,y);
int ans=;ll maxx=;
for(int i=;i<;i++)
{
if((dep[y]-dep[x])>>i&)
{
if(maxx<ma[i][y])
{
maxx=ma[i][y];
ans=id[i][y];
}
y=fa[i][y];
}
}
if(x==y)return ans;
for(int i=;i>=;i--)
{
if(fa[i][x]!=fa[i][y])
{
if(maxx<ma[i][x])
{
maxx=ma[i][x];
ans=id[i][x];
}
if(maxx<ma[i][y])
{
maxx=ma[i][y];
ans=id[i][y];
}
x=fa[i][x];
y=fa[i][y];
}
}
if(maxx<ma[][x])
{
maxx=ma[][x];
ans=id[][x];
}
if(maxx<ma[][y])
{
maxx=ma[][y];
ans=id[][y];
}
return ans;
}
void solve(ll sum)
{
ll ans=1e18,pos;
for(int i=;i<=m;i++)
{
if(vis[i])
{
if(ans>sum-s/e[i].c)
{
ans=sum-s/e[i].c;
pos=i;
}
}
else
{
int ff=lca(e[i].u,e[i].v);
if(ans>sum-e[ff].w+e[i].w-s/e[i].c)
{
ans=sum-e[ff].w+e[i].w-s/e[i].c;
pos=i;
}
}
}
if(!vis[pos])vis[lca(e[pos].u,e[pos].v)]=;
printf("%lld\n",ans);
for(int i=;i<=m;i++)
{
if(pos==i)printf("%d %d\n",i,e[i].w-s/e[i].c);
else if(vis[i])printf("%d %d\n",i,e[i].w);
}
}
int main()
{
scanf("%d%d",&n,&m);
ll sum=prim();
scanf("%lld",&s);
// for(int i=1;i<=n;i++)
// {
// for(int j=0;j<v[i].size();j++)
// {
// printf("%d %d %d %lld %lld\n",v[i][j].u,v[i][j].v,v[i][j].id,v[i][j].w,v[i][j].c);
// }
// puts("");
// }
init();
solve(sum);
return ;
}
/***********************
3 3
1 1 1000
5 5 2
2 1
3 1
3 2
2
***********************/
Codeforces Round #378 (Div. 2)F - Drivers Dissatisfaction GNU的更多相关文章
- Codeforces Round #378 (Div. 2) F - Drivers Dissatisfaction
F - Drivers Dissatisfaction 题目大意:给你n个点,m条边,每个边都有一个权重w,每条边也有一个c表示,消耗c元可以把这条边的权重减1,求最多消耗s元的最小生成树. 思路:因 ...
- Codeforces Round #378 (Div. 2)F
题目:一个带权连通无向图,给第i条边权值减1需要花费ci元,你一共有S元,求最小生成树. 容易得出钱全部花在一条边上是最优的. 我们先做一遍最小生成树. 然后我们枚举减哪一条边. 如果这条边是树上的, ...
- codeforce 378 div 2 F —— Drivers Dissatisfaction (最小生成树,LCA,倍增)
官方题解: If you choose any n - 1 roads then price of reducing overall dissatisfaction is equal to min(c ...
- Codeforces Round #485 (Div. 2) F. AND Graph
Codeforces Round #485 (Div. 2) F. AND Graph 题目连接: http://codeforces.com/contest/987/problem/F Descri ...
- Codeforces Round #486 (Div. 3) F. Rain and Umbrellas
Codeforces Round #486 (Div. 3) F. Rain and Umbrellas 题目连接: http://codeforces.com/group/T0ITBvoeEx/co ...
- Codeforces Round #501 (Div. 3) F. Bracket Substring
题目链接 Codeforces Round #501 (Div. 3) F. Bracket Substring 题解 官方题解 http://codeforces.com/blog/entry/60 ...
- Codeforces Round #499 (Div. 1) F. Tree
Codeforces Round #499 (Div. 1) F. Tree 题目链接 \(\rm CodeForces\):https://codeforces.com/contest/1010/p ...
- Codeforces Round #378 (Div. 2) D题(data structure)解题报告
题目地址 先简单的总结一下这次CF,前两道题非常的水,可是第一题又是因为自己想的不够周到而被Hack了一次(或许也应该感谢这个hack我的人,使我没有最后在赛后测试中WA).做到C题时看到题目情况非常 ...
- Codeforces Round #376 (Div. 2)F. Video Cards(前缀和)
题目链接:http://codeforces.com/contest/731/problem/F 题意:有n个数,从里面选出来一个作为第一个,然后剩下的数要满足是这个数的倍数,如果不是,只能减小为他的 ...
随机推荐
- B-、B+、B*树
B-树:多路搜索树,每个结点存储M/2到M个关键字,非叶子结点存储指向关键字范围的子结点:所有关键字在整颗树中出现,且只出现一次,非叶子结点可以命中: B+树:在B-树基础上,为叶子结点增加链表指针, ...
- python16_day20【Django_继续抽屉项目】
一.djangoAdmin和表结构 1.项目名称 python manage startapp web # 前端页面 python manage startapp repository # 只是数 ...
- 基于HTML5 FileSystem API的使用介绍(转)
FileSystem提供了文件夹和文件的创建.移动.删除等操作,大大方便了数据的本地处理, 而且所有的数据都是在沙盒(sandboxed)中,不同的web程序不能互相访问,这就保证了数据 的完整和安全 ...
- (译)Windows Azure的7月更新:SQL数据库,流量管理,自动缩放,虚拟机
Windows Azure的7月更新:SQL数据库,流量管理,自动缩放,虚拟机 今早我们释出一些很棒的Windows Azure更新.这些新的提升包括:SQL数据库:支持SQL自动导出和一个新的高级层 ...
- 《Unity3D》通过对象池模式,管理场景中的元素
池管理类有啥用? 在游戏场景中,我们有时候会需要复用一些游戏物体,比如常见的子弹.子弹碰撞类,某些情况下,怪物也可以使用池管理,UI部分比如:血条.文字等等 这些元素共同的特性是:存在固定生命周期,使 ...
- ios 常见问题解决 以及小技巧
1.使用cocoaPods引用第三方类库,报错:file not found . 解决方案:设置 Project->Info->Configurations之后 clear ,然后再 ...
- 吉哥系列故事——完美队形I
/*hud4512 dp[i]表示当前以下标i结束的最长公共上升子序列. 我们让第一个序列为原序列,第二个序列为原系列的反向. 则,也就是说,第二个序列的顺序为原序列的下标[n-1,0],设为j 当j ...
- snapshot与release
总结自:https://www.jianshu.com/p/084fd2408d9a 这两个概念是用于描述jar包,jar包提供给其他系统作为依赖. 1. snapshot版本代表不稳定.尚处于开发中 ...
- PHP 利用文件锁处理高并发
利用flock()函数对文件进行加锁(排它锁),实现并发按序进行. flock(file,lock,block)有三个参数. file:已经打开的文件 lock:锁的类型 LOCK_SH:共享锁(读锁 ...
- 扒开系统调用的三层皮(下)/给MenuOS增加time和time-asm命令
上周从用户态的角度去理解系统调用 这周通过内核的方式 调试和跟踪系统调用来理解 rm menu -rf 强制删除原menu文件 git clone https://github.com/mengni ...