当前位置: 首页 > news >正文

Python实现图形学曲线和曲面的Bezier曲线算法

目录

    • 使用Python实现图形学曲线和曲面的Bezier曲线算法
      • 引言
      • Bezier曲线的数学原理
        • 1. Bezier曲线定义
        • 2. Bezier曲线的递归形式
      • Python实现Bezier曲线算法
        • 1. 代码实现
      • 代码详解
      • 使用示例
      • Bezier曲线的特点
      • Bezier曲面的扩展
        • Bezier曲面类实现
      • 总结

使用Python实现图形学曲线和曲面的Bezier曲线算法

引言

在计算机图形学中,Bezier曲线(贝塞尔曲线)是绘制平滑曲线的常用工具,广泛应用于计算机绘图、动画、字体设计、图形设计和CAD系统中。Bezier曲线由法国工程师Pierre Bézier在1960年代发明,最常用于表示光滑的二次或三次曲线。通过几个控制点,Bezier曲线能够构建出非常平滑的曲线。

本文将详细介绍Bezier曲线的数学原理,并通过Python的面向对象编程思想实现该算法,绘制曲线和曲面。

Bezier曲线的数学原理

1. Bezier曲线定义

Bezier曲线是由一组控制点定义的平滑曲线。在二维空间中,给定 n + 1 个控制点 P 0 , P 1 , . . . , P n P_0, P_1, ..., P_n P0,P1,...,Pn,我们可以用下面的公式来表示一条 n 阶 Bezier曲线:

B ( t ) = ∑ i = 0 n ( n i ) ( 1 − t ) n − i t i P i B(t) = \sum_{i=0}^{n} \binom{n}{i} (1-t)^{n-i} t^i P_i B(t)=i=0n(in)(1t)nitiPi

其中:

  • B ( t ) B(t) B(t) 是曲线上的点,参数 ( t ) 的范围为 [0, 1]。
  • ( n i ) \binom{n}{i} (in) 是组合数,表示二项式系数。
  • P i P_i Pi 是控制点,定义了曲线的形状。

对于常见的情况:

  • 二次Bezier曲线有 3 个控制点 P 0 , P 1 , P 2 P_0, P_1, P_2 P0,P1,P2
  • 三次Bezier曲线有 4 个控制点 P 0 , P 1 , P 2 , P 3 P_0, P_1, P_2, P_3 P0,P1,P2,P3
2. Bezier曲线的递归形式

Bezier曲线的另一个常见实现方法是递归求解,称为 de Casteljau算法。该算法的思想是通过线性插值逐步逼近曲线上的点。假设有控制点 P 0 , P 1 , . . . , P n P_0, P_1, ..., P_n P0,P1,...,Pn,计算过程如下:

  1. 对每对相邻的控制点 P i P_i Pi P i + 1 P_{i+1} Pi+1,进行线性插值,计算出新的点 P i ′ P'_i Pi
    P i ′ ( t ) = ( 1 − t ) P i + t P i + 1 P'_i(t) = (1-t)P_i + tP_{i+1} Pi(t)=(1t)Pi+tPi+1
  2. 重复这一过程,直到只剩下一个点,即为曲线在 t t t 处的点。

Python实现Bezier曲线算法

我们将实现如下几个类:

  • Point2D:表示一个二维平面上的点。
  • BezierCurve:用于计算和绘制Bezier曲线的类。
  • BezierSurface:用于计算和绘制Bezier曲面的类。
1. 代码实现
import numpy as np# 定义二维点类
class Point2D:def __init__(self, x, y):self.x = xself.y = ydef __repr__(self):return f"({self.x}, {self.y})"# 定义Bezier曲线类
class BezierCurve:def __init__(self, control_points):"""初始化Bezier曲线:param control_points: 控制点的列表,每个控制点是一个 Point2D 对象"""self.control_points = control_pointsdef calculate_point(self, t):"""使用de Casteljau算法计算Bezier曲线在参数t处的点:param t: 曲线参数 t, 范围为 [0, 1]:return: 返回曲线在 t 处的 Point2D 点"""points = self.control_points.copy()n = len(points) - 1for k in range(1, n + 1):for i in range(n - k + 1):# 使用线性插值计算points[i].x = (1 - t) * points[i].x + t * points[i + 1].xpoints[i].y = (1 - t) * points[i].y + t * points[i + 1].yreturn points[0]def generate_curve_points(self, num_points=100):"""生成Bezier曲线上的点:param num_points: 生成的曲线上点的数量:return: 返回点列表,表示Bezier曲线"""curve_points = []for i in np.linspace(0, 1, num_points):curve_points.append(self.calculate_point(i))return curve_points# 使用示例
if __name__ == "__main__":# 定义控制点control_points = [Point2D(0, 0), Point2D(1, 2), Point2D(3, 3), Point2D(4, 0)]# 创建Bezier曲线对象bezier_curve = BezierCurve(control_points)# 生成并输出曲线上的点curve_points = bezier_curve.generate_curve_points()print("Bezier曲线上的点:")for point in curve_points:print(point)

代码详解

  1. Point2D 类:表示二维平面上的一个点,包含点的 (x) 和 (y) 坐标。

  2. BezierCurve 类:这个类负责计算和生成Bezier曲线。它主要实现了以下功能:

    • calculate_point(t):使用 de Casteljau算法 递归计算Bezier曲线在参数 ( t ) 处的点。该算法通过不断插值计算中间控制点,直到只剩下一个点,即为曲线在 ( t ) 处的位置。
    • generate_curve_points(num_points):生成并返回Bezier曲线上的若干个点,这些点均匀分布在 ( t ) 的范围 [0, 1] 内,用于表示曲线的整体形状。
  3. 递归计算过程:在 calculate_point(t) 方法中,控制点之间进行线性插值,不断缩小点的数量,直到得到最终的曲线点。

使用示例

在使用示例中,我们定义了一条由4个控制点组成的三次Bezier曲线,起点为 (0, 0),控制点分别为 (1, 2)(3, 3),终点为 (4, 0)。通过生成曲线上的100个点,我们可以近似出这条曲线的形状。

输出曲线上的点坐标:

Bezier曲线上的点:
(0.0, 0.0)
(0.11816792066666665, 0.22376795333333333)
...
(3.8818320793333333, 0.22376795333333333)
(4.0, 0.0)

Bezier曲线的特点

  • 平滑性:Bezier曲线通过控制点的线性插值构造,具有非常平滑的曲线形状。
  • 简单性:通过少量控制点即可定义复杂的曲线。常用的二次和三次Bezier曲线分别由3个和4个控制点组成。
  • 灵活性:Bezier曲线不仅可以表示简单的曲线,还能表示复杂的路径。控制点越多,曲线越复杂。

Bezier曲面的扩展

Bezier曲线不仅可以用于绘制平面曲线,还可以扩展到三维曲面。Bezier曲面是由多个控制点定义的,可以通过类似的递归插值计算生成。

Bezier曲面类实现
# 定义Bezier曲面类
class BezierSurface:def __init__(self, control_points_grid):"""初始化Bezier曲面:param control_points_grid: 控制点的二维网格,每个点是Point2D对象"""self.control_points_grid = control_points_griddef calculate_point(self, u, v):"""计算Bezier曲面在参数(u, v)处的点:param u: 曲面参数 u, 范围为 [0, 1]:param v: 曲面参数 v, 范围为 [0, 1]:return: 返回曲面在 (u, v) 处的 Point2D 点"""# 计算每行的Bezier曲线点curve_points_u = [BezierCurve(row).calculate_point(u) for row in self.control_points_grid]# 对这些点再使用Bezier曲线进行插值return BezierCurve(curve_points_u).calculate_point(v)def generate_surface_points(self, num_points_u=10, num_points_v=10):"""生成Bezier曲面上的点:param num_points_u: u方向点的数量:param num_points_v: v方向点的数量:return: 返回二维点列表,表示Bezier曲面"""surface_points = []for u in np.linspace(0, 1, num_points_u):row = []for v in np.linspace(0, 1, num_points_v):row.append(self.calculate_point(u, v))surface_points.append(row)return surface_points

总结

Bezier曲线在计算机图形学中有着广泛的应用,它能通过少量的控制点生成平滑且复杂的曲线。本文介绍了Bezier曲线的数学原理,并用Python面向对象的方法实现了该算法。同时,我们还扩展到了Bezier曲面,使得该算法可以用于更复杂的三维图形建模。

通过掌握Bezier曲线的算法,读者可以在各种绘图工具中生成平滑的曲线,并进一步探索曲面的生成。

相关文章:

Python实现图形学曲线和曲面的Bezier曲线算法

目录 使用Python实现图形学曲线和曲面的Bezier曲线算法引言Bezier曲线的数学原理1. Bezier曲线定义2. Bezier曲线的递归形式 Python实现Bezier曲线算法1. 代码实现 代码详解使用示例Bezier曲线的特点Bezier曲面的扩展Bezier曲面类实现 总结 使用Python实现图形学曲线和曲面的Be…...

Unity数据持久化4——2进制

概述 基础知识 各类型数据转字节数据 文件操作相关 文件相关 文件流相关 文件夹相关 练习题 using System; using System.Collections; using System.Collections.Generic; using System.IO; using System.Text; using UnityEngine;public class Exercises1 : MonoBehaviour {/…...

经典sql题(八)SQL 查询详细指南总结一

SQL 查询详细指南 SQL(Structured Query Language)是一种用于管理和操作关系数据库的标准语言。本文将详细介绍 SQL 中的一些常见操作及其用法,包括 DISTINCT 去重、LIMIT 限制、排序、开窗函数、NULL 值替换、JOIN 与 UNION 等。 1. DISTI…...

用Python实现时间序列模型实战——Day 30: 学习总结与未来规划

在第30天,我们将对整个学习过程进行总结,复习关键知识点,并展望未来的学习与应用方向。我们将涵盖时间序列分析过程中涉及的主要模型、技术和工具,总结它们的优势和应用场景。此外,规划未来如何进一步深入学习&#xf…...

ChatGPT居然主动勾引用户,OpenAI又测试新功能? 一文教你学会订阅

有网友表示,他收到了ChatGPT主动给他发送的消息,询问“你高中的第一周过得怎么样?还适应吗?” 他很懵逼的回了一句“你刚才是给我发消息吗?”。也就是说,在没有任何先前文本提示下,ChatGPT主动…...

基于SpringBoot+Vue的考研百科网站系统

作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、SSM项目源码 精品专栏:Java精选实战项目源码、Python精…...

深度之眼(三十)——pytorch(一)--深入浅出pytorch(附安装流程)

文章目录 一、前言一、pytoch二、六个部分三、如何学习四、学习路径(重要)五、安装pytorch5.1 坑15.2 坑2 一、前言 我看了下目录 第一章和第二章都是本科学的数字图像处理。 也就是这一专栏:数字图像实验。 所以就不准备学习前两章了,直接…...

麒麟银河桌面版,成功安装cuda12.6,mysql

一、 要卸载并禁用 nouveau 驱动程序,可以按照以下步骤进行: 1. 确认 nouveau 驱动的当前状态: 首先,你可以使用以下命令查看 nouveau 驱动是否正在运行: lsmod | grep nouveau如果有输出,说明 nouveau …...

CentOS 7 YUM源不可用

CentOS 7 操作系统在2024年6月30日后将停止官方维护,并且官方提供的YUM源将不再可用。 修改:nano /etc/yum.repos.d/CentOS-Base.repo # CentOS-Base.repo [base] nameCentOS-$releasever - Base baseurlhttp://mirrors.aliyun.com/centos/$rel…...

Java反序列化利用链篇 | URLDNS链

文章目录 URLDNS链调用链分析Payload编写 系列篇其他文章,推荐顺序观看~ Java反序列化利用链篇 | JdbcRowSetImpl利用链分析Java反序列化利用链篇 | CC1链_全网最菜的分析思路Java反序列化利用链篇 | CC1链的第二种方式-LazyMap版调用链Java反序列化利用链篇 | URLD…...

Android 短信验证码自动填充

本文主要介绍国外google上线的app 短信自动填充方案。 本方案主要是使用google提出的,防止开发者使用SMS相关权限造成的用户信息泄露 目录 注意点: 1、本方式不适合华为手机,华为有自己的获取方式 2、本方式不需要添加任何短信权限 3、…...

数据库 MySQL 是否需要容器化?

容器的定义:容器是为了解决“在切换运行环境时,如何保证软件能够正常运行”这一问题。 目前,容器和 Docker 依旧是技术领域最热门的词语,无状态的服务容器化已经是大势所趋,同时也带来了一个热点问题被大家所争论不以…...

Kettle的安装及简单使用

Kettle的安装及简单使用一、kettle概述二、kettle安装部署和使用Windows下安装案例1:MySQL to MySQL案例2:使用作业执行上述转换,并且额外在表stu2中添加一条数据案例3:将hive表的数据输出到hdfs案例4:读取hdfs文件并将…...

Golang | Leetcode Golang题解之第420题强密码检验器

题目: 题解: func strongPasswordChecker(password string) int {hasLower, hasUpper, hasDigit : 0, 0, 0for _, ch : range password {if unicode.IsLower(ch) {hasLower 1} else if unicode.IsUpper(ch) {hasUpper 1} else if unicode.IsDigit(ch)…...

面试金典题3

URL化。编写一种方法,将字符串中的空格全部替换为%20。假定该字符串尾部有足够的空间存放新增字符,并且知道字符串的“真实”长度。 示例 1: 输入:"Mr John Smith ", 13 输出:"Mr%20John%20Smith&…...

FFmpeg开发笔记(五十六)使用Media3的Exoplayer播放网络视频

Android早期的MediaPlayer控件对于网络视频的兼容性很差,所以后来单独推出了Exoplayer库增强支持网络视频,在《Android Studio开发实战:从零基础到App上线(第3版)》一书第14章的“14.3.3 新型播放器ExoPlayer”就详细介绍了Exoplayer库的详细…...

Python使用总结之py-docx将word文件中的图片保存,并将内容返回

Python使用总结之py-docx将word文件中的图片保存,并将内容返回 使用py-docx读取word文档的内容,其中包含标题、文本和图片等信息。该方法将标题和内容返回,并将文件中的图片保存到指定的文件夹中。 实现步骤 加载文件内容读取文件的段落对文…...

Radware 报告 Web DDoS 攻击活动

新一代 HTTPS 洪水攻击的频率和强度急剧增加,攻击者引入的复杂程度也在迅速提高。2024 年上半年,Web 分布式拒绝服务 (DDoS) 攻击的频率和强度显著增加。其中很大一部分活动可以归因于受政治紧张局势驱使的黑客活动分子。 众所周知,当今的黑…...

OpenCV运动分析和目标跟踪(2)累积操作函数accumulateSquare()的使用

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 将源图像的平方加到累积器图像中。 该函数将输入图像 src 或其选定区域提升到2的幂次方,然后加到累积器 dst 中: dst ( …...

PCIe进阶之TL:Common Packet Header Fields TLPs with Data Payloads Rules

1 Transaction Layer Protocol - Packet Definition TLP有四种事务类型:Memory、I/O、Configuration 和 Messages,两种地址格式:32bit 和 64bit。 构成 TLP 时,所有标记为 Reserved 的字段(有时缩写为 R)都必须全为0。接收者Rx必须忽略此字段中的值,PCIe Switch 必须对…...

.Net框架,除了EF还有很多很多......

文章目录 1. 引言2. Dapper2.1 概述与设计原理2.2 核心功能与代码示例基本查询多映射查询存储过程调用 2.3 性能优化原理2.4 适用场景 3. NHibernate3.1 概述与架构设计3.2 映射配置示例Fluent映射XML映射 3.3 查询示例HQL查询Criteria APILINQ提供程序 3.4 高级特性3.5 适用场…...

关于nvm与node.js

1 安装nvm 安装过程中手动修改 nvm的安装路径, 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解,但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后,通常在该文件中会出现以下配置&…...

Leetcode 3577. Count the Number of Computer Unlocking Permutations

Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接:3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯,要想要能够将所有的电脑解锁&#x…...

Robots.txt 文件

什么是robots.txt? robots.txt 是一个位于网站根目录下的文本文件(如:https://example.com/robots.txt),它用于指导网络爬虫(如搜索引擎的蜘蛛程序)如何抓取该网站的内容。这个文件遵循 Robots…...

OpenLayers 分屏对比(地图联动)

注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 地图分屏对比在WebGIS开发中是很常见的功能,和卷帘图层不一样的是,分屏对比是在各个地图中添加相同或者不同的图层进行对比查看。…...

【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统

目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...

项目部署到Linux上时遇到的错误(Redis,MySQL,无法正确连接,地址占用问题)

Redis无法正确连接 在运行jar包时出现了这样的错误 查询得知问题核心在于Redis连接失败,具体原因是客户端发送了密码认证请求,但Redis服务器未设置密码 1.为Redis设置密码(匹配客户端配置) 步骤: 1).修…...

c++第七天 继承与派生2

这一篇文章主要内容是 派生类构造函数与析构函数 在派生类中重写基类成员 以及多继承 第一部分:派生类构造函数与析构函数 当创建一个派生类对象时,基类成员是如何初始化的? 1.当派生类对象创建的时候,基类成员的初始化顺序 …...

MySQL:分区的基本使用

目录 一、什么是分区二、有什么作用三、分类四、创建分区五、删除分区 一、什么是分区 MySQL 分区(Partitioning)是一种将单张表的数据逻辑上拆分成多个物理部分的技术。这些物理部分(分区)可以独立存储、管理和优化,…...

aardio 自动识别验证码输入

技术尝试 上周在发学习日志时有网友提议“在网页上识别验证码”,于是尝试整合图像识别与网页自动化技术,完成了这套模拟登录流程。核心思路是:截图验证码→OCR识别→自动填充表单→提交并验证结果。 代码在这里 import soImage; import we…...