[HDU4729]An Easy Problem for Elfness

题目大意:

给你一棵\(n(n\le10^5)\)个点的树,树上每条边都有容量。

\(m(m\le10^5)\)次询问,每次询问你有\(k\)的预算,可以花\(a\)的代价在任意两点间建一条流容量为\(1\)的边(包括重边),或者花费\(b\)的代价将某条边的容量加\(1\),问在不超过预算的情况下,从\(s\)到\(t\)的最大流量。

思路:

首先,考虑没有\(k,a,b\)的情况,答案就是\(s,t\)路径上的权值最小值\(min\)。

对于\(a\le b\)的情况,由于我们每在\(s\)和\(t\)之间新建一条边都能增加\(1\)的容量,则我们把所有的预算都用来新建边肯定是最优策略。答案是\(min+\lfloor\frac ka\rfloor\)。

对于\(a>b\)的情况,我们有以下两种可能最优的策略:

  1. 花费\(a\)的代价新建一条边,然后用剩下的所有预算扩充这条新边的容量。
  2. 不断将路径上最小边扩充\(1\)的容量。

对于第一种策略,我们不难得到答案就是\(min+1+\lfloor\frac{k-a}b\rfloor\)。

对于第二种策略,我们可以二分答案\(mid\)。设路径上边容量\(<mid\)的边数有\(c\)个,权值和为\(w\)。则\(c\times mid-w\)就是需要扩充的容量之和。若\(c\times mid-w\le\lfloor\frac kb\rfloor\),则\(ans>=mid\),反之\(ans<mid\)。

上文提及的边数与权值和可以用树上主席树方便地求得。

时间复杂度\(\mathcal O(n\log^2n)\)。

源代码:

#include<cstdio>
#include<cctype>
#include<climits>
#include<algorithm>
#include<forward_list>
inline int getint() {
register char ch;
while(!isdigit(ch=getchar()));
register int x=ch^'0';
while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return x;
}
using int64=long long;
constexpr int N=1e5+1,logN=17,logW=15,W=10000;
using Edge=std::pair<int,int>;
std::forward_list<Edge> e[N];
inline void add_edge(const int &u,const int &v,const int &w) {
e[u].emplace_front((Edge){v,w});
e[v].emplace_front((Edge){u,w});
}
int n,m,s,t,k,a,b;
class FotileTree {
#define mid ((b+e)>>1)
private:
struct Node {
int sum;
int64 val;
int left,right;
std::pair<int,int64> operator + (const Node &rhs) const {
return std::make_pair(sum+rhs.sum,val+rhs.val);
}
friend std::pair<int,int64> operator - (const std::pair<int,int64> &lhs,const Node &rhs) {
return std::make_pair(lhs.first-rhs.sum,lhs.second-rhs.val);
}
Node operator * (const int &rhs) const {
return {sum*rhs,val*rhs};
}
};
Node node[N*logW];
int sz,new_node(const int &p) {
node[++sz]=node[p];
return sz;
}
public:
int root[N];
void reset() {
sz=0;
}
void insert(int &p,const int &b,const int &e,const int &x) {
p=new_node(p);
node[p].sum++;
node[p].val+=x;
if(b==e) return;
if(x<=mid) insert(node[p].left,b,mid,x);
if(x>mid) insert(node[p].right,mid+1,e,x);
}
std::pair<int,int64> query(const int &p,const int &q,const int &r,const int &b,const int &e,const int64 &x) const {
if(node[p]+node[q]-node[r]*2==std::make_pair(0,0ll)) return std::make_pair(0,0);
if(x>=e) return node[p]+node[q]-node[r]*2;
if(x<=mid) {
return query(node[p].left,node[q].left,node[r].left,b,mid,x);
} else {
const auto p1=query(node[p].left,node[q].left,node[r].left,b,mid,x);
const auto p2=query(node[p].right,node[q].right,node[r].right,mid+1,e,x);
return std::make_pair(p1.first+p2.first,p1.second+p2.second);
}
}
#undef mid
};
FotileTree tr;
void reset() {
for(register int i=1;i<=n;i++) e[i].clear();
tr.reset();
}
inline int lg2(const float &x) {
return ((unsigned&)x>>23&255)-127;
}
int anc[N][logN],min[N][logN]={{INT_MAX}},dep[N];
void dfs(const int &x,const int &par) {
anc[x][0]=par;
dep[x]=dep[par]+1;
for(register int i=1;i<=lg2(dep[x]);i++) {
anc[x][i]=anc[anc[x][i-1]][i-1];
min[x][i]=std::min(min[x][i-1],min[anc[x][i-1]][i-1]);
}
for(auto &i:e[x]) {
const int &y=i.first,&w=i.second;
if(y==par) continue;
tr.insert(tr.root[y]=tr.root[x],0,W,min[y][0]=w);
dfs(y,x);
}
}
int get_min(int x,int y) {
if(dep[x]<dep[y]) std::swap(x,y);
int ret=INT_MAX;
for(register int i=lg2(dep[x]-dep[y]);i>=0;i--) {
if(dep[anc[x][i]]>=dep[y]) {
ret=std::min(ret,min[x][i]);
x=anc[x][i];
}
}
if(x==y) return ret;
for(register int i=lg2(dep[x]);i>=0;i--) {
if(anc[x][i]!=anc[y][i]) {
ret=std::min(ret,min[x][i]);
ret=std::min(ret,min[y][i]);
x=anc[x][i];
y=anc[y][i];
}
}
ret=std::min(ret,min[x][0]);
ret=std::min(ret,min[y][0]);
return ret;
}
inline int get_lca(int x,int y) {
if(dep[x]<dep[y]) std::swap(x,y);
for(register int i=lg2(dep[x]-dep[y]);i>=0;i--) {
if(dep[anc[x][i]]>=dep[y]) {
x=anc[x][i];
}
}
if(x==y) return x;
for(register int i=lg2(dep[x]);i>=0;i--) {
if(anc[x][i]!=anc[y][i]) {
x=anc[x][i];
y=anc[y][i];
}
}
return anc[x][0];
}
inline bool check(const int64 &mid) {
const auto tmp=tr.query(tr.root[s],tr.root[t],tr.root[get_lca(s,t)],0,W,mid);
return tmp.first*mid-tmp.second<=k/b;
}
int main() {
const int T=getint();
for(register int i=1;i<=T;i++) {
printf("Case #%d:\n",i);
n=getint(),m=getint();
for(register int i=1;i<n;i++) {
const int u=getint(),v=getint(),w=getint();
add_edge(u,v,w);
}
dfs(1,0);
for(register int i=0;i<m;i++) {
s=getint(),t=getint(),k=getint(),a=getint(),b=getint();
const int min=get_min(s,t);
if(a<=b) {
printf("%lld\n",min+(int64)k/a);
continue;
}
int64 l=min+(int64)k/a+1,r=min+(int64)k/b;
while(l<=r) {
const int64 mid=(l+r)>>1;
if(check(mid)) {
l=mid+1;
} else {
r=mid-1;
}
}
printf("%lld\n",k>=a?std::max(min+1+(int64)(k-a)/b,l-1):l-1);
}
reset();
}
return 0;
}

[HDU4729]An Easy Problem for Elfness的更多相关文章

  1. 数据结构(主席树):HDU 4729 An Easy Problem for Elfness

    An Easy Problem for Elfness Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 65535/65535 K (J ...

  2. HDU 4729 An Easy Problem for Elfness(主席树)(2013 ACM/ICPC Asia Regional Chengdu Online)

    Problem Description Pfctgeorge is totally a tall rich and handsome guy. He plans to build a huge wat ...

  3. HDU 4729 An Easy Problem for Elfness(树链剖分边权+二分)

    题意 链接:https://cn.vjudge.net/problem/HDU-4729 给你n个点,然你求两个点s和t之间的最大流.而且你有一定的钱k,可以进行两种操作 1.在任意连个点之间建立一个 ...

  4. 【HDOJ】4729 An Easy Problem for Elfness

    其实是求树上的路径间的数据第K大的题目.果断主席树 + LCA.初始流量是这条路径上的最小值.若a<=b,显然直接为s->t建立pipe可以使流量最优:否则,对[0, 10**4]二分得到 ...

  5. HDU 4729 An Easy Problem for Elfness (主席树,树上第K大)

    转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlove 题意:给出一个带边权的图.对于每一个询问(S , ...

  6. HDU 4729 An Easy Problem for Elfness 主席树

    题意: 给出一棵树,每条边有一个容量. 有若干次询问:\(S \, T \, K \, A \, B\),求路径\(S \to T\)的最大流量. 有两种方法可以增大流量: 花费\(A\)可以新修一条 ...

  7. UVA-11991 Easy Problem from Rujia Liu?

    Problem E Easy Problem from Rujia Liu? Though Rujia Liu usually sets hard problems for contests (for ...

  8. An easy problem

    An easy problem Time Limit:3000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Sub ...

  9. UVa 11991:Easy Problem from Rujia Liu?(STL练习,map+vector)

    Easy Problem from Rujia Liu? Though Rujia Liu usually sets hard problems for contests (for example, ...

随机推荐

  1. GPU硬件加速

    现代浏览器大都可以利用GPU来加速页面渲染.每个人都痴迷于60桢每秒的顺滑动画.在GPU的众多特性之中,它可以存储一定数量的纹理(一个矩形的像素点集合)并且高效地操作这些纹理(比如进行特定的移动.缩放 ...

  2. USB 3.1 與 USB Type-C 解釋

    https://tw.transcend-info.com/Support/FAQ-940 以下的內容皆來自上面這個網址. 什麼是USB 3.1? 什麼是USB 3.1? USB 3.1為USB協會制 ...

  3. 105.Construct Binary Tree from Preorder and Inorder Traversal---《剑指offer》面试6

    题目链接 题目大意:根据先序遍历和中序遍历构造二叉树. 法一:DFS.根据模拟步骤,直接从先序和中序数组中找值然后加入二叉树中,即先从先序数组中确定根结点,然后再去中序数组中确定左子树和右子树的长度, ...

  4. 2017多校第9场 HDU 6166 Senior Pan 堆优化Dij

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6166 题意:给你一个有向图,然后给你k个点,求其中一个点到另一个点的距离的最小值. 解法:枚举二进制位 ...

  5. 运输层和TCP/IP协议

    0. 基本要点 运输层是为相互通信的应用进程提供逻辑通信. 端口和套接字的意义 什么是无连接UDP 什么是面向连接的TCP 在不可靠的网络上实现可靠传输的工作原理,停止等待协议和ARQ协议 TCP的滑 ...

  6. ORACLE导入Excel数据

    首先建好一个和Excel表字段对应字段的表,然后 select t.* from 表名 t  for update; 点击这个锁子,打开它 粘贴,然后 再提交事务即可

  7. win7下安装 LINUX虚拟机

    文件名: VMware-workstation-full-10.0.6-2700073.exe 百度云共享链接: pan.baidu.com/s/1o6McGmI VMware workstation ...

  8. Leetcode 之Regular Expression Matching(31)

    正则表达式的匹配,还是挺难的.可根据下一个字符是不是*分为两种情况处理,需要考虑多种情况. bool isMatch(const char *s, const char *p) { if (*p == ...

  9. 关于ueditor在Java中文件上传问题,404问题

    问题困扰了两天,部署要求导入到webcontent下,我导入到了整个项目目录下,自己粗心犯错,导致页面访问不到404. 解决了上面的问题,试着进行文件上传,却一直找不到图片: 调出浏览器控制台: 刚开 ...

  10. Zookeeper 入门第一篇

    转载原文地址: ZooKeeper学习总结 第一篇:ZooKeeper快速入门 ZooKeeper学习总结 第二篇:ZooKeeper深入探讨 ZooKeeper学习第一期---Zookeeper简单 ...