Python使用DDA算法和中点Bresenham算法画直线
title: "Python使用DDA算法和中点Bresenham算法画直线"
date: 2018-06-11T19:28:02+08:00
tags: ["图形学"]
categories: ["Python"]
先上效果图

代码
#!/usr/bin/env python
# coding=utf-8
from pylab import *
from matplotlib.ticker import MultipleLocator
import matplotlib.patches as patches
'''
1. 输入直线两端点 x0,y0 xn,yn
2. 计算初始值delta_x, delta_y,k=delta_y/delta_x,d=0, x=x0,y=y0
3. 绘制点x,y
4. d更新为d+k,若d>0.5,则x,y更新为x+1,y+1,d=d-1;否则x,y更新为x+1,y
5. 重复3,4直到直线画完
'''
def init(ax, width):
# 设置长宽
ax.axis([0, width, 0, width])
# 设置主刻度标签的位置,标签文本的格式
majorLocator = MultipleLocator(1)
minorLocator = MultipleLocator(0.5)
ax.xaxis.set_major_locator(majorLocator)
ax.yaxis.set_major_locator(majorLocator)
# ax.xaxis.set_minor_locator(minorLocator)
# ax.yaxis.set_minor_locator(minorLocator)
ax.grid(True) # x坐标轴的网格使用主刻度
def add_pixel(x, y, ax, c):
x = round(x)
y = round(y)
if c == 1:
ax.add_patch(patches.Rectangle((x - 0.5, y - 0.5), 1, 1, color='b'))
ax.plot(x, y, 'r.')
else:
ax.add_patch(patches.Rectangle((x - 0.5, y - 0.5), 1, 1))
ax.plot(x, y, 'y.')
if __name__ == '__main__':
# 将一行的字符串分割并转化为数字
x0, y0, x1, y1, width = map(int, input("输入直线的两点和画布的边长: ").split(' '))
if x0>x1:
x0,x1=x1,x0
y0,y1=y1,y0
ax = subplot(121, aspect='equal',
title='modified Bresenham') # 改进的bresenham
ax.plot([x0, x1], [y0, y1], '-k')
bx = subplot(122, aspect='equal', title='DDA') # DDA
bx.plot([x0, x1], [y0, y1], '-k')
# 图形初始化
init(ax, width)
init(bx, width)
delta_x = x1 - x0
delta_y = y1 - y0
d = 0
if delta_x == 0:
k = 999999999
else:
k = delta_y / delta_x
x = round(x0)
y = round(y0)
'''
DDA算法
'''
if k > -1 and k < 1:
# X 最大位移
while True:
if x > x1:
break
add_pixel(x, y, bx, 1)
x = x+1
y = y+k
elif k >= 1:
# Y 最大位移
while True:
if y > y1:
break
add_pixel(x, y, bx, 1)
y = y+1
x = x+1/k
else:
while True:
if y < y1:
break
add_pixel(x, y, bx, 1)
y = y-1
x = x-1/k
'''
k的范围
1. (0,1) x为最大位移,y正向增加
2. (1,+inf) y为最大位移,x正向增加
3. (0,-1) x为最大位移,y负向增加
4. (-1,-inf)y为最大位移,y减小。x正向增加
'''
x = x0
y = y0
if k > 1:
while True:
if y > y1:
break
add_pixel(x, y, ax, 0)
y = y + 1
d = d + 1 / k
if d > 0.5:
x = x + 1
d = d - 1
elif k > 0:
while True:
if x > x1:
break
add_pixel(x, y, ax, 0)
x = x + 1
d = d + k
if d > 0.5:
y = y + 1
d = d - 1
elif k > -1:
while True:
if x > x1:
break
add_pixel(x, y, ax, 0)
x = x + 1
d = d - k
if d > 0.5:
y = y - 1
d = d - 1
else:
while True:
if y < y1:
break
add_pixel(x, y, ax, 0)
y = y - 1
d = d - 1 / k
if d > 0.5:
x = x + 1
d = d - 1
show()
Python使用DDA算法和中点Bresenham算法画直线的更多相关文章
- 《图形学》实验七:中点Bresenham算法画椭圆
开发环境: VC++6.0,OpenGL 实验内容: 使用中点Bresenham算法画椭圆. 实验结果: 代码: #include <gl/glut.h> #define WIDTH 50 ...
- 《图形学》实验六:中点Bresenham算法画圆
开发环境: VC++6.0,OpenGL 实验内容: 使用中点Bresenham算法画圆. 实验结果: 代码: #include <gl/glut.h> #define WIDTH 500 ...
- 《图形学》实验四:中点Bresenham算法画直线
开发环境: VC++6.0,OpenGL 实验内容: 使用中点Bresenham算法画直线. 实验结果: 代码: //中点Bresenham算法生成直线 #include <gl/glut.h& ...
- 利用canvas实现的中点Bresenham算法
Bresenham提出的直线生成算法的基本原理是,每次在最大位移方向上走一步,而另一个方向是走步还是不走步取决于误差项的判别,具体的实现过程大家可以去问度娘.我主要是利用canvas画布技术实现了这个 ...
- [DEBUG]椭圆的中点Bresenham算法边缘绘制出现错误
在使用椭圆的中点Bresenham算法绘制椭圆时, 当椭圆足够大时, 椭圆的边缘会出现下面这种情况. 出错原因: 将a, b声明为了int类型, 导致中点判别式中发生溢出 关注后面的a*b*a*b当a ...
- 【转】Bresenham快速画直线算法
一. 算法原理简介: 算法原理的详细描述及部分实现可参考: http://www.cs.helsinki.fi/group/goa/mallinnus/lines/bresen ...
- Bresenham快速画直线算法
现在的计算机的图像的都是用像素表示的,无论是点.直线.圆或其他图形最终都会以点的形式显示.人们看到屏幕的直线只不过是模拟出来的,人眼不能分辨出来而已.那么计算机是如何画直线的呢,其实有比较多的算法,这 ...
- 《图形学》实验五:改进的Bresenham算法画直线
开发环境: VC++6.0,OpenGL 实验内容: 使用改进的Bresenham算法画直线. 实验结果: 代码: //中点Bresenham算法生成直线 #include <gl/glut.h ...
- 直线的中点Bresenham算法的实现
一.实验目的 1.掌握在MFC中搭建图形绘制的基本框架的方法: 2.将直线的中点Bresenham算法转化成可执行代码. 二.实验内容 1. 通过分析具体数据在中点Bresenham算法上的执行过程, ...
随机推荐
- WRITE T AFTER ADVANCING 2 LINES
WRITE T AFTER ADVANCING 2 LINES 意思是“在前进两行之后打印T记录内容” (ADVANCING 和 LINE 字可省略) WRITE T BEFORE ADVANCIN ...
- JavaScript异步加载的三种方式——async和defer、动态创建script
一.script标签的位置 传统的做法是:所有script元素都放在head元素中,必须等到全部js代码都被下载.解析.执行完毕后,才能开始呈现网页的内容(浏览器在遇到<body>标签时才 ...
- Docker操作笔记(二)容器
容器 一.启动容器 启动一个容器有两种方式: 1.基于镜像新键并启动一个容器: 所需要的主要命令为docker run docker run ubuntu:18.04 /bin/echo " ...
- AMPPZ-2015 (MIPT Workshop Open 1)
A. Album of Numbers 设$cnt[i]$表示数字$i$的个数,则$ans=\frac{\sum_{i} i\times cnt[i]\prod_{j>i}(cnt[j]+1)} ...
- [HDU4864]Task (贪心)
此图和上一篇博客的图一起看有奇效 题意 https://vjudge.net/problem/HDU-4864 思路 贪心 代码 by lyd 我实在是敲不来 #include <iostrea ...
- Oracle命令行中显示:ORA-04076: 无效的 NEW 或 OLD 说明
Oracle命令行进行操作时可能出现"ORA-04076: 无效的 NEW 或 OLD 说明" 需要在条件语句中JOB前面添加“old.”即可(因为是在when条件里面,所以不用“ ...
- js一些代码
1判断金额正则 var reg = /(^[1-9]([0-9]+)?(\.[0-9]{1,2})?$)|(^(0){1}$)|(^[0-9]\.[0-9]([0-9])?$)/; var money ...
- webpack3_脚手架
webpack 记得 --save-dev 装入开发依赖 更新迭代快,需要有根据报错解决问题的能力,来融会贯通这个工具 这里的是 webpack3,其实已经到了 webpack4 了 采用了 w ...
- CSS3_扇形导航_transitionend 事件
扇形导航 圆形按钮,切换一系列图片导航的显示与隐藏. 如果涉及过渡动画,定位的 top 和 left 必须写 Math.sin(弧度) 一圈弧度 = 2π,一圈角度 = 360 弧度 = (deg*2 ...
- linux学习:wget与lynx用法整理
指令:wget.lynx.axel wget url #下载数据写入文件,下载的文件名与url中的文件名保持一致,下载信息或进度写入stdoutwget url1 url2 url3 #下载多 ...