基于python 凸包问题的解决
作者:学要fur_dich 发布时间:2021-04-11 02:56:41
最近在看python的算法书,之前在年前买的书,一直在工作间隙的时候,学习充电,终于看到这本书,但是确实又有点难,感觉作者写的代码太炫技 了,有时候注释也不怎么能看懂,终于想到一个方法,就是里面说的算法问题,我就百度python解决他,觉得这个挺好。
下面是凸包问题的一个代码。
# -*- coding: utf-8 -*-
import turtle
import random
import time
f=open('point.txt','w')
for i in range(100):
x=random.randrange(-250,250,10)
y=random.randrange(-200,200,10)
f.write(str(x)+','+str(y)+'\n')
f.close()
point=[]
f=open('point.txt')
for i in f:
try:
temp=i.split(',')
x=float(temp[0]); y=float(temp[1])
point.append((x,y))
except :
print 'a err'
f.close()
point=list(set(point))#去除重复的点
n=0
for i in range(len(point)):
if point[n][1]>point[i][1]:
n=i
p0=point[n]
point.pop(n)
def compare(a,b):
x=[a[0]-p0[0],a[1]-p0[1]]
y=[b[0]-p0[0],b[1]-p0[1]]
dx=(x[0]**2+x[1]**2)**0.5
dy=(y[0]**2+y[1]**2)**0.5
cosa=x[0]/dx
cosb=y[0]/dy
if cosa < cosb:
return 1
elif cosa > cosb:
return -1
else:
if dx<dy:
return -1
elif dx>dy:
return 1
else:
return 0
point.sort(compare)
point.insert(0,p0)
ep=point[:]#复制元素,操作ep不会对point产生影响
tag=0
while tag==0:
tag=1
l=len(ep)
for i in range(l):
i1,i2,i3=(i,(i+1)%l,(i+2)%l)
x,y,z=(ep[i1],ep[i2],ep[i3])
a1,a2=((y[0]-x[0],y[1]-x[1]),(z[0]-y[0],z[1]-y[1]))
if a1[0]*a2[1]-a1[1]*a2[0] < 0:
tag=0
ep.pop(i2)
break
elif a1[0]*a2[1]-a1[1]*a2[0]==0 and a1[0]*a2[0]<0:
#==0应改写,360度的情况
tag=0
ep.pop(i2)
break
def drawpoint(point,color,line):
p=turtle.getturtle()
p.hideturtle()
turtle.delay(1)
if(line=='p'):
p.speed(0)
for i in point:
p.pu()
p.color(color)
p.goto(i)
p.pd()
p.dot()
elif(line=='l'):
p.pu()
p.speed(1)
for i in point:
p.color(color)
p.goto(i)
p.pd()
p.dot()
p.goto(point[0])
drawpoint(point,'black','p')
drawpoint(ep,'red','l')
time.sleep(1)
补充知识:凸包问题的蛮力算法及python实现
蛮力法的基本思想是先用排除法确定凸包的顶点,然后按逆时针顺序输出这些顶点。在判断点P是不是凸包上的顶点时,有如下性质:
给定平面点集S,P,Pi,Pj,Pk是S中四个不同的点,如果P位于Pi,Pj,Pk组成的三角形内部或边界上,则P不是S的凸包顶点。
那么如何判断点P在三角形内部或边界上?给定平面两点AB,直线方程g(A,B,P)=0时,P位于直线上,g(A,B,P)>0和g(A,B,P)<0时,P分别位于直线的两侧。
判断点P在三角形内部或边界上只需依次检查P和三角形的每个顶点是否位于三角形另外两个顶点确定的直线的同一侧,即判断:
t1=g(pj,pk,p)*g(pj,pk,pi)>=0 ,
t2=g(pi,pk,p)*g(pi,pk,pj)>=0,
t3=g(pj,pi,p)*g(pj,pi,pk)>=0
是否同时成立
凸包问题的蛮力算法伪代码如下:
BruteForce(S):
输入:平面n个点的集合S
输出:按逆时针顺序输出S的凸包的所有顶点
If n=3 Then 以逆时针顺序输出S的顶点,算法结束 找到S中纵坐标最小的点P,该点一定位于凸包上
For S中任意三点Pi,Pj,Pk Do If Pi,Pj,Pk 一点位于其他两点与P构成的三角形内 Then 删除该点
找出S中横坐标最小的点A和横坐标最小的点B
将S划分问直线AB上方点集SU,直线AB下方点集SL,A,B两点属于SL
将SL按横坐标递增排序,SU按横坐标递减排序顺序输出SL,SU
首先随机生成点集S
import random
import itertools
def generate_num(n):
random_list = list(itertools.product(range(1, 100), range(1, 100)))
lists=random.sample(random_list, n)
return lists
判断点P在三角形内部或边界上,即判断点P是否在凸包上
在具体的判断过程中,尤其时坐标点比较密集的情况下,还有有三种比较特殊的情况
组成直线的两点垂直于x轴
除点P外其余三点在一条直线上时,不应删除点P,因为此时点P可能时凸包上的点
除点P外其余三点在一条直线上且垂直于x轴时,不应删除点P,因为此时点P可能时凸包上的点
#判断pi是否位于pj,pk,p0组成三角形内,返回t1,t2,t3三个变量
def isin(pi,pj,pk,p0):
x1 = float(p0[0])
x2 = float(pj[0])
x3 = float(pi[0])
x4 = float(pk[0])
y1 = float(p0[1])
y2 = float(pj[1])
y3 = float(pi[1])
y4 = float(pk[1])
k_j0=0
b_j0=0
k_k0=0
b_k0=0
k_jk=0
b_jk=0
perpendicular1=False
perpendicular2 = False
perpendicular3 = False
#pj,p0组成的直线,看pi,pk是否位于直线同一侧
if x2 - x1 == 0:
#pj,p0组成直线垂直于X轴时
t1=(x3-x2)*(x4-x2)
perpendicular1=True
else:
k_j0 = (y2 - y1) / (x2 - x1)
b_j0 = y1 - k_j0 * x1
t1 = (k_j0 * x3 - y3 + b_j0) * (k_j0 * x4 - y4 + b_j0)
#pk,p0组成的直线,看pi,pj是否位于直线同一侧
if x4 - x1 == 0:
#pk,p0组成直线垂直于X轴时
t2=(x3-x1)*(x2-x1)
perpendicular2=True
else:
k_k0 = (y4 - y1) / (x4 - x1)
b_k0 = y1 - k_k0 * x1
t2 = (k_k0 * x3 - y3 + b_k0) * (k_k0 * x2 - y2 + b_k0)
# pj,pk组成的直线,看pi,p0是否位于直线同一侧
if x4 - x2 == 0:
# pj,pk组成直线垂直于X轴时
t3=(x3-x2)*(x1-x2)
perpendicular3 = True
else:
k_jk = (y4 - y2) / (x4 - x2)
b_jk = y2 - k_jk * x2
t3 = (k_jk * x3 - y3 + b_jk) * (k_jk * x1 - y1 + b_jk)
#如果pk,p0,pj,三点位于同一直线时,不能将点删除
if (k_j0 * x4 - y4 + b_j0)==0 and (k_k0 * x2 - y2 + b_k0)==0 and (k_jk * x1 - y1 + b_jk)==0 :
t1=-1
#如果pk,p0,pj,三点位于同一直线时且垂直于X轴,不能将点删除
if perpendicular1 and perpendicular2 and perpendicular3:
t1=-1
return t1,t2,t3
接下来是实现算法主要部分,用来找出凸包上的点
import isintriangle
def force(lis,n):
#集合S中点个数为3时,集合本身即为凸包集
if n==3:
return lis
else:
#集合按纵坐标排序,找出y最小的点p0
lis.sort(key=lambda x: x[1])
p0=lis[0]
#除去p0的其余点集合lis_brute
lis_brute=lis[1:]
#temp是用来存放集合需要删除的点在lis_brute内的索引,四个点中如果有一个点在其余三个点组成的三角形内部,则该点一定不是凸包上的点
temp=[]
#三重循环找到所有这样在三角形内的点
for i in range(len(lis_brute)-2):
pi=lis_brute[i]
#如果索引i已经在temp中,即pi一定不是凸包上的点,跳过这次循环
if i in temp:
continue
for j in range(i+1,len(lis_brute) - 1):
pj=lis_brute[j]
#如果索引j已经在temp中,即pj一定不是凸包上的点,跳过这次循环
if j in temp:
continue
for k in range(j + 1, len(lis_brute)):
pk=lis_brute[k]
#如果索引k已经在temp中,即pk一定不是凸包上的点,跳过这次循环
if k in temp:
continue
#判断pi是否在pj,pk,p0构成的三角形内
(it1,it2,it3)=isintriangle.isin(pi,pj,pk,p0)
if it1>=0 and it2>=0 and it3>=0:
if i not in temp:
temp.append(i)
# 判断pj是否在pi,pk,p0构成的三角形内
(jt1,jt2,jt3)=isintriangle.isin(pj,pi,pk,p0)
if jt1>=0 and jt2>=0 and jt3>=0:
if j not in temp:
temp.append(j)
# 判断pk是否在pj,pi,p0构成的三角形内
(kt1, kt2, kt3) = isintriangle.isin(pk, pi, pj, p0)
if kt1 >= 0 and kt2 >= 0 and kt3 >= 0:
if k not in temp:
temp.append(k)
#listlast是最终选出的凸包集合
lislast=[]
for coor in lis_brute:
loc = [i for i, x in enumerate(lis_brute) if x == coor]
for x in loc:
ploc = x
if ploc not in temp:
lislast.append(coor)
#将p0加入凸包集合
lislast.append(p0)
return lislast
最后将凸包集合输出就不多说了,按照伪码上实现就可以,凸包蛮力算法在点集大小为1000时结果
来源:https://blog.csdn.net/xx5595480/article/details/54933571
猜你喜欢
- Bootstrap Validator是为Bootstrap3设计的一款表单验证jQuery插件,非常适合基于Bootstrap框架的网站。
- 1. yum list installed | grep php 查看安装的php版本mod_php72w.x86_64 7.2.1-1.w
- //遍历option和添加、移除optionfunction changeShipMethod(shipping){ var le
- 在ASP中,除了ADODB、Scripting 等一些常用组件外,我们还可以用微软的ActiveX方法来轻松捕获哟: <%u
- 在 Google 搜索结果页面中,将其 Logo 图标右键另存为后可以发现,它并非单纯的
- 使用php,定义php的默认语言. php.ini中: default_charset = "gb2312" 在网页中输
- 首先安装pip install ruamel.yaml用于修改yaml文件#coding:utf-8from ruamel import y
- CHAR和VARCHAR类型相似,差别主要在存储,尾随空格和检索方式上。CHAR和VARCHAR相同的是:CHAR和VARCHAR都指定了字
- 1.byte和str互转b = b"example" s = "example" bytes(s,
- 一、使用步骤 1.引入库(安装Python环境、PyQt、PyQt-tools)from PyQt5 import QtCore,
- 个人认为python的paramiko模块是运维人员必学模块之一,其ssh登录功能是旅行居家必备工具。安装paramiko很简单,pip i
- 使用环境 C#VSCodeM11. 安装MySQL下载MySQL软件,傻瓜式安装即可,安装完之后,在系统偏好号设置里会出现一个My
- 1.数组的索引我用的是iloc函数。导入数据是data,索引data.iloc[i,j],i代表行,j代表列。如果要索引i行之后的所有行元素
- 三元运算又称三目运算,是对简单的条件语句的简写简单条件语句:if 条件成立: val = 1else: val =
- 本文实例讲述了Python实现二分查找算法的方法。分享给大家供大家参考。具体实现方法如下:#!/usr/bin/env pythonimpo
- 1 停机方案发布公告停止服务离线数据迁移(拆分,重新分配数据)数据校验更改配置恢复服务回滚预案2 停写方案支持读写分离升级公告中断写操作,隔
- 最近由于业务的原因,需要在Web端页面接入调试各类的网络摄像头,遇到了很多匪夷所思的问题(说的就是读得出摄像头的品牌,读不出摄像头的分辨率)
- el-table格式化el-table-column内容遇到一个需求,一个循环展示的table中的某项,或者某几项需要格式化。对于格式化的方
- profile是什么当我们要对某一条sql的性能进行分析时,可以使用它。Profiling是从 mysql5.0.3版本以后才开放的。启动p
- 新建server.jsyarn init -yyarn add express nodemon -Dvar express = requir