luogu题目传送门!

Car的旅行路线 

问题描述

  又到暑假了,住在城市A的Car想和朋友一起去城市B旅游。她知道每个城市都有四个飞机场,分别位于一个矩形的四个顶点上,同一个城市中两个机场之间有一 条笔直的高速铁路,第I个城市中高速铁路了的单位里程价格为Ti,任意两个不同城市的机场之间均有航线,所有航线单位里程的价格均为t。
  那么Car应如何安排到城市B的路线才能尽可能的节省花费呢?她发现这并不是一个简单的问题,于是她来向你请教。
  找出一条从城市A到B的旅游路线,出发和到达城市中的机场可以任意选取,要求总的花费最少。
输入格式
  数据的第一行有四个正整数s,t,A,B。
  S表示城市的个数,t表示飞机单位里程的价格,A,B分别为城市A,B的序号,(1<=A,B<=S)。
  接下来有S行,其中第I行均有7个正整数xi1,yi1,xi2,yi2,xi3,yi3,Ti,这当中的(xi1,yi1),(xi2,yi2),(xi3,yi3)分别是第I个城市中任意三个机场的坐标,Tl为第I个城市高速铁路单位里程的价格。
输出格式
  共有n行,每行一个数据对应测试数据,保留一位小数。
 
样例输入
 
1
1 10 1 3
1 1 1 3 3 1 30
2 5 7 4 5 2 1
8 6 8 8 11 6 3
 
样例输出
 
47.5
 
数据规模和约定
 
 对于100%的数据,  1 <= n <= 10, 1 <= S <= 100, 1 <= A, B <= S.
 
 
 

题目中让我们求出从城市A到城市B的最小花费。

不难看出,这是一道最短路的问题,将最小花费看作每条边的长度,用SPFA(Floyd)跑最短路即可。

然而,每个城市有4个机场,去每个机场的花费都不一样。因此,考虑将一个城市拆分成4个点,在存点的时候储存下城市编号(类似强联通编号),进行特判就好了。

如果是同一个城市的,边长以为铁路票价 * dis, 不同城市则为 plane * dis。

答案即为终点四个机场中最小的。

(勾股定理好评!)

#include <bits/stdc++.h>
using namespace std;
#define N 401
#define isdigit(c) ((c)>='0'&&(c)<='9')
#define min(a,b) ((a)>(b)?(b):(a))
/*比 STL 快?*/ inline int read(){
int x = , s = ;
char c = getchar();
while(!isdigit(c)){
if(c == '-') s = -;
c = getchar();
}
while(isdigit(c)){
x = (x << ) + (x << ) + (c ^ '');
c = getchar();
}
return x * s;
} struct node{//存每个机场的信息
int x, y;
int city, price;
} t[N << ];
double d[N << ];
int n, pla_pri, s, ht; inline int square(int x){
return x * x;
} inline int dis(int a, int b){//暂时先不开根号,使用起来方便一些
return square(t[a].x - t[b].x) + square(t[a].y - t[b].y); //编号机场的距离
} inline void get4(int n1,int n2,int n3){
int x4, y4;
int x1 = t[n1].x, y1 = t[n1].y, x2 = t[n2].x, y2 = t[n2].y, x3 = t[n3].x, y3 = t[n3].y;
int ab = dis(n1, n2);
int ac = dis(n1, n3);
int bc = dis(n2, n3);
if(ab == ac + bc) x4 = x1 + x2 - x3, y4 = y1 + y2 - y3;//勾股定理,求出第四个点
else if(ac == ab + bc) x4 = x1 + x3 - x2, y4 = y1 + y3 - y2;
else if(bc == ac + ab) x4 = x2 + x3 - x1, y4 = y2 + y3 - y1;
t[n3+].x = x4, t[n3+].y = y4;
/* printf("x1:%d y1: %d x2: %d y2: %d x3: %d y3: %d x4: %d y4: %d\n", x1,y1,x2,y2,x3,y3,x4,y4);
分段检查程序可以防止写完之后 Debug 两小时 */
return ;
} void init(){
int cac_city = ;
for(int i = ;i <= (n << ); i += ){
t[i].x = read(), t[i].y = read();
t[i+].x = read(), t[i+].y = read();
t[i+].x = read(), t[i+].y = read();
t[i].price = t[i+].price = t[i+].price = t[i+].price = read();//把价格记录下来
t[i].city = t[i+].city = t[i+].city = t[i+].city = ++cac_city;//城市记录下来
get4(i, i+, i+);//寻找第四个点
}
return ;
} queue <int> q;
bool vis[N << ]; void spfa(){
/*时刻不忘 n 是4倍!!不然玄学错误!!*/
for(int i = ;i <= (n << ); i++)
d[i] = 99999999.99;
for(int i = * (s - ) + ;i <= * (s - ) + ; i++){//都能作为起点,所以4个点全部推进去
vis[i] = ;
q.push(i);
d[i] = ;
}
while(!q.empty()){//SPFA
int now = q.front();q.pop();
vis[now] = ;
for(int i = ;i <= (n << ); i++){//反正全部有连边,直接 for 循环不香吗
if(i == now)continue;
double cost;
if(t[i].city == t[now].city){//如果在同一个城市
cost = t[i].price * (double)sqrt((double)dis(i, now));
}
else cost = pla_pri * (double)sqrt((double)dis(i, now));//连边的价值(距离)
if(d[i] > d[now] + cost){
d[i] = d[now] + cost;
if(!vis[i]){
vis[i] = ;
q.push(i);
}
}
}
}
return ;
} int main(){
// freopen("hh.txt", "r", stdin);
int T = read();
while(T--){
n = read(), pla_pri = read(), s = read(), ht = read();
if(s == ht){ //这里用SPFA可以不特判,但是如果用Floyd就要特判了 (初始化成了极大值)
puts("0.0");
continue;
}
init();
spfa();
double ans = ~0u >> ;
for(int i = * (ht - ) + ;i <= * (ht - ) + + ; i++)
ans = min(d[i], ans);//答案为终点四个机场里面最小的
printf("%.1lf\n", ans);
}
return ;
}

Car的旅行路线 luogu P1027 (Floyd玄学Bug有点毒瘤)的更多相关文章

  1. 洛谷P1027 Car的旅行路线

    洛谷P1027 Car的旅行路线 题目描述 又到暑假了,住在城市A的Car想和朋友一起去城市B旅游.她知道每个城市都有四个飞机场,分别位于一个矩形的四个顶点上,同一个城市中两个机场之间有一条笔直的高速 ...

  2. 洛谷 P1027 Car的旅行路线

    P1027 Car的旅行路线 题目描述 又到暑假了,住在城市A的Car想和朋友一起去城市B旅游.她知道每个城市都有四个飞机场,分别位于一个矩形的四个顶点上,同一个城市中两个机场之间有一条笔直的高速铁路 ...

  3. 洛谷 P1027 Car的旅行路线 最短路+Dijkstra算法

    目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例 输出样例 说明 思路 AC代码 总结 题面 题目链接 P1027 Car的旅行路线 题目描述 又到暑假了,住在 ...

  4. AC日记——Car的旅行路线 洛谷 P1027

    Car的旅行路线 思路: 这题不难,就是有点恶心: 而且,请认真读题目(就是题目卡死劳资): 来,上代码: #include <cmath> #include <cstdio> ...

  5. P1027 car的旅行路线

    car的旅行路线 洛谷链接 这个题关键就是 如何把每个点表示出来,其实求出四个点的坐标后,只需要把这些点连接起来,用一遍folyed求出最短路径就好了. 代码: #include<cmath&g ...

  6. NOIP2001 Car的旅行路线

    题四 Car的旅行路线(30分) 问题描述 又到暑假了,住在城市A的Car想和朋友一起去城市B旅游.她知道每个城市都有四个飞机场,分别位于一个矩形的四个顶点上,同一个城市中两个机场之间有一条笔直的高速 ...

  7. [NOIP2001提高组]CODEVS1014 Car的旅行路线(最短路)

    最短路,这个不难想,但是要为它加边就有点麻烦..还好写完就过了(虽然WA了一次,因为我调试用的输出没删了..),不然实在是觉得挺难调的.. ------------------------------ ...

  8. GDOI2015小Z的旅行路线

    GDOI2015小Z的旅行路线 题意: \(n\)个点的无根树,边上有权值. \(q\)个询问\(s\)和\(s\),问从\(s\)出发,找一条最长路(不经过重复点),保证路径上所有边边权不超过\(x ...

  9. 【Foreign】旅行路线 [倍增]

    旅行路线 Time Limit: 20 Sec  Memory Limit: 256 MB Description Input Output 仅一行一个整数表示答案. Sample Input 3 2 ...

随机推荐

  1. P4932 浏览器(统计二进制1的个数)

    P4932 浏览器 有\(n\)个数,\(x_1,x_2,\cdots,x_n\),问你有多少对\((u,v)\),使得\(x_u\operatorname{xor}x_v\)的二进制表示中有奇数个\ ...

  2. Python的内存管理和垃圾回收

    内存管理 与Python对象创建相关的结构体 #define _PyObject_HEAD_EXTRA \ struct _object *_ob_next; \ struct _object *_o ...

  3. FPGA实现-shift_ram_3x3矩阵

    shift_ram_3x3-FPGA实现 实现的方法为方法二,可以参考上一节关于中值滤波的介绍 shift_ram核介绍 https://www.cnblogs.com/ninghechuan/p/6 ...

  4. B - Planning 早训 贪心

    B - Planning 这个题目我知道要贪心,也知道怎么贪,但是写不出来,感觉自己好菜. 这个题目要用优先队列维护. 题目大意是飞机延误,不同的飞机每次延误一分钟,它的代价不同,然后问,怎么安排才能 ...

  5. zabbix 告警信息与恢复信息

    名称: Action-Email 默认接收人: 故障{TRIGGER.STATUS},服务器:{HOSTNAME1}发生: {TRIGGER.NAME}故障! 默认信息: 告警主机:{HOSTNAME ...

  6. python学习之if条件句的使用

    if循环 if 条件: 代码块 运行 if else的用法 if elseif else用法 if 条件1: elif 条件2: elif条件3: else:

  7. STM32 CubeIDE快速创建工程(图文详解)

    使用STM32CubeIDE快速创建STM32的HAL库工程. 文章目录 1 STM32CubeIDE Home 2 生成工程 3 程序下载 1 STM32CubeIDE Home 进入到官网的下载界 ...

  8. 设计模式GOF23之工厂模式01

    简单工厂模式和工厂方法模式 工厂模式核心:分工 简单工厂模式不符合OCP(Open-Closed Princinple)原则,扩展时需要更改原代码 工厂方法模式增加了类复杂度代码复杂度等,所以一般使用 ...

  9. [java作业]Fan、求直线交点、Triangle2D、选课

    public class Fan { public static void main(String[] args) { Fan fan1 = new Fan(), fan2 = new Fan(); ...

  10. USACO 3.1 Contact

    http://www.nocow.cn/index.php/Translate:USACO/contact 题目大意:给一个只含0和1的序列,统计每个子序列的重复次数,并按次数递减来输出 考虑子序列时 ...