HackerRank savita-and-friends
Description
在一条边上求一个点,使得这个点到所有点的最长的最短距离 最短. \(n \leqslant 10^5\)
Sol
Dijkstra+扫描线+单调队列.
这个好像叫什么最小直径生成树?不过变成了在一个边上...
对于一个点,我们可以求出当分割点在哪个位置的时候,它应该到 \(A\) ,什么时候它应该到 \(B\) .
\(dis_A(i)+t \leqslant dis_B(j)+(dis_{AB}-t)\)
移项得 \(t\leqslant \frac{dis_B(i)-dis_A(i)+dis_{AB}}{2}\)
这样就把线段 \(AB\) 分成了几个区间,枚举区间,然后分别用两个单调队列维护一下到分割点左右到 \(A,B\) 距离的最大值.
对于一个区间,我们可以 \(O(1)\) 的求出在这个区间应该选择哪个点作为分割点,因为两边的点都是确定的最大距离也知道,然后统计答案就可以了.
Code
#include<cstdio>
#include<cstring>
#include<vector>
#include<queue>
#include<algorithm>
#include<functional>
#include<iostream>
using namespace std; #define mpr(a,b) make_pair(a,b)
#define debug(a) cout<<#a<<"="<<a<<" "
#define Fst first
#define Sec second
typedef long long LL;
typedef pair< LL,int > pr;
const int N = 100005; LL T,n,m,k,A,B,X;
vector<pr> g[N];
LL ds[N],dt[N];
pair< double,int > p[N];double lm[N];
pair< double,double > ans;
int q1[N],q2[N],h1,h2,t1,t2;
priority_queue<pr,vector<pr>,greater<pr> > q; inline LL in(LL x=0,char ch=getchar()){ while(ch>'9' || ch<'0') ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();return x; }
void clr(){
t1=t2=0,h1=h2=1;ans=mpr(1e18,-1);
for(int i=0;i<N;i++) g[i].clear();
}
void Up(int i,int j){
double l=max(min((X+ds[j]+dt[i])/2.0-ds[j],lm[j]),lm[i]),tmp=max(ds[j]+l,X-l+dt[i]);
// debug(i),debug(j),debug(dt[i]),debug(ds[j]),debug((X+ds[j]+dt[i])/2.0-dt[i]),debug(lm[i]),debug(lm[j]),debug(l),debug(tmp)<<endl;
if(tmp < ans.Fst||(tmp == ans.Fst && l<ans.Sec)) ans=mpr(tmp,l);
}
void Dijkstra(int s,LL *d){
static bool b[N];memset(b,0,sizeof(b));for(int i=1;i<=n;i++) d[i]=1000000000000000000LL;
d[s]=0,q.push(mpr(0,s));
for(int u;!q.empty();){
u=q.top().Sec,q.pop();if(b[u]) continue;b[u]=1;
for(int i=0;i<g[u].size();i++){
LL p=g[u][i].Fst;int v=g[u][i].Sec;
if(d[u]+p < d[v]) d[v]=d[u]+p,q.push(mpr(d[v],v));
}
}
}
int main(){
for(T=in();T--;){
clr();
n=in(),m=in(),k=in();
for(int i=1,u,v,x;i<=m;i++){
u=in(),v=in(),x=in();
g[u].push_back(mpr(x,v)),g[v].push_back(mpr(x,u));
if(i == k) A=u,B=v,X=x;
}
Dijkstra(A,ds),Dijkstra(B,dt); // cout<<X<<endl; // for(int i=1;i<=n;i++) cout<<ds[i]<<" ";cout<<endl;
// for(int i=1;i<=n;i++) cout<<dt[i]<<" ";cout<<endl; for(int i=1;i<=n;i++) p[i]=mpr((dt[i]-ds[i]+X)/2.0,i),lm[i]=(dt[i]-ds[i]+X)/2.0;
sort(p+1,p+n+1); // for(int i=1;i<=n;i++) printf("%lf %d\n",p[i].Fst,p[i].Sec); for(int i=1;i<=n;i++){
while(h2<=t2 && ds[q2[t2]] <= ds[p[i].Sec]) t2--;
q2[++t2]=p[i].Sec;
}
// cout<<ds[q2[h2]]+X<<endl;
ans=mpr(ds[q2[h2]],0);
for(int i=1,r;i<=n;i=r+1){
r=i;
while(p[r].Fst == p[r+1].Fst) r++;
/// debug(r)<<endl;
for(int j=i;j<=r;j++){ while(h1<=t1 && dt[q1[t1]] <= dt[p[j].Sec]) t1--;q1[++t1]=p[j].Sec; }
for(int j=i;j<=r;j++) if(q2[h2] == p[j].Sec) h2++;
// debug(q1[h1]),debug(q2[h2])<<endl;
Up(q1[h1],q2[h2]);
}
if(dt[q1[h1]]<ans.Fst) ans=mpr(dt[q1[h1]],X);
printf("%.5lf %.5lf\n",ans.Sec,ans.Fst);
}
return 0;
}
HackerRank savita-and-friends的更多相关文章
- 日常小测:颜色 && Hackerrank Unique_colors
题目传送门:https://www.hackerrank.com/challenges/unique-colors 感谢hzq大神找来的这道题. 考虑点分治(毕竟是路经统计),对于每一个颜色,它的贡献 ...
- HackerRank "Square Subsequences" !!!
Firt thought: an variation to LCS problem - but this one has many tricky detail. I learnt the soluti ...
- HackerRank "Minimum Penalty Path"
It is about how to choose btw. BFS and DFS. My init thought was to DFS - TLE\MLE. And its editorial ...
- HackerRank "TBS Problem" ~ NPC
It is marked as a NPC problem. However from the #1 code submission (https://www.hackerrank.com/Charl ...
- HackerRank Extra long factorials
传送门 今天在HackerRank上翻到一道高精度题,于是乎就写了个高精度的模板,说是模板其实就只有乘法而已. Extra long factorials Authored by vatsalchan ...
- HackerRank "Lucky Numbers"
Great learning for me:https://www.hackerrank.com/rest/contests/master/challenges/lucky-numbers/hacke ...
- HackerRank "Playing with numbers"
This is 'Difficult' - I worked out it within 45mins, and unlocked HackerRank Algorithm Level 80 yeah ...
- HackerRank "The Indian Job"
A sly knapsack problem in disguise! Thanks to https://github.com/bhajunsingh/programming-challanges/ ...
- HackerRank "Array and simple queries" !
The most interesting, flexible and juicy binary tree problem I have ever seen. I learnt it from here ...
- HackerRank "Self Balancing Tree"
Something to learn: Rotation ops in AVL tree does not require recursion. https://github.com/andreima ...
随机推荐
- Java观察者设计模式
在java.util包中提供了Observable类和Observer接口,使用它们即可完成观察者模式. 多个观察者都在关注着价格的变化,只要价格一有变化,则所有的观察者会立即有所行动. //==== ...
- Java关键字——super
使用super关键字可以从子类中调用父类中的构造方法.普通方法和属性 与this调用构造方法的要求一样,语句必须放在子类构造方法的首行 this和super都可以调用构造方法,但是两者不能同时出现,调 ...
- 使用ASP.NET Web Api构建基于REST风格的服务实战系列教程【三】——Web Api入门
系列导航地址http://www.cnblogs.com/fzrain/p/3490137.html 前言 经过前2节的介绍,我们已经把数据访问层搭建好了,从本章开始就是Web Api部分了.在正式开 ...
- js取float型小数点后两位数的方法
四舍五入以下处理结果会四舍五入:' var num =2.446242342; num = num.toFixed(2); // 输出结果为 2.45 不四舍五入以下处理结果不会四舍五入:第一种, ...
- cannot start session without errors
访问phpmyadmin出现如标题错误,解决方案如下: cd /var/lib/php 更改目录权限chown nginx:nginx -R session/ That's All!
- VS代码段扩展Snippet Designer is a Visual Studio plug in which allows you to create and search for snippets inside the IDE
Snippet Designer is a Visual Studio plug in which allows you to create and search for snippets insid ...
- redis使用watch完成秒杀抢购功能(转)
redis使用watch完成秒杀抢购功能: 使用redis中两个key完成秒杀抢购功能,mywatchkey用于存储抢购数量和mywatchlist用户存储抢购列表. 它的优点如下: 1. 首先选用内 ...
- vim配色方案设置(更换vim配色方案)
vim配色后,我的 设定底色为黑色,字体为绿色,然后将文件夹设为洋红,默认的注释换为淡黄:其实有一种简单的方法,就是设定为系统配置好的配色方案:转载文章如下: ---------------- ( ...
- linux安装-版本选择-终极决定
选用64位或32位的版本,注意看硬件: 内存大于4G的用64位, 小于4G的用32位 同时, 64位的版本在软件源, 软件的兼容性等问题. ----------------------------- ...
- linuxMint下安装ftp工具--filezilla
windows下ftp工具有好多,linux下推荐用filezilla 安装filezilla很简单,安装完后,使用方式和windows下面一样. 第一种方式: 直接去filezilla官网下载软件包 ...