洛谷题解 P1027 【Car的旅行路线】

萧萧江上荻花秋,做弄许多愁。

半竿落日DEBUG,两行新雁,一叶扁舟。

惜分长怕君WA,直待全A醉时休。

今宵眼底,明朝心上,后日眉头。两行泪!

出自宋代贺铸的《眼儿媚·萧萧江上荻花秋》

可谓程序20分钟,DEBUG 2小时啊!!(痛哭)

扎眼一看,这不就是最短路吗?而且才100个点

(心里想这题目也能是蓝题?)

仔细分析一下题目,只有我们建好图,一切大事搞定!

建图也不难想到我们可以把s个点扩展到s*4

这样第一个城市就是[1,4],第二个城市就是[2,5];以此类推

毒瘤的题目竟然只告诉你三个点,还有我们求出第四个点,数学蒟蒻的我瑟瑟发抖!赶紧百度了一下在直角系中对角线上点的横纵坐标和相等我们只有判断出直角就万事大吉了!!

可惜,想要A掉还要很多微操,在代码中一一展示!

#include<cstdio>
#include<cmath>
#include<cstring>
#define maxn 200002
#define I return
#define LIKE 0
#define luogu ;
using namespace std;
struct tnode{
    int x,y;
    int city;//x代表城市横坐标,y代表纵坐标,city代表它所属的城市 
}edge[maxn];
int s,t,A,B;
double f[20002];//存火车费用(注意存的是城市里的) 
double fll[1001][1001];//等会儿建图 
double fac(int x) { 
    return x*x; 
}
double distan(int x1, int y1, int x2, int y2) { 
       return 1.0*sqrt(1.0*fac(x1-x2)+1.0*fac(y1-y2)); 
}
void universe(int x1,int y1,int x2,int y2,int x3,int y3,int i) {
    int ab=fac(x1-x2)+fac(y1-y2);
    int bc=fac(x2-x3)+fac(y2-y3);
    int ac=fac(x1-x3)+fac(y1-y3);
    int x4,y4; //求第四个点的坐标,先判断出直角,然后就好办了 
    if (ab+ac==bc) x4=x2+x3-x1, y4=y2+y3-y1;
    if (ab+bc==ac) x4=x1+x3-x2, y4=y1+y3-y2;
    if (bc+ac==ab) x4=x1+x2-x3, y4=y1+y2-y3;
    edge[i+3].x=x4;//存入第四个点里 
    edge[i+3].y=y4;
}
double mon(tnode x1,tnode x2){//读入试先已经准备好微操,所以调用很简单 
    double juli=distan(x1.x,x1.y,x2.x,x2.y);
    if(x1.city==x2.city) return 1.0*juli*f[x1.city];
    return 1.0*juli*t;
}
int main() {
    int n;
    double ans=29292992.0;
    scanf("%d",&n);
    while(n--){
        memset(fll,98,sizeof(fll));//注意一定是98,(好像没有也没问题) 
        memset(edge,0,sizeof(edge));
        scanf("%d%d%d%d",&s,&t,&A,&B);
        for (int i=1; i<=4*s; i=i+4) {//读入微操,扩展点后,city存储原来的点 
            scanf("%d%d%d%d%d%d%lf",&edge[i].x,&edge[i].y,&edge[i+1].x,&edge[i+1].y,&edge[i+2].x,&edge[i+2].y,&f[i/4+1]);
            edge[i].city=edge[i+1].city=edge[i+2].city=edge[i+3].city=i/4+1;
            universe(edge[i].x,edge[i].y,edge[i+1].x,edge[i+1].y,edge[i+2].x,edge[i+2].y,i);
        }
        for(int i=1;i<=s*4;i++)//建图初始化 
           for(int j=1;j<=s*4;j++)
                fll[i][j]=1.0*mon(edge[i],edge[j]);
        for(int k=1;k<=s*4;k++)//标准弗洛伊德 
           for(int i=1;i<=s*4;i++)
              if(i!=k)
              for(int j=1;j<=s*4;j++)
                 if(i!=j&&j!=k){
                    if(fll[i][j]>fll[i][k]+fll[k][j])
                       fll[i][j]=fll[i][k]+fll[k][j];
                }
                //在起点和终点四个点中寻找最小值 
        for(int i=A*4-3;i<=A*4;i++)
           for(int j=B*4-3;j<=B*4;j++){
              if(fll[i][j]<ans) ans=fll[i][j];
           }
        printf("%.1lf",ans);
    }
    I LIKE luogu
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值