首页 > 其他分享 >Codeforces 1715E - Long Way Home

Codeforces 1715E - Long Way Home

时间:2022-08-21 14:33:47浏览次数:106  
标签:ch dist int ll Codeforces Long dijkstra Way qu

又是废掉的一个div2啊

第一次在学校熬夜打cf,开心还看到了自己最喜欢的斜率优化ohhh

链接 :E - Long Way Home

看到那个平方就可以靠感觉认为是斜率优化了....

感觉似不似有点想法??k只有20...

可以试着去考虑最后一步用飞机,然后跑dijkstra求出走普通路径的。

其实就这样了...


考虑k=1的情况:

在最后坐飞机,前面都普通路径行走

则:\(d_{new}[i]=min(d_{old}[j]+(i-j)^2)\)

\(d_{new}\)为最后的答案,就是在最后坐飞机,\(d_{old}\)就是前面已经处理的走普通路径

然后跑一遍dijkstra,看看是否会更新(即假设存在A->C的最短路径为:A->B最后坐飞机,B->C走普通路径)


处理k>1的情况也是一样的,做k次dp,每一次在dp后跑最短路看能否用普通路径更新。

注意的是在一开始要跑一遍最短路,代表坐0次飞机的最短路径。

然后就是对于这个dp式子的处理了:

观察发现可以把式子转换为:\(d_{old}[j]+j^2=2ij+d_{new}[i]-i^2\)

斜率为\(2i\),X为\(j\),Y为\(d_{old}[j]+j^2\)。斜率和X都单调递增,所以可以画图发现决策点是下凸(为什么不转换式子要画图?因为j有特殊的点)

这个j特殊的点就在于没有范围的限制,也就是范围是(1<=j<=n)

所以就不可以在循环中加入单调队列

可以先一遍循环把所有决策点存入单调队列,然后再一遍循环更新答案

就完了....

对了!还有一个点:因为m可能小于n,图可能不连通,也就是说从只1号点跑dijkstra就不一定可以遍历完,所以就需要在dp时如果更新了\(d_{new}\)就要把它放入优先队列中,以保证每个点都可以更新到。

Over.

一定要在计算Y的时候把\(j^2\)开成long long啊!!!麻了,改了一个小时(ku)

总的复杂度\(O(k(mlogn+n))\)

代码:

#include<bits/stdc++.h>
#define ll long long 
using namespace std;
inline int read(){
    register int x = 0, t = 1;
    register char ch=getchar();
    while(ch<'0'||ch>'9'){
        if(ch=='-')
            t=-1;
        ch=getchar();
    }
    while(ch>='0'&&ch<='9'){
        x=(x<<1)+(x<<3)+(ch^48);
        ch=getchar();
    }
    return x*t;
}
struct node{
	int to,nxt,w;
}e[200010];
int head[100010],cnt,vis[100010];
void add(int u,int v,int w){
	e[++cnt].to=v;
	e[cnt].nxt=head[u];
	e[cnt].w=w;
	head[u]=cnt;
}
ll dist[100010],dis[110000]/*old*/;
struct h{
	int id;
	ll dis;
	bool operator<(const h &other)const{
		return dis>other.dis;
	}
};
priority_queue<h>qu;
void dijkstra(){//更新以普通边结尾 
	memset(vis,0,sizeof(vis));
	qu.push((h){1,0});
	while(!qu.empty()){
		int p=qu.top().id;
		qu.pop();
		if(vis[p])continue;
		vis[p]=1;
		for(int i=head[p];i;i=e[i].nxt){
			int v=e[i].to;
			if(dist[v]>dist[p]+e[i].w){
				dist[v]=dist[p]+e[i].w;
				qu.push((h){v,dist[v]});
			}
		}
	}
}
int q[100010],hd,tl;
ll Y(int i){return dis[i]+(ll)i*i;/*爆了!!*/}
ll X(int i){return i;}
ll K(int i){return 2*i;}
double slope(int x,int y){//x>y
	if(X(x)==X(y)){
		if(Y(x)<Y(y))return -1e18;
		else return 1e18;
	}
	return (double)(Y(x)-Y(y))/(X(x)-X(y));
}
int main(){
	int n=read(),m=read(),k=read();
	for(int i=1;i<=m;i++){
		int x=read(),y=read(),w=read();
		add(x,y,w),add(y,x,w);
	}
	for(int i=2;i<=n;i++)dist[i]=1e18;
	dijkstra();
	while(k--){
		for(int i=1;i<=n;i++)dis[i]=dist[i];
		hd=tl=0;
		for(int i=1;i<=n;i++){
			//下凸壳
			while(hd+1<tl&&slope(i,q[tl-1])<=slope(q[tl-1],q[tl-2]))tl--;
			q[tl++]=i;
		}
		for(int i=1;i<=n;i++){
			while(hd+1<tl&&slope(q[hd+1],q[hd])<=K(i))hd++;
			if(hd<tl){
				int j=q[hd];
				if(dist[i]>dis[j]+(ll)(j-i)*(j-i)){
					dist[i]=dis[j]+(ll)(j-i)*(j-i);
					qu.push((h){i,dist[i]});
				}
			}
		}
		dijkstra();
	}
	for(int i=1;i<=n;i++)printf("%lld ",dist[i]);
	return 0;
}

标签:ch,dist,int,ll,Codeforces,Long,dijkstra,Way,qu
From: https://www.cnblogs.com/lefy959/p/16609952.html

相关文章

  • Codeforces Round #816 (Div. 2)
    题面A.Crossmarket不妨设\(n\gem\),则答案为\(n+2(m-1)\)。特别地,\(n=1,m=1\)时答案为\(0\),注意特判。ViewCode#include<stdio.h>#include<algorithm>int......
  • [Codeforces Round #816 (Div. 2)] D. 2+ doors
    这次Div.2比之前我打的有些要难啊,前三道题就耗了好多时间,D题干脆摆烂了。。。还是太逊了对于一个\(x\),有\(x|y_i=z_i\),那么我们设\(num[x]=z_1\)&\(z_2\)&\(z_3\)..........
  • 【长期】板刷Codeforces 1500-1700 的构造题
    【长期】板刷Codeforces1500-1700的构造题https://codeforces.com/problemset/page/1?tags=constructive+algorithms%2C1500-1700&order=BY_RATING_ASC每天三道,记录一......
  • CF1715E Long Way Home
    套路题。先不考虑额外的边跑一次最短路。然后考虑一下额外的边,单独拿出来转移一次。式子为\(dis_u=min\{olddis_v+(u-v)^2,1\leqv\leqn\}\)。简单的,把凸包建出来,二......
  • [Google] LeetCode 1048 Longest String Chain
    YouaregivenanarrayofwordswhereeachwordconsistsoflowercaseEnglishletters.\(word_A\)isapredecessorof\(word_B\)ifandonlyifwecaninserte......
  • Codeforces 1720 D, E
    D1设\(dp(i)\)表示考虑前i个数的最长子序列。枚举\(j\),从\(dp(j)+1\)转移到\(dp(i)\),转移条件就是题中给的那个不等式。发现\(i-j\)不能超过\(300\),暴力枚举即可。时间......
  • [题解] Codeforces 1720 E Misha and Paintings 结论
    算是诈骗题?令一开始就存在的颜色数为cnt。k>=cnt的情况,显然每次找一个出现不止一次的颜色,然后把这个颜色的恰好一个方块替换成一种没有出现过的颜色就可以了,\(k-cnt\)次解......
  • codeforces963D. Frequency of String【哈希】
    我的腿让我停下,可是我的心却不许我这么做今天又是为了明知多半不可能的事情奔波一早,一天里,出了很多丑,犯了很多错,见了很多人,有了很多意想不到的收获,我选择了我的生存方式......
  • Codeforces Round #815 (Div. 2) 题解
    CF1720A.BurenkaPlayswithFractions给出两个分数$\dfrac{a}{b}$和\(\dfrac{c}{d}\),你每次操作能够选择其中一个分数的分子或分母,将其乘上任意一个整数(当然不能......
  • Codeforces 杂题集
    本文主要把近期\(CF-Div.2\)的\(A,B,C,D\)题进行做Round815A题意给你两个分数\(\frac{a}{b},\frac{c}{d}\),问你最少几次使两个分数相等。Solution首先考虑,最......