题目描述:

  如何判断一个点是否在一个三角形内。

测试样例:

自定义的POINT类:

class POINT{int x;int y;public POINT(int x,int y){this.x = x;this.y = y;}
}

思路一:

面积法:

  如果一个点在三角形内,其与三角形的三个点构成的三个子三角形的面积等于大三角形的面积。否则,大于大三角形的面积。

  所以,这个问题就转化成如何在知道三角形的三个点的情况下,求这个三角形的面积的问题了。

  因为所有点的坐标已知,我们有几种方式计算面积:

  1)首先可以计算出每条边的长度及周长,我们就可以利用海伦公式计算面积,然后进行比较。

S = p ( p − a ) ( p − b ) ( p − c ) S = \sqrt{p(p-a)(p-b)(p-c)} S=p(p−a)(p−b)(p−c) ​

p = ( a + b + c ) 2 p = \frac{(a+b+c)}{2} p=2(a+b+c)​

  2)向量法:先求出这个三角形的对应的平行四边形的面积。然后这个面积的1/2就是三角形的面积了。

  先随意选择两个点,如B、C通过其坐标相减得向量(B,C)。记得谁减另一个就是指向谁。然后求出其中一个点和剩下一个点的向量。这两个向量的叉乘的便是平行四边形的面积。除以2就是三角形的面积。(注意这里是叉乘 (cross product),而非点乘(dot product))。

  (补充)向量之间的积分为两种:叉乘和点乘。叉乘求面积,点乘求投影。这是两者的意义。而且,叉乘理论得到的是一个向量,而点乘得到的是一个标量。

代码:

 private static final double ABS_DOUBLE_0 = 0.0001; public static boolean isInTriangle(POINT A, POINT B, POINT C, POINT P) {double ABC = triAngleArea(A, B, C);double ABp = triAngleArea(A, B, P);double ACp = triAngleArea(A, C, P);double BCp = triAngleArea(B, C, P);double sumOther = ABp + ACp + BCp;if (-ABS_DOUBLE_0 < (ABC - sumOther) && (ABC - sumOther) < ABS_DOUBLE_0) { // 若面积之和等于原三角形面积,证明点在三角形内return true;} else {return false;}}private static double triAngleArea(POINT A, POINT B, POINT C) { // 由三个点计算这三个点组成三角形面积POINT ab,bc;ab = new POINT(B.x - A.x,B.y - A.y);//bc = new POINT(C.x - B.x,C.y - B.y);return Math.abs((ab.x * bc.y - ab.y * bc.x) / 2.0);}

思路二:

同向法:

  假设点P位于三角形内,会有这样一个规律,当我们沿着ABCA的方向在三条边上行走时,你会发现点P始终位于边AB,BC和CA的右侧。我们就利用这一点,但是如何判断一个点在线段的左侧还是右侧呢?我们可以从另一个角度来思考,当选定线段AB时,点C位于AB的右侧,同理选定BC时,点A位于BC的右侧,最后选定CA时,点B位于CA的右侧,所以当选择某一条边时,我们只需验证点P与该边所对的点在同一侧即可。问题又来了,如何判断两个点在某条线段的同一侧呢?可以通过叉积来实现,连接PA,将PA和AB做叉积,再将CA和AB做叉积,如果两个叉积的结果方向一致,那么两个点在同一测。

代码:


public static boolean isInTriangle(POINT A, POINT B, POINT C, POINT P) {/*利用叉乘法进行判断,假设P点就是M点*/  int a = 0, b = 0, c = 0;  POINT MA = new POINT(P.x - A.x,P.y - A.y);  POINT MB = new POINT(P.x - B.x,P.y - B.y);  POINT MC = new POINT(P.x - C.x,P.y - C.y);  /*向量叉乘*/  a = MA.x * MB.y - MA.y * MB.x;  b = MB.x * MC.y - MB.y * MC.x;  c = MC.x * MA.y - MC.y * MA.x;  if((a <= 0 && b <= 0 && c <= 0)||  (a > 0 && b > 0 && c > 0))  return true;     return false;  }

思路三:

重心法:

  我们都知道,三角形的三个点在同一个平面上,如果选中其中一个点,其他两个点不过是相对该点的位移而已,比如选择点A作为起点,那么点B相当于在AB方向移动一段距离得到,而点C相当于在AC方向移动一段距离得到。

  所以对于平面内任意一点,都可以由如下方程来表示。

P = A + u ∗ ( C − A ) + v ∗ ( B − A ) P = A + u * (C-A) + v * (B-A) P=A+u∗(C−A)+v∗(B−A)

  如果系数u或v为负值,那么相当于朝相反的方向移动,即BA或CA方向。那么如果想让P位于三角形ABC内部,u和v必须满足什么条件呢?有如下三个条件:

u >= 0v >= 0u + v <= 1

  几个边界情况:

当u = 0,v = 0 时,就是点A;当u = 0,v = 1 时,就是点B;当u = 1,v = 0 时,就是点C。

  整理方程1得到:

P − A = u ( C − A ) + v ( B − A ) P-A = u(C-A) + v(B-A) P−A=u(C−A)+v(B−A)

  令 v0 = C – A, v1 = B – A, v2 = P – A,则

v 2 = u ∗ v 0 + v ∗ v 1 v2 = u * v0 + v * v1 v2=u∗v0+v∗v1

  现在是一个方程,两个未知数,无法解出u和v,所以将等式两边分别点乘v0和v1的到两个等式:(ps.下面公式中的·符号代表“点乘”)

( v 2 ) ⋅ v 0 = ( u ∗ v 0 + v ∗ v 1 ) ⋅ v 0 (v2)·v0 = (u * v0 + v * v1)·v0 (v2)⋅v0=(u∗v0+v∗v1)⋅v0

( v 2 ) ⋅ v 1 = ( u ∗ v 0 + v ∗ v 1 ) ⋅ v 1 (v2)·v1 = (u * v0 + v * v1)·v1 (v2)⋅v1=(u∗v0+v∗v1)⋅v1

  注意到这里u和v是数,而v0,v1和v2是向量,所以可以将点积展开得到下面的式子。

v 2 ⋅ v 0 = u ∗ ( v 0 ⋅ v 0 ) + v ∗ ( v 1 ⋅ v 0 ) v2·v0 = u * (v0·v0) + v * (v1·v0) v2⋅v0=u∗(v0⋅v0)+v∗(v1⋅v0)

v 2 ⋅ v 1 = u ∗ ( v 0 ⋅ v 1 ) + v ∗ ( v 1 ⋅ v 1 ) v2·v1 = u * (v0·v1) + v * (v1·v1) v2⋅v1=u∗(v0⋅v1)+v∗(v1⋅v1)

  解这个方程得到:

u = ( ( v 1 ⋅ v 1 ) ( v 2 ⋅ v 0 ) − ( v 1 ⋅ v 0 ) ( v 2 ⋅ v 1 ) ) / ( ( v 0 ⋅ v 0 ) ( v 1 ⋅ v 1 ) − ( v 0 ⋅ v 1 ) ( v 1 ⋅ v 0 ) ) u = ((v1·v1)(v2·v0)-(v1·v0)(v2·v1)) / ((v0·v0)(v1·v1) - (v0·v1)(v1·v0)) u=((v1⋅v1)(v2⋅v0)−(v1⋅v0)(v2⋅v1))/((v0⋅v0)(v1⋅v1)−(v0⋅v1)(v1⋅v0))

v = ( ( v 0 ⋅ v 0 ) ( v 2 ⋅ v 1 ) − ( v 0 ⋅ v 1 ) ( v 2 ⋅ v 0 ) ) / ( ( v 0 ⋅ v 0 ) ( v 1 ⋅ v 1 ) − ( v 0 ⋅ v 1 ) ( v 1 ⋅ v 0 ) ) v = ((v0·v0)(v2·v1)-(v0·v1)(v2·v0)) / ((v0·v0)(v1·v1) - (v0·v1)(v1·v0)) v=((v0⋅v0)(v2⋅v1)−(v0⋅v1)(v2⋅v0))/((v0⋅v0)(v1⋅v1)−(v0⋅v1)(v1⋅v0))

代码

private static boolean isInTriangle(POINT p, POINT a, POINT b, POINT c) {POINT AB, AC, AP;AB = new POINT(b.x - a.x, b.y - a.y);AC = new POINT(c.x - a.x, c.y - a.y);AP = new POINT(p.x - a.x, p.y - a.y);float dot00 = dotProduct(AC, AC);float dot01 = dotProduct(AC, AB);float dot02 = dotProduct(AC, AP);float dot11 = dotProduct(AB, AB);float dot12 = dotProduct(AB, AP);float inverDeno = 1 / (dot00 * dot11 - dot01 * dot01);// 计算重心坐标float u = (dot11 * dot02 - dot01 * dot12) * inverDeno;float v = (dot00 * dot12 - dot01 * dot02) * inverDeno;return (u >= 0) && (v >= 0) && (u + v < 1);}private static float dotProduct(POINT p1, POINT p2) {return p1.x * p2.x + p1.y * p2.y;
}

------至所有正在努力奋斗的程序猿们!加油!!
有码走遍天下 无码寸步难行
1024 - 梦想,永不止步!
爱编程 不爱Bug
爱加班 不爱黑眼圈
固执 但不偏执
疯狂 但不疯癫
生活里的菜鸟
工作中的大神
身怀宝藏,一心憧憬星辰大海
追求极致,目标始于高山之巅
一群怀揣好奇,梦想改变世界的孩子
一群追日逐浪,正在改变世界的极客
你们用最美的语言,诠释着科技的力量
你们用极速的创新,引领着时代的变迁

——乐于分享,共同进步,欢迎补充
——Treat Warnings As Errors
——Any comments greatly appreciated
——Talking is cheap, show me the code
——诚心欢迎各位交流讨论!QQ:1138517609
——CSDN:https://blog.csdn.net/u011489043
——简书:https://www.jianshu.com/u/4968682d58d1
——GitHub:https://github.com/selfconzrr

Java 判断一个点是否在一个三角形内相关推荐

  1. 能否构成三角形的条件代码_Java 如何判断一个点是否在一个三角形内

    题目描述: 如何判断一个点是否在一个三角形内. 测试样例: 自定义的POINT类: class POINT{ int x; int y; public POINT(int x,int y){ this ...

  2. Algorithm: 如何判断一个点是否在一个三角形内

    昨日因为机缘巧合,做了一道阿里的实习生编程题.题目很有趣,其中涉及到了如何判断一个点是否在一个三角形内. 其中,判断这个问题最简单的方法是面积法.(图片来源:http://www.cnblogs.co ...

  3. Java 判断一个点是否在一个多边形内

    工具类 提供:监测点的X轴.Y轴.多边形的多个坐标.如果存在多边形里面返回true,反之返回false import java.awt.geom.Point2D; import java.awt.ge ...

  4. JAVA判断一个地理坐标是否在一个多边形区域内和是否在一个圆形区域内(经纬度)

    怎么样判断一个坐标点在一个多边形区域内?包括规则多边形,不规则多边形,还有圆... 1 判断一个坐标是否在圆形区域内? 多边形和圆分开写,首先简单的就是判断是否在圆里面,如何判断一个坐标是否在圆形区域 ...

  5. 面试高频算法题补充系列:如何判断一个点是否在三角形内?

    前言 了解更多常考高频算法题可以关注 公众号:一个搬砖的胖子 企业面试题库:https://codetop.cc/ 小程序:CodeTop 该题曾出现在字节跳动.腾讯.网易.美团.小马智行等公司的面试 ...

  6. 如何判断一个点是否在一个多边形内?

    提示:对多边形进行分割,成为一个个三角形,判断点是否在三角形内. 一个非常有用的解析几何结论:如果P2(x1,y1),P2(x2,y2), P3(x3,y3)是平面上的3个点,那么三角形P1P2P3的 ...

  7. MFC判断一个点是否在一个区域内

    1.判断一点是否在矩形区域内的方法: CRect rc (point_1,point_2);//构造矩形区域 调用CRect::PtInRect     BOOL flag = rc.PtInRect ...

  8. 判断一个点是否在三角形内

    转自:https://www.cnblogs.com/simplekinght/p/9218310.html 面积法:若点P在三角形ABC内,则三角形ABP+三角形ACP+三角形BCP的面积等于三角形 ...

  9. 关于java中判断一个点是否在一个长方形内和一个长方形是否包含另一个长方形或者两个相重叠的问题。

    案例代码 import java.util.Scanner; public class rhh {     public static void main(String[] args){        ...

最新文章

  1. Oracle 11g 客户端使用
  2. 使用py2exe打包python脚本为exe可执行程序
  3. 结构体 CString QString 成员赋值出错
  4. ASP.NET 3.5 新特性开发向导实践(附项目源码下载)
  5. 【caffe】OpenCV Load caffe model
  6. 增大iphone音量技巧_原来苹果手机隐藏功能这么好用!调整这个设置,一键增大外放音量...
  7. 腾讯云TStack获“下一代云计算技术创新奖”,助力云生态信息创新发展
  8. 车流检测之halcon光流法算法实现
  9. HDU 2444 The Accomodation of Students 二分图匹配
  10. dj鲜生-15-用户的激活-默认激活置为0-点击链接状态置为1
  11. 5-1 可维护性的度量和构造原则
  12. 如何自定义一个注解(@Annotation)
  13. Java基础编程题:输入两个正整数,求它们的最大公约数和最小公倍数
  14. 微信小程序——案例:收货信息表单
  15. 网站建设计划书 网站建设计划书格式
  16. 什么模式下不可使用曝光补偿_摄影从零到入门 曝光模式与测光方法详解
  17. jit和jitx区别_JIT模式
  18. android保存到本地的图片、视频相册里找不到怎么办?
  19. Java时间处理(UTC时间和本地时间转换)
  20. 主窗口(08):【类】QMdiSubWindow [官翻]

热门文章

  1. 阿里彻底去中台了,你真以为中台不行了?
  2. [野狐行网游研究][二期][8.21更新]
  3. Python爬虫 金融股票数据
  4. R语言安装包出现 Error in loadNamespace(j - i[[1L]], c(lib.loc, .libPaths())错误
  5. 【Bluetooth|蓝牙开发】三、一篇文章,带你总览蓝牙协议
  6. Unix时间戳(Unix timestamp)转换工具 - 站长工具
  7. 对话黄骁俭:SAP的工程师文化
  8. 大华人脸门禁(人脸闸机)sdk集成对接javaweb接口springboot版
  9. 【244】Python 说出下面list1,list2,list3的输出值(新的默认列表仅仅只在函数被定义时创建一次)...
  10. jquery中serialize()