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算法画直线的更多相关文章

  1. 《图形学》实验七:中点Bresenham算法画椭圆

    开发环境: VC++6.0,OpenGL 实验内容: 使用中点Bresenham算法画椭圆. 实验结果: 代码: #include <gl/glut.h> #define WIDTH 50 ...

  2. 《图形学》实验六:中点Bresenham算法画圆

    开发环境: VC++6.0,OpenGL 实验内容: 使用中点Bresenham算法画圆. 实验结果: 代码: #include <gl/glut.h> #define WIDTH 500 ...

  3. 《图形学》实验四:中点Bresenham算法画直线

    开发环境: VC++6.0,OpenGL 实验内容: 使用中点Bresenham算法画直线. 实验结果: 代码: //中点Bresenham算法生成直线 #include <gl/glut.h& ...

  4. 利用canvas实现的中点Bresenham算法

    Bresenham提出的直线生成算法的基本原理是,每次在最大位移方向上走一步,而另一个方向是走步还是不走步取决于误差项的判别,具体的实现过程大家可以去问度娘.我主要是利用canvas画布技术实现了这个 ...

  5. [DEBUG]椭圆的中点Bresenham算法边缘绘制出现错误

    在使用椭圆的中点Bresenham算法绘制椭圆时, 当椭圆足够大时, 椭圆的边缘会出现下面这种情况. 出错原因: 将a, b声明为了int类型, 导致中点判别式中发生溢出 关注后面的a*b*a*b当a ...

  6. 【转】Bresenham快速画直线算法

    一.             算法原理简介: 算法原理的详细描述及部分实现可参考: http://www.cs.helsinki.fi/group/goa/mallinnus/lines/bresen ...

  7. Bresenham快速画直线算法

    现在的计算机的图像的都是用像素表示的,无论是点.直线.圆或其他图形最终都会以点的形式显示.人们看到屏幕的直线只不过是模拟出来的,人眼不能分辨出来而已.那么计算机是如何画直线的呢,其实有比较多的算法,这 ...

  8. 《图形学》实验五:改进的Bresenham算法画直线

    开发环境: VC++6.0,OpenGL 实验内容: 使用改进的Bresenham算法画直线. 实验结果: 代码: //中点Bresenham算法生成直线 #include <gl/glut.h ...

  9. 直线的中点Bresenham算法的实现

    一.实验目的 1.掌握在MFC中搭建图形绘制的基本框架的方法: 2.将直线的中点Bresenham算法转化成可执行代码. 二.实验内容 1. 通过分析具体数据在中点Bresenham算法上的执行过程, ...

随机推荐

  1. WRITE T AFTER ADVANCING 2 LINES

    WRITE T AFTER ADVANCING 2 LINES 意思是“在前进两行之后打印T记录内容”  (ADVANCING 和 LINE 字可省略) WRITE T BEFORE ADVANCIN ...

  2. JavaScript异步加载的三种方式——async和defer、动态创建script

    一.script标签的位置 传统的做法是:所有script元素都放在head元素中,必须等到全部js代码都被下载.解析.执行完毕后,才能开始呈现网页的内容(浏览器在遇到<body>标签时才 ...

  3. Docker操作笔记(二)容器

    容器 一.启动容器 启动一个容器有两种方式: 1.基于镜像新键并启动一个容器: 所需要的主要命令为docker run docker run ubuntu:18.04 /bin/echo " ...

  4. 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)} ...

  5. [HDU4864]Task (贪心)

    此图和上一篇博客的图一起看有奇效 题意 https://vjudge.net/problem/HDU-4864 思路 贪心 代码 by lyd 我实在是敲不来 #include <iostrea ...

  6. Oracle命令行中显示:ORA-04076: 无效的 NEW 或 OLD 说明

    Oracle命令行进行操作时可能出现"ORA-04076: 无效的 NEW 或 OLD 说明" 需要在条件语句中JOB前面添加“old.”即可(因为是在when条件里面,所以不用“ ...

  7. js一些代码

    1判断金额正则 var reg = /(^[1-9]([0-9]+)?(\.[0-9]{1,2})?$)|(^(0){1}$)|(^[0-9]\.[0-9]([0-9])?$)/; var money ...

  8. webpack3_脚手架

    webpack    记得 --save-dev 装入开发依赖 更新迭代快,需要有根据报错解决问题的能力,来融会贯通这个工具 这里的是 webpack3,其实已经到了 webpack4 了 采用了 w ...

  9. CSS3_扇形导航_transitionend 事件

    扇形导航 圆形按钮,切换一系列图片导航的显示与隐藏. 如果涉及过渡动画,定位的 top 和 left 必须写 Math.sin(弧度) 一圈弧度 = 2π,一圈角度 = 360 弧度 = (deg*2 ...

  10. linux学习:wget与lynx用法整理

    指令:wget.lynx.axel wget url  #下载数据写入文件,下载的文件名与url中的文件名保持一致,下载信息或进度写入stdoutwget url1 url2 url3    #下载多 ...