算法:poj1066 宝藏猎人问题。
package practice;
import java.util.Scanner;
public class TreasureHunt {
public static void main(String[] args) {
Scanner cin = new Scanner(System.in);
int[][] walls;
float x, y;
int doors = Integer.MAX_VALUE, temp1, temp2;
int n = cin.nextInt();
walls = new int[n][4];
for (int j = 0; j < n; j++) {
walls[j][0] = cin.nextInt();
walls[j][1] = cin.nextInt();
walls[j][2] = cin.nextInt();
walls[j][3] = cin.nextInt();
}
x = cin.nextFloat();
y = cin.nextFloat();
for (int j = 0; j < n; j++) {
temp1 = find(walls[j][0], walls[j][1], x, y, walls, j);
temp2 = find(walls[j][2], walls[j][3], x, y, walls, j);
doors = temp1 < temp2 ? (temp1 < doors ? temp1 : doors)
: (temp2 < doors ? temp2 : doors);
}
if (n == 0)
doors = 0;
doors++;
System.out.println("Number of doors = " + doors);
}
private static int find(int x1, int y2, float x, float y, int[][] walls,
int j) {
int count = 0;
for (int i = 0, len = walls.length; i < len; i++) {
if (i == j)
continue;
if (isIntersect(x1, y2, x, y, walls[i]))
count++;
}
return count;
}
/**
*
* 跨立实验
*/
private static boolean isIntersect(int startX, int startY, float endX,
float endY, int[] wall) {
if ((Math.max(startX, endX) >= Math.min(wall[0], wall[2]))
&& (Math.max(wall[0], wall[2]) >= Math.min(startX, endX))
&& (Math.max(startY, endY) >= Math.min(wall[1], wall[3]))
&& (Math.max(wall[1], wall[3]) >= Math.min(startY, endY))
&& (multiply(wall[0], wall[1], endX, endY, startX, startY)
* multiply(endX, endY, wall[2], wall[3], startX, startY) > 0)
&& (multiply(startX, startY, wall[2], wall[3], wall[0], wall[1])
* multiply(wall[2], wall[3], endX, endY, wall[0],
wall[1]) > 0))
return true;
else
return false;
}
private static double multiply(float x1, float y1, float x2, float y2,
float x3, float y3) {
return ((x1 - x3) * (y2 - y3) - (x2 - x3) * (y1 - y3));
}
}
点我展开代码
该问题是:
一伙寻宝人探测到了埃及金字塔底层的宝藏,但是宝藏被n堵墙围着,如果要爆破,只能在每堵墙的中点开门。现在问题来了,对于随机给定的n堵墙,算出最少需要的开门数。
输入数据如下所示,第一行的整数n表示墙数,后面的n行表示墙体两端坐标,以金字塔边缘左下角为(0,0)右上角为(100,100),每行的四个数据分别为(x1,y1,x2,y2)
最后一行的数据表示宝藏点P的位置。
7
20 0 37 100
40 0 76 100
85 0 0 75
100 90 0 90
0 71 100 61
0 14 100 38
100 47 47 100
54.5 55.4
算法的思路很简单(但是实现有点儿问题,比较耗时,正在改进中):
用给出的所有点对宝藏点P做线段,找出交点最少的那一组,既是最小开门数。
以上结论需要证明多个推论,证明过程如下:
1、对于任意P,如果和P只间隔一堵墙,那么,P只需要开一扇门
证明:无需证明显而易见。
结论:只要证明墙数,就可以证明门数。
2、边缘上的任意一中点P0对P做线段,交点数等于P到Pi间隔的最小墙数
证明:
连接P0-P,交点设为N。
设,存在一条开门路径,连接P0到P,中间穿过了M堵墙。
可知,P0-P与开门路径之间是封闭的多边形(路径是直线且与P0-P重合不讨论)。
有定理一:最小完整路径不会两次闯过同一堵墙。(如果有两次闯过,则至少有三个交点A\B\C,连接A-C,则路径更少,易证)
有定理二:两条子路径之间有且仅有一堵墙(易证,穿过墙只能抵达对侧,不能抵达同侧)
由定理二可知,墙一定会经过多边形内部,由定理一可知,墙不会再次经过路径,则,墙一定会经过P0-P
即是M<=N
又有公理一:两条线段最多只有一个交点
经过了P0-P的线段一定和封闭多边形相交。(由于线段端点落在金字塔边缘,易证)
即N<=M
所以M=N。
3、任意现存线段端点与P连线,交点数等于 邻近两个中点分别与P连线 的交点数 中的小值
证明:
设有两线段L1,L2分别交金字塔外墙为P1,P2,且P1、P2之间再无其他现存线段的交点
设有一点Pn处于P1,P2之间
对Pn-P做连线,设有一现存直线Lx,与L1相交,但不与Pn-P相交,则此时满足Pn对P的交点数小于P1。
此时,Lx与P1-P2外墙的交点,必然落在Pn和P1之间(此点易证),与P1、P2之间无现存线段交点违背。
可证,Pn-P的交点数一定大于等于P1-P。
同理可证Pn和P2关系。
当等于的时候,Pn与P1等价。
当大于的时候,推论如下:
设有一点Px,Px在Pn-P1延长线上,为P1-P0(P0为另一相交点或者金字塔顶点,相交点等于顶点情况另行讨论)线段上的任意某点。
可知,Px和P1之间再无任意交点。
那么,沿用先前的推论,P1-P交点数大于等于Px-P0交点数。
当大于的时候,必有一线段与P1-P相交,不与Px-P0相交,此线段不能落在P1-Px段,只能落在P1-Pn,且不为P1本身(相交非重合),与P1-Pn之间无交点违背。
所以,必然是等于关系(同理可得L1即是Pn-P多出的那一相交线)
相交点等于顶点情况,如果有P点同侧相交线,相交线必然与金字塔边缘有两个交点,无论落点如何,都能多次引用前半部分推论来同理证得。
证毕。
结论:
只需要求线段端点与P的交点数,即可得到最小值。
算法:poj1066 宝藏猎人问题。的更多相关文章
- PKUWC 2017 Day 2 简要题解
*注意:题面请移步至loj查看. 从这里开始 Problem A 随机算法 Problem B 猎人杀 Problem C 随机游走 怎么PKU和THU都编了一些假算法,然后求正确率[汗]. 之前听说 ...
- PIXI 宝物猎人(7)
介绍 ,本实例来自官网 代码结构 打开 treasureHunter.html 文件,你将会看到所有的代码都在一个大的文件里.下面是一个关于如何组织所有代码的概览: //Setup Pixi and ...
- 2000G电脑大型单机游戏合集
激活码 游戏名称(ctrl+F查找) 下载链接005875 艾迪芬奇的记忆 游戏下载链接http://pan.baidu.com/s/1t2PYRAj546_1AcOB-khJZg554158 暗影: ...
- BFS/DFS算法介绍与实现(转)
广度优先搜索(Breadth-First-Search)和深度优先搜索(Deep-First-Search)是搜索策略中最经常用到的两种方法,特别常用于图的搜索.其中有很多的算法都用到了这两种思想,比 ...
- acm常见算法及例题
转自:http://blog.csdn.net/hengjie2009/article/details/7540135 acm常见算法及例题 初期:一.基本算法: (1)枚举. (poj17 ...
- ACM算法总结及刷题参考
参考:http://bbs.byr.cn/#!article/ACM_ICPC/11777 OJ上的一些水题(可用来练手和增加自信)(poj3299,poj2159,poj2739,poj1083,p ...
- poj 算法 分类
转载请注明出处:優YoU http://blog.csdn.net/lyy289065406/article/details/6642573 最近AC题:2528 更新时间:2011.09.22 ...
- ACM常用算法及练习(1)
ACM常用算法及练习 第一阶段:练经典常用算法,下面的每个算法给我打上十到二十遍,同时自己精简代码,因为太常用,所以要练到写时不用想,10-15分钟内打完,甚至关掉显示器都可以把程序打出来. 1.最短 ...
- Tarjan算法求有向图的强连通分量
算法描述 tarjan算法思想:从一个点开始,进行深度优先遍历,同时记录到达该点的时间(dfn记录到达i点的时间),和该点能直接或间接到达的点中的最早的时间(low[i]记录这个值,其中low的初始值 ...
随机推荐
- Security.website-that-focus-on-mobile-app-security
Mobile App Security 1. DATA THEOREM LAB https://datatheorem.github.io/ Data Theorem's technical blog ...
- Oracle查询某段日期内某个时间段的数据
示例: 查询最近一周12:30分到13:00这段时间内的订单量: SELECT * FROM T_ORDER O WHERE O.CREATEDATETIME BETWEEN SYSDATE AND ...
- 关于python的requests库抓取源文件中文乱码的情况
import requests html=requests.get('http://www.12306.cn/mormhweb/') print html.text 刚开始是这么写的,运行时一直弹出错 ...
- whoami与who am i
whoami显示的是当前“操作用户”的用户名,而who am i显示的是“登录用户”的用户名. 若你使用root用户登录,中间su abc切换,whoami的结果是abc,who am i 的结果是r ...
- (待续)C#语言中的动态数组(ArrayList)模拟常用页面置换算法(FIFO、LRU、Optimal)
目录 00 简介 01 算法概述 02 公用方法与变量解释 03 先进先出置换算法(FIFO) 04 最近最久未使用(LRU)算法 05 最佳置换算法(OPT) 00 简介 页面置换算法主要是记录内存 ...
- 我利用网上特效开发的Jquery插件
我利用网上特效开发的Jquery插件 代码如下 (function($){ $.fn.Dialogx = function(options) { var defaults={ Width:" ...
- Access to the path '' is denied 解决
环境:iis6 使用silverlight做的上传控件上传文件到某共享目录. 已将在目录的共享安全和安全中加了 共享用户的 权限. 但通过浏览器访问共享目录文件报错:Access to the pat ...
- ZOJ3791_An Easy Game
给出两个等长的字符串,每次需要改变m个数字,每次必须改变k个数字,求从第一个串变化到第二个串的方案数. DP.f[i][j]改变i步后,有j个位置被改变的方案数.然后直接枚举当前改变的几个位置是前面重 ...
- 【温故Delphi】Win32API之CreateMutex
解决问题 如何让一个软件在一台机器上只能运行一个实例呢?这个问题用专业术语就是进程互斥.这个问题可以通过CreateMutex来解决. 进程互斥的核心思想:进程在启动时首先检查是否存在此进程实例,如果 ...
- java 读取文件内容 三种形式及效率对比
IOUtils.getStringFromReader() 读取方式为最快的 InputStream in = null; String line = ""; long start ...