【u026】房间最短路问题
描述
在一个长宽均为10,入口出口分别为(0,5)、(10,5)的房间里,有几堵墙,每堵墙上有两个缺口,求入口到出口的最短路经。

在一个长宽均为10,入口出口分别为(0,5)、(10,5)的房间里,有几堵墙,每堵墙上有两个缺口,求入口到出口的最短路经。

格式
输入格式
第一排为n(n<=20),墙的数目。
接下来n排,每排5个实数x,a1,b1,a2,b2。
x表示墙的横坐标(所有墙都是竖直的),a1-b1和a2-b2之间为空缺。
a1、b1、a2、b2保持递增,x1-xn也是递增的。
第一排为n(n<=20),墙的数目。
接下来n排,每排5个实数x,a1,b1,a2,b2。
x表示墙的横坐标(所有墙都是竖直的),a1-b1和a2-b2之间为空缺。
a1、b1、a2、b2保持递增,x1-xn也是递增的。
输出格式
输出最短距离,保留2位小数。
输出最短距离,保留2位小数。
样例1
样例输入1[复制]
2
4 2 7 8 9
7 3 4.5 6 7
2
4 2 7 8 9
7 3 4.5 6 7
样例输出1[复制]
10.06
10.06
【题解】
这是一道最短路问题。
这样做。
给每个墙上的两个“裂缝”的端点标号。则每个墙有4个端点(起点和终点除外)。
则第i个墙的第j个端点它的编号为(i - 1) + j;
然后起点的编号为0,终点的编号为4*n + 1;
最后输出0到4*n+1的最短距离;
然后建图的过程如下
枚举第ii个墙和第jj个墙
然后第ii个墙枚举的是第i个端点,第jj个墙枚举的是第j个端点。
设两个端点的坐标为(x1,y1),(x2,y2);
然后我们就要看看ii+1..jj-1这些墙里面是否有会挡住这两点所构成的直线的墙。
如果有这样的墙,则这两个点不能连线,否则在它们两个点之间建立一条边。
这条边的权值就是两点之间的距离。
可以用两点之间的距离公式求得。
sqrt((x1-x2)*(x1-x2)+(y1-y2)*(y1-y2));
然后要判断会不会有墙挡住这条直线则需要用到直线方程的两点式;
已知两个点(x1,y1),(x2,y2);
则它们所构成的直线方程为
(x-x1)/(x2-x1)=(y-y1)/(y2-y1)
然后两边消一下,换一下。
得到y=(y2-y1)*(x-x1)/(x2-x1) + y1
然后假设我们枚举到ii+1..jj-1中的墙k。
则x=墙k的横坐标
代入之后得到y0;
看看这个y0是不是在k墙的两个窟窿中。如果不是则会撞墙返回false跳过这条边。
如果ii+1..jj-1这些墙都不会阻碍这条直线。则可以在这两个点之间建一条边。
然后还要单独处理起点到其他墙的、其他墙到终点的以及起点到终点的这3种情况。
最后选择用floyd来求最短路即可(简单!)
【代码】
#include <cstdio>
#include <cmath>
#include <cstring> struct point
{
double x;
double height[5];
}; int n, t[200][200] = { 0 };
double w[200][200];
point a[200]; int main()
{
for (int i = 0; i <= 199; i++)
for (int j = 0; j <= 199; j++) //这是做floyd算法用的数组。一开始赋值为无穷大。
w[i][j] = 2100000000 / 3;
scanf("%d", &n);
for (int i = 1; i <= n; i++) //输入n个墙的信息。
{
double x, a1, b1, a2, b2;
scanf("%lf%lf%lf%lf%lf", &x, &a1, &b1, &a2, &b2);
a[i].x = x, a[i].height[1] = a1, a[i].height[2] = b1, a[i].height[3] = a2, a[i].height[4] = b2;
} for (int ii = 2; ii <= n; ii++) //第ii列
{
for (int i = 1;i <= 4;i++)
for (int jj = 1; jj <= ii - 1; jj++) //第jj列
{
for (int j = 1; j <= 4; j++) //这层要先求出i,j两个点构成的直线方程。
{
double x1, y1, x2, y2;
x1 = a[ii].x;y1 = a[ii].height[i]; //这是所枚举的两个点的坐标。
x2 = a[jj].x;y2 = a[jj].height[j];
bool judge = true;
for (int kk = jj + 1; kk <= ii - 1; kk++)//它们中间的墙。
{
double x = a[kk].x;
double y0 = ((y2 - y1)*(x - x1) / (x2 - x1)) + y1; //这是这两列的相应的点的直线方程,把第kk列的点的信息代入。
if ((y0 >= a[kk].height[1] && y0 <= a[kk].height[2]) || (y0 >= a[kk].height[3] && y0 <= a[kk].height[4]))
judge = true; //如果在两个窟窿里面则是可以添加的边
else//否则只要有一个不行就判错
{
judge = false;
break;//结束循环了。
}
}
if (judge) //如果可以添加这条边
{
int fr = (jj - 1) * 4 + j, to = (ii - 1) * 4 + i;//这是两个点的编号。
t[fr][0]++; //起点的出度增加。
t[fr][t[fr][0]] = to;
w[fr][to] = sqrt((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));//权值为两点之间的距离。
}
}
}
} //其实下面这两段直接复制上面的同时枚举两个点的片段即可。 for (int jj = 1; jj <= n; jj++) //这是起点到其他墙的点。(不包括终点)
{
for (int j = 1; j <= 4; j++)
{
double x1, y1, x2, y2;
x1 = 0; y1 = 5; //这下其中的一个点变成是固定的了。
x2 = a[jj].x; y2 = a[jj].height[j];//现在只需要枚举一个点了。
bool judge = true;
for (int kk = 1; kk <= jj- 1; kk++)//然后枚举它们中间的墙,如果它们中间没有墙,则说明一定可以添加,所以初值是true;
{
double x = a[kk].x;
double y0 = ((y2 - y1)*(x - x1) / (x2 - x1)) + y1;//用同样的办法判断是否可以添加这条边。
if ((y0 >= a[kk].height[1] && y0 <= a[kk].height[2]) || (y0 >= a[kk].height[3] && y0 <= a[kk].height[4]))
judge = true;
else
{
judge = false;
break;
}
}
if (judge)
{
int fr = 0, to = (jj - 1) * 4 + j;//如果可以就添加这条边。
t[fr][0]++;
t[fr][t[fr][0]] = to;
w[fr][to] = sqrt((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));
}
}
} for (int jj = 1; jj <= n; jj++) //其他墙上的点(不包括起点)到终点(4*n+1)
{
for (int j = 1; j <= 4; j++)
{
double x1, y1, x2, y2;
x1 = a[jj].x; y1 = a[jj].height[j];
x2 = 10; y2 = 5;//这下其中一个固定的点变成了终点
bool judge = true;
for (int kk = jj+1; kk <= n; kk++)
{
double x = a[kk].x;
double y0 = ((y2 - y1)*(x - x1) / (x2 - x1)) + y1;
if ((y0 >= a[kk].height[1] && y0 <= a[kk].height[2]) || (y0 >= a[kk].height[3] && y0 <= a[kk].height[4]))
judge = true;
else
{
judge = false;
break;
}
}
if (judge) //如果可以则添加这条边。
{
int fr = (jj - 1) * 4 + j, to = 4*n+1;
t[fr][0]++;
t[fr][t[fr][0]] = to;
w[fr][to] = sqrt((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));
}
}
} //起点到终点,因为可能可以直接从起点画一条直线就到终点了。所以一定要判断这个。不然会WA!
double x1, y1, x2, y2;
x1 = 0; y1 = 5;
x2 = 10; y2 = 5; //两个点都是固定的了。
bool judge = true;
for (int kk = 1; kk <= n; kk++)//看看起点和终点直接的墙会不会阻碍它们俩画一条直线。
{
double x = a[kk].x;
double y0 = ((y2 - y1)*(x - x1) / (x2 - x1)) + y1;
if ((y0 >= a[kk].height[1] && y0 <= a[kk].height[2]) || (y0 >= a[kk].height[3] && y0 <= a[kk].height[4]))
judge = true;
else
{
judge = false;
break;
}
}
if (judge)//如果不会被阻碍,就画一条边。
{
int fr = 0, to = 4 * n + 1;
t[fr][0]++;
t[fr][t[fr][0]] = to;
w[fr][to] = sqrt((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));
} n = 4 * n + 1;//让n变成节点的编号。
for (int k = 0; k <= n; k++)//从0-n做floyd;
for (int i = 0; i <= n; i++)
for (int j = 0; j <= n; j++)
if (w[i][j] > w[i][k] + w[k][j])
w[i][j] = w[i][k] + w[k][j];
printf("%.2lf", w[0][n]); //保留两位小数输出即可。
return 0;
}
【u026】房间最短路问题的更多相关文章
- P1354 房间最短路问题
传送门 可以发现,最短路一定要经过墙壁的断点. 那么把房间看作一个有向图,墙壁的断点为节点,求从起点到终点的最短路. 这道题的难点在于建图.枚举所有的断点,若可以走则加入这条边. 判断两点是否连通,即 ...
- luogu 1354 房间最短路问题 线段与直线相交 最短路
题目链接 题目描述 在一个长宽均为10,入口出口分别为(0,5).(10,5)的房间里,有几堵墙,每堵墙上有两个缺口,求入口到出口的最短路经. 输入输出格式 输入格式: 第一排为n(n<=20) ...
- [Luogu P1354]房间最短路问题
这是一道紫题,然而实际上我觉得也就蓝题难度甚至不到. and,这道题就是一道数学题,代码模拟计算过程. 求最短路嘛,肯定要考虑建图,只需要把中间的墙上每个口的边缘处的点作为图中的点就行.至于为什么,显 ...
- luoguP1354房间最短路问题
判断两点间连通性,建图跑floyed #include<bits/stdc++.h> using namespace std; ; struct node { ],x; }q[N];dou ...
- luogu P1354 房间最短路问题 计算几何_Floyd_线段交
第一次写计算几何,还是很开心的吧(虽然题目好水qaq) 暴力枚举端点,暴力连边即可 用线段交判一下是否可行. Code: #include <cstdio> #include <al ...
- ACM/ICPC 之 DP解有规律的最短路问题(POJ3377)
//POJ3377 //DP解法-解有规律的最短路问题 //Time:1157Ms Memory:12440K #include<iostream> #include<cstring ...
- ACM 房间安排
房间安排 时间限制:3000 ms | 内存限制:65535 KB 难度:2 描述 2010年上海世界博览会(Expo2010),是第41届世界博览会.于2010年5月1日至10月31日期间, ...
- 房间安排-nyoj168
描述 2010年上海世界博览会(Expo2010),是第41届世界博览会.于2010年5月1日至10月31日期间,在中国上海市举行.本次世博会也是由中国举办的首届世界博览会.上海世博会以“城市,让生活 ...
- 房间声学原理与Schroeder混响算法实现
一.混响时间的计算与预测 所谓混响就是声音的直达声与反射声很紧凑的重合在一起时人耳所听到的声音,这个效果在语音的后期处理时特别有用.能产生混响最常见的场景就是房间内,尤其是空旷的房间中. 混响有直达声 ...
随机推荐
- 趣闻|Python之禅(The Zen of Python)
在Python解释器中输入“import this”会发生什么?如果你不知道这个彩蛋,推荐继续阅读这篇文章. 2001年秋,Foretec(一家会议组织公司)正在准备召开第十届Internationa ...
- 【习题 7-4 UVA-818】Cutting Chains
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 二进制枚举要解开哪些环. 把所有和它相关的边都删掉. 对于剩下的联通分量. 看看是不是每一个联通分量都是一条链 ->每个点的度 ...
- Java 函数的参数说
java函数参数传递的到底是值还是引用对确实容易让人迷糊.而很多时候因为对这个问题的模糊甚至造成一些错误.最常见的说法是基本类型传的是值,对象传的引用.对于基本类型,大家都达成共识,没有什么可以争论的 ...
- 洛谷 P1601 A+B Problem(高精)
P1601 A+B Problem(高精) 题目背景 无 题目描述 高精度加法,x相当于a+b problem,[b][color=red]不用考虑负数[/color][/b] 输入输出格式 输入格式 ...
- Android圆环控件
Android圆环控件 近期在做一个功能.界面效果要求例如以下: 看到这个界面,我首先想到了曾经在做phone模块的时候,我们定制的来电界面InCallTouchUi,界面效果是相似的. 来电控件使用 ...
- nginx源代码分析--事件模块 & 琐碎
通过HUP信息使得NGINX实现又一次读取配置文件,使用USR2信号使得NGINX实现平滑升级. 在nginx中有模块这么一说,对外全部的模块都是ngx_module_t类型,这个结构体作为全部模块的 ...
- iOS_01_C语言简介
1.先学C语言的原因 * oc基于C. * oc 跟 C的思想和语法很多地方不太一样,而且OC能和C混用. * C 是所有编程语言中的经典,很多高级语言都是从C语言中衍生出来的,比如 C++,C#.O ...
- springboot集成shiro 实现权限控制(转)
shiro apache shiro 是一个轻量级的身份验证与授权框架,与spring security 相比较,简单易用,灵活性高,springboot本身是提供了对security的支持,毕竟是自 ...
- TCP的可靠传输机制(简单好理解:分段与流,滑窗,连接,流量控制,重新发送,堵塞控制)
TCP的几大模块:分段与流,滑窗,连接,流量控制,重新发送,堵塞控制. 1.checksum:在发送TCP报文的时候,里面的信息可能会因为环境的问题,发送变化,这时,接收信号的时候就需要通过check ...
- vue项目实现导出数据到excel
实现导出功能分两种,一种是客户端拿到数据做导出,第二种是服务器端处理好,返回一个数据流实现导出 第一种网上很容易找到,也很好用,本文要说的是第二种. fetchExport({ id: this.so ...