首页 > 其他分享 >【21 ZR联赛集训 day10】不知道高到哪里去了

【21 ZR联赛集训 day10】不知道高到哪里去了

时间:2024-09-27 14:24:14浏览次数:10  
标签:ch ei 21 pf int que day10 ZR dis

【21 ZR联赛集训 day10】不知道高到哪里去了

二分答案。设敌人的速度是 \(1\),二分我的速度 \(v\),我可以从 \(C\) 走到 \(T\) 当对于每个我到达的点 \(u\),敌人无法比我先到达,即敌人到达 \(u\) 最短用时比我大。

先求敌人到每个结点的最短路,然后对于二分的一个 \(v\),从 \(C\) 开始搜,我到一个点的最短用时小于敌人到这个点的最短用时时,我可以到达这个点,维护 \(C\) 出发最短路即可(\(t=\frac{s}{v}\))。如果我可以到达 \(T\) 返回 \(true\)。

Code

#include<bits/stdc++.h>
#define pf printf
#define sf scanf
#define rep(x,y,z) for(int x=y;x<=z;x++)
#define isdigit(x) (x>='0'&&x<='9')
//#define int ll
using namespace std;
typedef long long ll;
const int N=1e4+7,M=5e5+7,inf=0x7f7f7f7f;
template <typename T>
void read(T &sum){
	sum=0;
	T fl=1;
	char ch=getchar();
	for(;!isdigit(ch);ch=getchar()) (ch=='-')&&(fl=-1);
	for(;isdigit(ch);ch=getchar()) sum=(sum<<3)+(sum<<1)+ch-'0';
	sum=sum*fl;
}
template <typename T>
void write(T x){
	static int st[36];
	int top=0;
	if(x<0) putchar('-'),x=-x;
	do{
		st[top++]=x%10,x/=10;
	}while(x);
	while(top) putchar(st[--top]+'0');
}
template <typename T>
void write(T x,char ch){
	write(x);
	putchar(ch);
}

int head[N],cnt;
struct edge{
	int to,val,ne;
}e[M<<1];
void addedge(int u,int v,int w){
	e[++cnt]={v,w,head[u]};head[u]=cnt;
}
map<pair<int,int>,int> mp;
vector<pair<int,int> > vec;

int n,m;
int u,v,w;
int S,A,T;

int dis[2][N],vi[N];
struct qst{
	int x,d;
	bool operator <(const qst b)const{
		return d>b.d;
	}
};
void getdis(int rt,int op=0){
//	pf("getdis(%d)\n",rt);
	memset(dis[op],0x7f,sizeof(dis[op]));
	memset(vi,0,sizeof(vi));
	priority_queue<qst> que;
	dis[op][rt]=0;
	que.push({rt,0});
	while(!que.empty()){
		qst k=que.top();
		que.pop();
		int u=k.x,d=k.d;
		if(vi[u]) continue;
//		pf("%d %d\n",u,d);
		vi[u]=1;
		for(int i=head[u];i;i=e[i].ne){
			int v=e[i].to,w=e[i].val;
			if(d+w<dis[op][v]){
				dis[op][v]=d+w;
				que.push({v,dis[op][v]});
			}
		}
	}
}

bool dfs(int rt,double vv,int op=1){
	memset(dis[op],0x7f,sizeof(dis[op]));
	memset(vi,0,sizeof(vi));
	priority_queue<qst> que;
	dis[op][rt]=0;
	que.push({rt,0});
	while(!que.empty()){
		qst k=que.top();
		que.pop();
		int u=k.x,d=k.d;
		if(vi[u]) continue;
		if(1.0*dis[1][u]/vv>=1.0*dis[0][u]) {dis[1][u]=inf;continue; }
//		pf("%d %d\n",u,d);
		vi[u]=1;
		for(int i=head[u];i;i=e[i].ne){
			int v=e[i].to,w=e[i].val;
			if(d+w<dis[op][v]){
				dis[op][v]=d+w;
				que.push({v,dis[op][v]});
			}
		}
	}
	return dis[op][T]!=inf;
}
bool check(double v){
	return dfs(S,v);
}

int main(){
//	freopen("test.out","w",stdout);
	read(n),read(m);
	rep(i,1,m){
		read(u),read(v),read(w);
		if(u==v) continue;
		vec.push_back({u,v});
		if(mp.find({u,v})==mp.end()) mp[{u,v}]=mp[{v,u}]=w;
		else mp[{u,v}]=mp[{v,u}]=min(mp[{u,v}],w);
	}
	for(pair<int,int> ei:vec){
		if(mp[ei]) {
			addedge(ei.first,ei.second,mp[ei]);
			addedge(ei.second,ei.first,mp[ei]);
			mp[ei]=0;
		}
	}
	read(S),read(A),read(T);
	if(S==A){
		pf("-1\n");return 0;
	}
	if(S==T){
		pf("0.00000\n");return 0;
	}
	getdis(A,0);
//	pf("%lld\n",dis[0][3]);
//	getdis(S,1);
	double l=0,r=100*N;
	while(l<r-0.00001){
		double mid=(l+r)/2;
//		pf("mid %.6lf\n",mid);
		if(check(mid)) r=mid;
		else l=mid;
	}
	if(r==100*N) pf("-1\n");
	else pf("%.5lf\n",r);
}

标签:ch,ei,21,pf,int,que,day10,ZR,dis
From: https://www.cnblogs.com/liyixin0514/p/18435629

相关文章

  • 【21 ZR联赛集训 day18】聚会
    【21ZR联赛集训day18】聚会给出一个由小的编号连向大的编号的DAG,有\(q\)次询问,每次给出\(t\)和若\(s\)个点,表示除这些点之外其他点到\(t\)的最大距离。问距离最远的那个结点编号。\(1\len\le10^5,1\lem\le2\times10^5,\sums\le10^5\)。根号分治。对每个点......
  • 【21 ZR联赛集训 day18】游戏
    【21ZR联赛集训day18】游戏给定长度为\(n\)的序列\(A,B\),每个数形如\(\frac{2^{p_1}3^{p_2}5^{p_3}7^{p_4}11^{p_5}13^{p_6}}{2^{p_7}3^{p_8}5^{p_9}7^{p_{10}}11^{p_{11}}13^{p_{12}}}\)。可以进行若干次操作,每次操作选定\(i(2\lei\len-1),(a_{i-1},a_i,a_{i+1})\gets......
  • 【2021提高组十连测day7】a
    【2021提高组十连测day7】a题意:求每个点和其他所有点曼哈顿距离的平方之和。形式化地,求\(\sum_{j=1}^{n}(|x_i-x_j|+|y_i-y_j|)^2(1\lei\len)\)。对于每个点,我们把其他点分成四部分:左上、左下、右上、右下。四个部分可以通过旋转整个坐标系来解决。因此这里讲如何处理左上......
  • 【21 ZR联赛集训 day10】跑得比谁都快
    【21ZR联赛集训day10】跑得比谁都快\(O(nq)\)做法显然,不讲。如果我们把所有红绿灯的位置\(mod(g+r)\),放到数据结构里,就可以\(O(\logn)\)的时间内找到第一个红灯的位置。然后我们预处理每个红绿灯红灯结束的时刻开始,走到终点要用的时间\(f_i\),DP倒序求解。对于每个询......
  • 【20联赛集训day10】玩游戏
    【20联赛集训day10】玩游戏给一个长度为\(n\)的序列,\(|a_i|\le10^{13}\)。给出一个\(k\)问从\(k\)出发不断向两端拓展,满足任何时刻的区间和\(\le0\),问能否拓展到区间\((1,n]\)。考虑贪心,分别维护\(k\)左边和右边的区间,维护一个指针。每次贪心地向一边走,走到能走到......
  • 【20zr提高组十连测day10】心
    【20zr提高组十连测day10】心首先不同的操作序列一定在某个时刻使数组内容不同,因此我们只需要统计合法的操作序列数量。一个合法的最终数组形如若干个\(1,M\),而且\(1,M\)之间可能有若干个\(x\),长度为\(n+1\)。造成这个数组的操作序列必须满足所有操作\(1,M\)按顺序排列,......
  • 【20zr提高组十连测day10】信
    【20zr提高组十连测day10】信给定\(n,m\),\(n,m\le10^5\),给定分别长度为\(n-1,m,n,m-1\)的单调不减的序列\(A,B,C,D\),然后形如该图建边:考虑到序列是递增的,对于除最左上角以外的每个点,每个点一定要选和自己相连的一条边才能形成一棵树。那么选择左边或上边一定是更优的,而且......
  • 【20联赛集训day10】排列
    【20联赛集训day10】排列一个长度为\(n\)的排列,每次操作删除所有存在相邻的数字比它更大的数字,问执行\(k\)次操作后剩下恰好一个数的排列方案数。首先因为每次删除至少删一半的数字,所以最多删\(\log\)次。对于一个排列,我们可以发现把序列按最大值劈开,左右两边互不干扰,成......
  • [SWPUCTF 2021 新生赛]easy_md5
    分析一下代码, name不等于password,namd,md5值和password,md5值相等。get传参name,post传参password。$name!=$password&&md5($name)==md5($password)属于MD5绕过中的php==弱类型绕过有两个办法,第一个办法就是importrequests#网站的URLurl="http://node7.a......
  • 【LeetCode Hot 100】21. 合并两个有序链表
    题目描述最简单粗暴的想法是将两个链表的所有元素拿出来放在一个数组中,再将此数组排序,最后生成一个新链表并返回。由于给出的参数本身就是两个排好序的链表,可以进行一次遍历,就能保证元素仍然是有序的:每次比较当前指针指向的两个元素的大小,较小的拿出来作为当前元素并将指针向前移......