1 条题解

  • 0
    @ 2025-8-24 21:28:56

    自动搬运

    查看原文

    来自洛谷,原作者为

    avatar ycyaw
    无他,唯勤尔

    搬运于2025-08-24 21:28:55,当前版本为作者最后更新于2019-03-03 10:24:08,作者可能在搬运后再次修改,您可在原文处查看最新版

    自动搬运只会搬运当前题目点赞数最高的题解,您可前往洛谷题解查看更多

    以下是正文


    模拟赛居然考了这道题,前一天刚看过,结果看了舍不得(不会)做,结果只骗到30pt

    讲课人:很容易想到最短路+dpdp(我靠一点都不容易)

    模拟赛后分析,才知道是处理出第i天到第j天都走同一条最短路的花费为co[i][j]co[i][j]

    然后进行dpdpdp[i]dp[i]表示前i天的最小花费

    转移方程很好想:dp[i]=min(dp[j]+co[j+1][i](ij)+k)dp[i]=min(dp[j]+co[j+1][i]*(i-j)+k),预处理要赋值为co[1][i]ico[1][i]*i

    dpdp方程的意思,即在第j+1j+1天改变路线,第j+1j+1天~第ii天都走同一条路线

    那么如何处理co[i][j]co[i][j]?

    很简单,对于每一个(i,j)(i,j),先把iijj天之间封闭的码头全部设为不可走,跑一遍最短路即可,初值为无穷

    数据辣么小,跑几遍以及跑什么都没关系嘤嘤嘤

    那我们就十分愉♂快的解决了此题~~~

    愉♂快的提交了然后居然只有90pt

    原谅我无耻的打开题解

    啊啊啊原来要开longlong longlong(明明数据辣么小)

    献上代码:

    #include<bits/stdc++.h>
    #define soo (1e8)
    #define ll long long
    using namespace std;
    int d,cnt,head[25],dis[25],vis[25],cant_vis[25];
    ll co[105][105],dp[105];
    int n,m,k,ee,cl[25][105];
    struct Edge{
    	int v,nx,s;
    }e[10005];
    inline int read(){
        int ret=0,ff=1;char ch=getchar();
        while(!isdigit(ch)){if(ch=='-') ff=-ff;ch=getchar();}
        while(isdigit(ch)){ret=(ret<<3)+(ret<<1)+ch-'0';ch=getchar();}
        return ret*ff;
    }
    void add(int x,int y,int z){
    	e[++cnt].v=y;
    	e[cnt].s=z;
    	e[cnt].nx=head[x];
    	head[x]=cnt;
    }
    void spfa(){//爱跑什么跑什么
    	for(int i=1;i<=m;i++) dis[i]=soo,vis[i]=0;
    	queue<int> q;
    	dis[1]=0;
    	q.push(1);
    	while(!q.empty()){
    		int x=q.front();
    		q.pop();
    		vis[x]=0;
    		for(int i=head[x];i;i=e[i].nx){
    			int v=e[i].v;
    			if(cant_vis[v]) continue;
    			if(dis[v]>dis[x]+e[i].s){
    				dis[v]=dis[x]+e[i].s;
    				if(!vis[v]){
    					vis[v]=1;
    					q.push(v);
    				}
    			}
    		}
    	}
    }
    signed main(){
    	n=read(),m=read(),k=read(),ee=read();
    	for(int i=1;i<=ee;i++){
    		int x=read(),y=read(),z=read();
    		add(x,y,z);
    		add(y,x,z);
    	}
    	d=read();
    	for(int i=1;i<=d;i++){
    		int t=read(),x=read(),y=read();
    		for(int j=x;j<=y;j++) cl[t][j]=1;
    	}
    	//cl[i][j]表示第i个码头在第j天不能走
    	for(int i=1;i<=n;i++)
    		for(int j=1;j<=n;j++){
    			memset(cant_vis,0,sizeof(cant_vis));
    			for(int r=i;r<=j;r++)
    				for(int l=1;l<=m;l++)
    					if(cl[l][r]) cant_vis[l]=1;
    			spfa();
    			co[i][j]=dis[m];
    		}
    	memset(dp,0x7f,sizeof(dp));
    	for(int i=1;i<=n;i++){
    		dp[i]=(ll)co[1][i]*i;
    		for(int j=i-1;j>=0;j--)
    			dp[i]=min(dp[i],dp[j]+co[j+1][i]*(i-j)+k);
    	}
    	printf("%lld",dp[n]);
        return 0;
    }
    
    • 1

    信息

    ID
    745
    时间
    1000ms
    内存
    125MiB
    难度
    5
    标签
    递交数
    0
    已通过
    0
    上传者