Triangle containment

Three distinct points are plotted at random on a Cartesian plane, for which -1000 ≤ x, y ≤ 1000, such that a triangle is formed.

Consider the following two triangles:

A(-340,495), B(-153,-910), C(835,-947)X(-175,41), Y(-421,-714), Z(574,-645)

It can be verified that triangle ABC contains the origin, whereas triangle XYZ does not.

Using triangles.txt (right click and ‘Save Link/Target As…’), a 27K text file containing the co-ordinates of one thousand “random” triangles, find the number of triangles for which the interior contains the origin.

NOTE: The first two examples in the file represent the triangles in the example given above.


包含原点的三角形

从笛卡尔平面中随机选择三个不同的点,其坐标均满足-1000 ≤ x, y ≤ 1000,这三个点构成一个三角形。

考虑下面两个三角形:

A(-340,495), B(-153,-910), C(835,-947)X(-175,41), Y(-421,-714), Z(574,-645)

可以验证三角形ABC包含原点,而三角形XYZ不包含原点。

在27K的文本文件triangles.txt(右击并选择“目标另存为……”)中包含了一千个“随机”三角形的坐标,找出其中包含原点在其内部的三角形的数量。

注意:文件中的前两个三角形就是上述样例。

解题

考虑了一下原点在三角形内部的三角形,原点到两个边的夹角应该有两个钝角,但是发现结果是510,表示不对,我是根据余弦定理就得costheta 这里有问题在实际中可能出现多于90度的情况然而在我计算中,我不知道怎么判断。还有个问题就是不知道我的这个想法是否有问题,下面的程序是不对的,留在这里待更改。

package Level4;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList; public class PE0102{
public static void run(){
int count = 0;
ArrayList<int[]> list = readData();
for(int i=0;i<list.size();i++){
int arr[] = list.get(i);
if(iscontainmentTriangle(arr)){
count+=1;
}
if(i<10)
System.out.println(iscontainmentTriangle(arr));
}
System.out.println(count); }
// 判断原点是否在三角形内部
public static boolean iscontainmentTriangle(int[] arr){
int count =0;
for(int i=0;i<arr.length-1;i+=2){
int x1 = arr[i];
int y1 = arr[i+1];
for(int j=i+2;j<arr.length-1;j+=2){
int x2 = arr[j];
int y2 = arr[j+1];
if(isObtuseAngle(x1,y1,x2,y2))
count ++;
if(count ==2)
return true;
}
}
return false;
} // 是不是钝角
public static boolean isObtuseAngle(int x1,int y1,int x2,int y2){
long costheta = x1*y1 + x2*y2;
if(costheta <0)
return true;
return false;
}
// 转换成整型数组
public static int [] StringtoInt(String[] strArr){
int[] IntArr = new int[strArr.length];
for(int i=0;i<strArr.length;i++)
IntArr[i] = Integer.parseInt(strArr[i]);
return IntArr;
}
// 读取数据
public static ArrayList<int[]> readData(){
String filename= "src/Level4/p102_triangles.txt";
ArrayList<int[]> list = new ArrayList<int[]>();
try {
BufferedReader bufferedReader = new BufferedReader(new FileReader(filename));
String line = "";
while((line=bufferedReader.readLine())!=null){
String[] strArr = line.split(",");
list.add(StringtoInt(strArr));
}
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return list;
}
public static void main(String[] args){
long t0 = System.currentTimeMillis();
run();
long t1 = System.currentTimeMillis();
long t = t1 - t0;
System.out.println("running time="+t/1000+"s"+t%1000+"ms"); }
}

Java Code

mathblog 中提到了根据三角形面积相等的方式求解,ABC = ABO + ACO +BCO

这里我们知道了三角形的三个点如何根据这三个点求面积,看了下面求解的方式,根据两个向量可以快速的求出向量所组成三角形的面积S= 向量交叉相乘差的绝对值的二分之一

wiki 中有说明

Java

package Level4;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.util.ArrayList; public class PE0102{
public static void run(){
int count = 0;
ArrayList<int[]> list = readData();
for(int i=0;i<list.size();i++){
int arr[] = list.get(i);
if(iscontainmentTria(arr)){
count+=1;
} }
System.out.println(count);
// 228
// running time=0s16ms
}
// 判断原点是否在三角形内部
public static boolean iscontainmentTria(int[] arr){
int area = 0;
int count =0;
int X1 = arr[0] - arr[2];
int Y1 = arr[1] - arr[3];
int X2 = arr[4] - arr[2];
int Y2 = arr[5] - arr[3];
int area2 = area(X1,Y1,X2,Y2);
for(int i=0;i<arr.length-1;i+=2){
int x1 = arr[i];
int y1 = arr[i+1];
for(int j=i+2;j<arr.length-1;j+=2){
int x2 = arr[j];
int y2 = arr[j+1];
area +=area(x1,y1,x2,y2);
}
}
if(area == area2)
return true;
return false;
}
// 这里面积的二倍
public static int area(int X1,int Y1,int X2,int Y2){
int area = Math.abs(X1*Y2 - X2*Y1);
return area;
}
// 转换成整型数组
public static int [] StringtoInt(String[] strArr){
int[] IntArr = new int[strArr.length];
for(int i=0;i<strArr.length;i++)
IntArr[i] = Integer.parseInt(strArr[i]);
return IntArr;
}
// 读取数据
public static ArrayList<int[]> readData(){
String filename= "src/Level4/p102_triangles.txt";
ArrayList<int[]> list = new ArrayList<int[]>();
try {
BufferedReader bufferedReader = new BufferedReader(new FileReader(filename));
String line = "";
while((line=bufferedReader.readLine())!=null){
String[] strArr = line.split(",");
list.add(StringtoInt(strArr));
}
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return list;
}
public static void main(String[] args){
long t0 = System.currentTimeMillis();
run();
long t1 = System.currentTimeMillis();
long t = t1 - t0;
System.out.println("running time="+t/1000+"s"+t%1000+"ms");
}
}

Python

# coding=gbk

import time as time
import re
import math
import numpy as np
def run():
filename = 'E:/java/projecteuler/src/Level4/p102_triangles.txt'
mat = readData(filename)
mat = np.array(mat)
count = 0
for line in mat:
if isContainmentTraingle(line):
count+=1
print count def isContainmentTraingle(triangle):
X1 = triangle[0] - triangle[2]
Y1 = triangle[1] - triangle[3]
X2 = triangle[4] - triangle[2]
Y2 = triangle[5] - triangle[3]
S = area(X1,Y1,X2,Y2)
for i in range(0,4,2):
for j in range(i+2,5,2):
S -= area(triangle[i],triangle[i+1],triangle[j],triangle[j+1])
return S == 0 def area(x1,y1,x2,y2):
S = np.abs(x1*y2 - x2*y1)
return S def readData(filename):
mat = list()
file = open(filename)
for line in file:
row = line.split(',')
row = [int(x) for x in row]
mat.append(row)
return mat t0 = time.time()
run()
t1 = time.time()
print "running time=",(t1-t0),"s"

Project Euler 102:Triangle containment 包含原点的三角形的更多相关文章

  1. Python练习题 040:Project Euler 012:有超过500个因子的三角形数

    本题来自 Project Euler 第12题:https://projecteuler.net/problem=12 # Project Euler: Problem 12: Highly divi ...

  2. [project euler] program 4

    上一次接触 project euler 还是2011年的事情,做了前三道题,后来被第四题卡住了,前面几题的代码也没有保留下来. 今天试着暴力破解了一下,代码如下: (我大概是第 172,719 个解出 ...

  3. Python练习题 029:Project Euler 001:3和5的倍数

    开始做 Project Euler 的练习题.网站上总共有565题,真是个大题库啊! # Project Euler, Problem 1: Multiples of 3 and 5 # If we ...

  4. Project Euler 9

    题意:三个正整数a + b + c = 1000,a*a + b*b = c*c.求a*b*c. 解法:可以暴力枚举,但是也有数学方法. 首先,a,b,c中肯定有至少一个为偶数,否则和不可能为以上两个 ...

  5. Project Euler 44: Find the smallest pair of pentagonal numbers whose sum and difference is pentagonal.

    In Problem 42 we dealt with triangular problems, in Problem 44 of Project Euler we deal with pentago ...

  6. project euler 169

    project euler 169 题目链接:https://projecteuler.net/problem=169 参考题解:http://tieba.baidu.com/p/2738022069 ...

  7. 【Project Euler 8】Largest product in a series

    题目要求是: The four adjacent digits in the 1000-digit number that have the greatest product are 9 × 9 × ...

  8. Project Euler 第一题效率分析

    Project Euler: 欧拉计划是一系列挑战数学或者计算机编程问题,解决这些问题需要的不仅仅是数学功底. 启动这一项目的目的在于,为乐于探索的人提供一个钻研其他领域并且学习新知识的平台,将这一平 ...

  9. Python练习题 049:Project Euler 022:姓名分值

    本题来自 Project Euler 第22题:https://projecteuler.net/problem=22 ''' Project Euler: Problem 22: Names sco ...

随机推荐

  1. Effective Objective-C 2.0之Note.03(属性详解)

    用Objective-C等面向对象语言编程时,“对象”(object)就是“基本构造单元”(building block),开发者可以通过对象来存储并传递数据.在对象之间传递数据并执行任务的过程就叫做 ...

  2. [iOS]MVVM-框架介绍

       我于 2011 年在 500px 找到自己的第一份 iOS 开发工作.虽然我已经在大学里做了好几年 iOS 外包开发,但这才是我的一个真正的 iOS 开发工作.我被作为唯一的 iOS 开发者被招 ...

  3. P3383: [Usaco2004 Open]Cave Cows 4 洞穴里的牛之四

    这个系列总算是做完了,这是我第一次高效率做完四道题,虽然中间有两道水题,但是第一和第四题还是蛮好的,但是只要能想到思路就很快能打完的. 像这道题,刚开始在想能不能用DP?但是苦于不知道怎么实施,后来又 ...

  4. UIView 添加子视图的常用方法

    1.  - (void)addSubview:(UIView *)view 这是最常用的方法有两个注意点 参数view可以是nil,运行不会报错,当然,父视图的subViews也不会增加. 此方法增加 ...

  5. Log4Net学习【二】

    Log4Net结构详解 当我们在描述为系统做日志这个动作的时候,实际上描述了3个点:做日志,其实就是在规定,在什么地方 用什么日志记录器 以什么样的格式做日志.把三个最重要的点抽取出来,即什么地方,日 ...

  6. loadrunner简单使用——HTTP,WebService,Socket压力测试脚本编写

    使用loadrunner进行压力测试主要分两步,第一步是编写脚本(比较重点),第二步执行测试(配置都是在界面上点点就行了,当然我只的是比较简单的,能满足日常需要的),第三步分析结果(这一步比较高深,但 ...

  7. Daily Scrum1--团队项目分工及估计时间

    团队项目分工及估计时间 PM(黄剑锟): 任务一:监督进度,将每一天完成的任务总结,在各个部分进行协调与帮助.(贯穿整个项目周期) 任务二:提高搜索反应时间,优化搜索算法.(估计时间8小时) 程序设计 ...

  8. PSP记录表

    学生     崔乐乐                 日期    2015/3/15 教师     王建民                 课程    软件工程 周活动总结表 任务 日期 听课 写程序 ...

  9. Node.js 项目搭建

    关于 本书致力于教会你如何用Node.js来开发应用,过程中会传授你所有所需的“高级”JavaScript知识.本书绝不是一本“Hello World”的教程. 状态 你正在阅读的已经是本书的最终版. ...

  10. JS 学习笔记--7---正则表达式

    正则表达式中的内容很多,也很深,下面只是一些基本的知识点,练习中使用的浏览器是IE10,若有不当处请各位朋友指正,我会在第一时间修改错误之处. 匹配的概念:是包含的意思,不是相等的意思 1.正则表达式 ...