搜索
bottom↓
回复: 28
打印 上一主题 下一主题

有偿(300¥):判断一个点是否在封闭形状内的算法

  [复制链接]

出0入475汤圆

跳转到指定楼层
1
发表于 2020-7-11 09:05:46 来自手机 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
还是300元寻一个算法函数:判断某一个点(x,y)是否在一个已知的封闭区域(目前起来看只有圆形和多边形这两种情况)内,封闭区域的参数为圆形的圆心及半径(圆心x1,y1;半径r米),以及多边形各个点(分别为x1,y1,x2,y2。。。xn,yn,n最大限定为100)。坐标均为浮点数的经纬度值。圆半径也是浮点。当然有更好的模式可以自行优化,因为用于mcu中,可能希望是速度快以及消耗资源小了:)

阿莫论坛20周年了!感谢大家的支持与爱护!!

月入3000的是反美的。收入3万是亲美的。收入30万是移民美国的。收入300万是取得绿卡后回国,教唆那些3000来反美的!

出0入0汤圆

2
发表于 2020-7-11 09:31:24 来自手机 | 只看该作者
圆形好判断,点到圆心的距离少于直径就在圆内。多边形,就判断点同在多边形所有边的的“右边”或同在“左边”。

出10入23汤圆

3
发表于 2020-7-11 09:34:23 来自手机 | 只看该作者
我有代码,报价3000

出0入0汤圆

4
发表于 2020-7-11 09:35:42 | 只看该作者
和 GUI 的裁剪算法 类似吧。 网上搜搜

出0入309汤圆

5
发表于 2020-7-11 09:44:31 来自手机 | 只看该作者
本帖最后由 iamseer 于 2020-7-11 10:27 编辑

圆形很简单,算一下到圆心距离然后和半径比比。

多边形最简单的方法是射线法,一点朝任意方向做射线,和多边形相交次数为奇数则在多边形中。最简单射线那就是朝右

  1. #define MIN(x,y) (x < y ? x : y)
  2. #define MAX(x,y) (x > y ? x : y)
  3. #define INSIDE 0
  4. #define OUTSIDE 1

  5. typedef struct {
  6.    double x,y;
  7. } Point;

  8. int InsidePolygon(Point *polygon,int N,Point p)
  9. {
  10.   int counter = 0;
  11.   int i;
  12.   double xinters;
  13.   Point p1,p2;

  14.   p1 = polygon[0];
  15.   for (i=1;i<=N;i++) {
  16.     p2 = polygon[i % N];
  17.     if (p.y > MIN(p1.y,p2.y)) {
  18.       if (p.y <= MAX(p1.y,p2.y)) {
  19.         if (p.x <= MAX(p1.x,p2.x)) {
  20.           if (p1.y != p2.y) {
  21.             xinters = (p.y-p1.y)*(p2.x-p1.x)/(p2.y-p1.y)+p1.x;
  22.             if (p1.x == p2.x || p.x <= xinters)
  23.               counter++;
  24.           }
  25.         }
  26.       }
  27.     }
  28.     p1 = p2;
  29.   }

  30.   if (counter % 2 == 0)
  31.     return(OUTSIDE);
  32.   else
  33.     return(INSIDE);
  34. }
复制代码




gif图片在手机上渲染不正确,补个jpg


另外 https://stackoverflow.com/questi ... is-within-a-polygon 还提到一点比较重要的优化:先找到多边形的外接长方形。如果点在外接长方形里面的话再进一步分析,否则就不用跑这套算法。

你既然提到是MCU上跑,那么分支的开销应该比三角函数运算的开销低。另一种算法卷绕法应该不合适。

http://www.eecs.umich.edu/course ... OJ2/InsidePoly.html

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入0汤圆

6
发表于 2020-7-11 09:55:14 | 只看该作者
网上找到的
struct Point
{
    double x, y;
};
bool IsInPolygon(Point p,Point *ptPolygon,int ncount)
{
    int ncross = 0;
    for (int i = 0; i < ncount; i++)
    {
        Point p1 = ptPolygon;
        Point p2 = ptPolygon[(i + 1) % ncount]; //相邻两条边p1,p2
        if (p1.y == p2.y)         
            continue;
        if (p.y < min(p1.y, p2.y))
            continue;
        if (p.y >= max(p1.y, p2.y))
            continue;
        double x = (p.y - p1.y)*(p2.x - p1.x) / (p2.y - p1.y) + p1.x;
        if (x > p.x)
            ncross++; //只统计单边交点
    }
    return(ncross % 2 == 1);
}

链接:https://blog.csdn.net/qq_27161673/article/details/52973866
https://blog.csdn.net/F2304/arti ... earnPai2-2.nonecase

出0入46汤圆

7
发表于 2020-7-11 09:57:20 | 只看该作者
本帖最后由 jasonzhu8888 于 2020-7-11 10:04 编辑

  1. struct Point
  2. {
  3.         float x;
  4.         float y;
  5. };

  6. bool circle_contains(Point* p_center, float radius, float x, float y)
  7. {
  8.         float xx = p_center->x - x;
  9.         float yy = p_center->y - y;
  10.         return xx*xx + yy*yy <= radius*radius;
  11. }

  12. bool polygon_contains(Point* pypts,int num,float x, float y)
  13. {
  14.         int hits = 0;

  15.         int n = num - 1;
  16.         Point* plastpt = &pypts[n];
  17.         float lastx = plastpt->x;
  18.         float lasty = plastpt->y;
  19.         float curx, cury;

  20.         for (int i = 0; i < n; lastx = curx, lasty = cury, i++)
  21.         {
  22.                 Point* p = &pypts[i];
  23.                 curx = p->x;
  24.                 cury = p->y;

  25.                 if (cury == lasty)
  26.                 {
  27.                         continue;
  28.                 }

  29.                 float leftx;
  30.                 if (curx < lastx)
  31.                 {
  32.                         if (x >= lastx)
  33.                         {
  34.                                 continue;
  35.                         }
  36.                         leftx = curx;
  37.                 }
  38.                 else
  39.                 {
  40.                         if (x >= curx)
  41.                         {
  42.                                 continue;
  43.                         }
  44.                         leftx = lastx;
  45.                 }

  46.                 float test1, test2;
  47.                 if (cury < lasty)
  48.                 {
  49.                         if (y < cury || y >= lasty)
  50.                         {
  51.                                 continue;
  52.                         }
  53.                         if (x < leftx)
  54.                         {
  55.                                 hits++;
  56.                                 continue;
  57.                         }
  58.                         test1 = x - curx;
  59.                         test2 = y - cury;
  60.                 }
  61.                 else
  62.                 {
  63.                         if (y < lasty || y >= cury)
  64.                         {
  65.                                 continue;
  66.                         }
  67.                         if (x < leftx)
  68.                         {
  69.                                 hits++;
  70.                                 continue;
  71.                         }
  72.                         test1 = x - lastx;
  73.                         test2 = y - lasty;
  74.                 }

  75.                 if (test1 < (test2 / (lasty - cury) * (lastx - curx)))
  76.                 {
  77.                         hits++;
  78.                 }
  79.         }

  80.         return ((hits & 1) != 0);
  81. }
复制代码

出0入46汤圆

8
发表于 2020-7-11 10:01:18 | 只看该作者
polygon_containes 前面两个参数是输入的多边形坐标,最好首尾的点是同一个形成闭环,也即是输入点至少4个
如  (x0 y0)  (x1 y1) ... (x n-1  y n-1) (x0 y0)

出0入0汤圆

9
发表于 2020-7-11 10:01:35 | 只看该作者
点到线的距离,
https://www.amobbs.com/thread-5728196-1-1.html

出0入162汤圆

10
发表于 2020-7-11 10:15:46 来自手机 | 只看该作者
这个是自动割草机上用的吧

出10入23汤圆

11
发表于 2020-7-11 10:39:21 | 只看该作者





本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入0汤圆

12
发表于 2020-7-11 11:43:55 | 只看该作者

厉害,matlab能生成 c 代码吧

出10入23汤圆

13
发表于 2020-7-11 12:05:05 来自手机 | 只看该作者
19504643 发表于 2020-7-11 11:43
厉害,matlab能生成 c 代码吧

c代码写的啊,matlab只是调用它做验证用的

出0入0汤圆

14
发表于 2020-7-11 12:19:33 来自手机 | 只看该作者
如何判断一个点是否在多边形内部? (1)面积和判别法:判断目标点与多边形的每条边组成的三角形面积和是否等于该多边形,相等则在多边形内部。--采纳 (2)夹角和判别法:判断目标点与所有边的夹角和是否为360度,为360度则在多边形内部。 (3)引射线法:从目标点出发引一条射线,看这条射线和多边形所有边的交点数目。如果有奇数个交点,则说明在内部,如果有偶数个交点,则说明在外部。  (4)转角法:按照多边形顶点逆时针顺序,根据顶点和判断点连线的方向正负(设定角度逆时针为正)求和判断;

出0入475汤圆

15
 楼主| 发表于 2020-7-14 09:24:54 | 只看该作者
感谢楼上各位的无私帮助,这几天耽搁了今天才来看到。
我不知道报酬该给哪一位了,当然我自己也都还没有来得及验证,不过一定会兑现的。
如果可以的话,可否私信给我,因为我是想直接使用经纬度作为参数,里面涉及到跨半球操作以及一些经纬度转换什么的不是很懂了,如果可以,就帮我一并处理一下 再次感谢

出0入0汤圆

16
发表于 2020-7-14 10:43:13 | 只看该作者
这不就是禁飞区用得很普遍的算法。

出105入79汤圆

17
发表于 2020-7-14 12:26:20 | 只看该作者
禁飞区算法合集

出0入0汤圆

18
发表于 2020-7-14 14:13:20 | 只看该作者
最早在部标的行车记录仪 上有过电子围栏算法

出0入0汤圆

19
发表于 2020-7-14 14:20:05 | 只看该作者
网上搜射线法                     

出0入4汤圆

20
发表于 2020-7-14 15:41:30 | 只看该作者
mark 最近可能会用到

出0入0汤圆

21
发表于 2020-7-14 15:46:42 | 只看该作者
高手如云啊

出0入0汤圆

22
发表于 2020-7-15 21:40:10 | 只看该作者
学习一下这个算法。

出0入0汤圆

23
发表于 2020-7-16 13:15:05 | 只看该作者
1a2b3c 发表于 2020-7-14 09:24
感谢楼上各位的无私帮助,这几天耽搁了今天才来看到。
我不知道报酬该给哪一位了,当然我自己也都还没有来 ...

用经纬度做参数的话,要做高斯-克吕格投影,有个复杂的计算公式。

出0入475汤圆

24
 楼主| 发表于 2020-7-16 14:49:18 | 只看该作者
wangzheyu 发表于 2020-7-16 13:15
用经纬度做参数的话,要做高斯-克吕格投影,有个复杂的计算公式。

是这样的,我的区域很小,撑死了就5公里的最大边距,所以是不是就无需要这样处理了呢?
而且也不是两极地区,误差也可以允许5米左右都可以。

出10入23汤圆

25
发表于 2020-7-16 17:52:16 来自手机 | 只看该作者
输入围栏经纬度,输入目标经纬度,输出目标是否在围栏内,支持半个球的跨程,单片机无缝移植,交期一个晚上,楼主可有性趣?

出0入475汤圆

26
 楼主| 发表于 2020-9-19 07:50:46 来自手机 | 只看该作者
抱歉你这个回复今天才看到,我私信你吧

出0入162汤圆

27
发表于 2020-9-19 08:11:01 来自手机 | 只看该作者
楼主是做自动割草机的吧

出0入475汤圆

28
 楼主| 发表于 2020-9-19 08:34:37 来自手机 | 只看该作者
回楼上和前面同样的问题,不是割草机和飞行的那些,之前都没有想到原来这个应该是一样的目的:)我是自己的一个想法,需要先验证一下,离实现整机还早,所以就需要个这样的功能,但自己又不会所以才来寻求帮助的,

出0入0汤圆

29
发表于 2024-3-21 08:38:02 | 只看该作者
不错,标记下
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-4-23 20:16

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表