Python 实现图形学几何变换算法

news2024/9/22 4:38:15

目录

    • Python 实现图形学几何变换算法
      • 几何变换介绍
      • 变换矩阵
      • Python 实现几何变换
      • 代码解释
      • 总结

Python 实现图形学几何变换算法

在计算机图形学中,几何变换是非常重要的概念。它们允许我们对对象的位置、大小、方向进行操作,比如平移、缩放、旋转、反射等。本文将详细介绍这些几何变换,并使用 Python 结合面向对象编程 (OOP) 的思想进行实现。

几何变换介绍

常见的几何变换包括:

  1. 平移 (Translation):在平面中移动对象。
  2. 缩放 (Scaling):根据某个比例改变对象的大小。
  3. 旋转 (Rotation):围绕某个点对对象进行旋转。
  4. 反射 (Reflection):沿某条轴对对象进行对称反射。
  5. 错切 (Shear):改变对象的形状但保持其体积不变。

这些几何变换都可以通过矩阵运算来描述,并应用于2D图形中的点和向量。

变换矩阵

在二维平面上,变换矩阵可以表示为 3x3 的齐次坐标矩阵。一般的二维变换可以使用下列矩阵表示:

  • 平移矩阵
    T = [ 1 0 t x 0 1 t y 0 0 1 ] T = \begin{bmatrix} 1 & 0 & t_x \\ 0 & 1 & t_y \\ 0 & 0 & 1 \end{bmatrix} T= 100010txty1
    其中, t x t_x tx t y t_y ty 分别是沿 x 和 y 方向的平移距离。

  • 缩放矩阵
    S = [ s x 0 0 0 s y 0 0 0 1 ] S = \begin{bmatrix} s_x & 0 & 0 \\ 0 & s_y & 0 \\ 0 & 0 & 1 \end{bmatrix} S= sx000sy0001
    其中, s x s_x sx s y s_y sy 分别是 x 和 y 方向的缩放因子。

  • 旋转矩阵(绕原点逆时针旋转):
    R = [ cos ⁡ θ − sin ⁡ θ 0 sin ⁡ θ cos ⁡ θ 0 0 0 1 ] R = \begin{bmatrix} \cos\theta & -\sin\theta & 0 \\ \sin\theta & \cos\theta & 0 \\ 0 & 0 & 1 \end{bmatrix} R= cosθsinθ0sinθcosθ0001
    其中, θ \theta θ 是旋转角度。

  • 反射矩阵(关于 x 轴或 y 轴):

    • 关于 x 轴反射:
      F x = [ 1 0 0 0 − 1 0 0 0 1 ] F_x = \begin{bmatrix} 1 & 0 & 0 \\ 0 & -1 & 0 \\ 0 & 0 & 1 \end{bmatrix} Fx= 100010001
    • 关于 y 轴反射:
      F y = [ − 1 0 0 0 1 0 0 0 1 ] F_y = \begin{bmatrix} -1 & 0 & 0 \\ 0 & 1 & 0 \\ 0 & 0 & 1 \end{bmatrix} Fy= 100010001
  • 错切矩阵
    H = [ 1 k x 0 k y 1 0 0 0 1 ] H = \begin{bmatrix} 1 & k_x & 0 \\ k_y & 1 & 0 \\ 0 & 0 & 1 \end{bmatrix} H= 1ky0kx10001
    其中, k x k_x kx k y k_y ky 分别是 x 和 y 方向的错切因子。

Python 实现几何变换

我们将使用 Python 的面向对象编程思想来实现几何变换类,这样可以更加灵活和可扩展。

import numpy as np

# 定义二维点类
class Point2D:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def to_homogeneous(self):
        """将点转化为齐次坐标形式"""
        return np.array([self.x, self.y, 1])

    def update_from_homogeneous(self, coords):
        """从齐次坐标更新点的坐标"""
        self.x, self.y = coords[0], coords[1]

# 定义几何变换基类
class Transformation2D:
    def __init__(self):
        self.matrix = np.eye(3)  # 初始化为单位矩阵

    def apply(self, point):
        """将变换应用到一个点上"""
        homogeneous_point = point.to_homogeneous()
        transformed_point = self.matrix.dot(homogeneous_point)
        point.update_from_homogeneous(transformed_point)

# 定义平移变换类
class Translation(Transformation2D):
    def __init__(self, tx, ty):
        super().__init__()
        self.matrix = np.array([
            [1, 0, tx],
            [0, 1, ty],
            [0, 0, 1]
        ])

# 定义缩放变换类
class Scaling(Transformation2D):
    def __init__(self, sx, sy):
        super().__init__()
        self.matrix = np.array([
            [sx, 0, 0],
            [0, sy, 0],
            [0, 0, 1]
        ])

# 定义旋转变换类
class Rotation(Transformation2D):
    def __init__(self, theta):
        super().__init__()
        cos_theta = np.cos(np.radians(theta))
        sin_theta = np.sin(np.radians(theta))
        self.matrix = np.array([
            [cos_theta, -sin_theta, 0],
            [sin_theta, cos_theta, 0],
            [0, 0, 1]
        ])

# 定义反射变换类
class Reflection(Transformation2D):
    def __init__(self, axis):
        super().__init__()
        if axis == 'x':
            self.matrix = np.array([
                [1, 0, 0],
                [0, -1, 0],
                [0, 0, 1]
            ])
        elif axis == 'y':
            self.matrix = np.array([
                [-1, 0, 0],
                [0, 1, 0],
                [0, 0, 1]
            ])

# 定义错切变换类
class Shear(Transformation2D):
    def __init__(self, kx, ky):
        super().__init__()
        self.matrix = np.array([
            [1, kx, 0],
            [ky, 1, 0],
            [0, 0, 1]
        ])

# 使用示例
if __name__ == "__main__":
    # 创建一个二维点
    point = Point2D(1, 1)
    
    # 平移变换
    translation = Translation(2, 3)
    translation.apply(point)
    print(f"After translation: ({point.x}, {point.y})")
    
    # 缩放变换
    scaling = Scaling(2, 2)
    scaling.apply(point)
    print(f"After scaling: ({point.x}, {point.y})")
    
    # 旋转变换
    rotation = Rotation(90)
    rotation.apply(point)
    print(f"After rotation: ({point.x}, {point.y})")
    
    # 反射变换
    reflection = Reflection('x')
    reflection.apply(point)
    print(f"After reflection: ({point.x}, {point.y})")
    
    # 错切变换
    shear = Shear(1, 0)
    shear.apply(point)
    print(f"After shear: ({point.x}, {point.y})")

代码解释

  1. Point2D类:表示一个二维点。该类提供了将点转化为齐次坐标形式的方法,并能从变换后的齐次坐标更新点的坐标。

  2. Transformation2D基类:几何变换的基类,定义了变换矩阵(初始为单位矩阵),并且定义了 apply 方法,用于将变换矩阵作用到一个点上。

  3. 具体的几何变换类:如 TranslationScalingRotationReflectionShear 类,都继承自 Transformation2D,并分别实现自己的变换矩阵。

  4. 使用示例:创建一个二维点,并依次应用平移、缩放、旋转、反射、错切等变换。

总结

通过面向对象的思想实现几何变换,可以使得代码结构更加清晰且易于扩展。每种变换都可以作为一个独立的类来实现,同时也可以轻松添加新的变换类型。通过这种方式,我们能够灵活地对二维图形进行复杂的变换操作。

希望这篇文章能帮助你更好地理解几何变换及其在图形学中的应用。如果有任何问题或建议,欢迎交流!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2154106.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

IDEA中Quarkus框架(3.13版本)开发、调试、部署、打包等

code-with-quarkus code-with-quarkus 是使用官网生成的demo项目 这个项目使用Quarkus(使用3.13.0版本,该版本支持JDK21),超音速亚原子Java框架。官网地址: https://quarkus.io/. 环境要求 OS: Windows 10.0 jdk 11 maven 3.9…

淘宝扭蛋机小程序,扭蛋机文化下的新体验

在数字化时代中,扭蛋机逐渐从传统的线下机器转移到了线上互联网中,市场得到了创新发展。扭蛋机小程序具有便捷、多样化、个性化的特点,迎合了当下消费者的线上消费习惯,又能够让扭蛋机玩家体验到新鲜有趣的扭蛋。 扭蛋机是一种热…

python简单的小项目-关于央行储蓄占比情况的数据可视化

该数据来源于锐思数据库,如果数据有偏差,可能是本人搜索的问题,希望大家谅解。 数据大纲: 其中我们制作折现统计图需要用到的是截止日期,表达数据最后获取的日期,而更新时间则是数据时效性的表示&#xff…

django项目添加测试数据的三种方式

文章目录 自定义终端命令Faker添加模拟数据基于终端脚本来完成数据的添加编写python脚本编写shell脚本执行脚本需要权限使用shell命令来完成测试数据的添加 添加测试数据在工作中一共有三种方式: 可以根据django的manage.py指令进行[自定义终端命令]可以采用第三方…

pthread_cond_signal 和pthread_cond_wait

0、pthread_join()函数作用: pthread_join() 函数会一直阻塞调用它的线程,直至目标线程执行结束(接收到目标线程的返回值),阻塞状态才会解除。如果 pthread_join() 函数成功等到了目标线程执行结束(成功获取…

【C++】list详解及模拟实现

目录 1. list介绍 2. list使用 2.1 修改相关 2.2 遍历 2.3 构造 2.4 迭代器 2.5 容量相关 2.6 元素访问 2.7 操作相关 3. 模拟实现 3.1 节点类 3.1.1 初始结构 3.1.2 节点的构造函数 3.2 迭代器类 3.2.1 初始结构 3.2.2 迭代器 3.2.3 迭代器-- 3.2.4 解引…

1.随机事件与概率

第一章 随机时间与概率 1. 随机事件及其运算 1.1 随机现象 ​ 确定性现象:只有一个结果的现象 ​ 确定性现象:结果不止一个,且哪一个结果出现,人们事先并不知道 1.2 样本空间 ​ 样本空间:随机现象的一切可能基本…

ML 系列:机器学习和深度学习的深层次总结(05)非线性回归

图 1.不同类型的回归 一、说明 非线性回归是指因变量和自变量之间存在非线性关系的模型。该模型比线性模型更准确、更灵活,可以获取两个或多个变量之间复杂关系的各种曲线。 二、关于 当数据之间的关系无法用直线预测并且呈曲线形式时,我们应该使用非线性…

MySQL篇(索引)(持续更新迭代)

目录 一、简介 二、有无索引情况 1. 无索引情况 2. 有索引情况 3. 优劣势 三、索引结构 1. 简介 2. 存储引擎对于索引结构的支持情况 3. 为什么InnoDB默认的索引结构是Btree而不是其它树 3.1. 二叉树(BinaryTree) 3.2. 红黑树(RB&a…

6、等级保护政策内容

数据来源:6.等级保护政策内容_哔哩哔哩_bilibili 信息安全产品管理与响应 等级管理 对信息系统中使用的信息安全产品实行按等级管理,信息安全事件应分等级响应与处置。 预测评服务由测评公司和咨询公司提供预测评服务,根据技术要求和测评要…

高校心理辅导系统:Spring Boot技术实现指南

目 录 摘 要 I ABSTRACT II 1绪 论 1 1.1研究背景 1 1.2设计原则 1 1.3论文的组织结构 2 2 相关技术简介 3 2.1Java技术 3 2.2B/S结构 3 2.3MYSQL数据库 4 2.4Springboot框架 4 3 系统分析 6 3.1可行性分析 6 3.1.1技术可行性 6 3.1.2操作可行性 6 3.1.3经济可行性 6 3.1.4法律…

[OpenGL]使用OpenGL绘制带纹理三角形

一、简介 本文介绍了如何使用使用OpenGL绘制带纹理三角形。 在绘制带纹理的三角形时, 首先使用.h读取准备好的.png格式的图片作为纹理,然后在fragment shader中使用 ... in vec2 textureCoord; uniform sampler2D aTexture1; void main() {FragColor …

嵌入式 开发技巧和经验分享

文章目录 前言嵌入式 开发技巧和经验分享目录1.1嵌入式 系统的 定义1.2 嵌入式 操作系统的介绍1.3 嵌入式 开发环境1.4 编译工具链和优化1.5 嵌入式系统软件开发1.6 嵌入式SDK开发2.1选择移植的系统-FreeRtos2.2FreeRtos 移植步骤2.3 系统移植之中断处理2.4系统移植之内存管理2…

奥比中光深度相机相关使用内容

奥比中光深度相机相关使用内容 Windows平台测试官方软件关于python环境的配置1、安装CMake2、安装Visual Studio3、项目地址下载4、配置Visual Studio5、完成基于Python的SDK配置官网教学视频地址 3D视觉开发者社区 官方文档地址 效果: Windows平台测试官方软件 Window…

《高等代数》线性相关和线性无关(应用)

说明:此文章用于本人复习巩固,如果也能帮到大家那就更加有意义了。 注:1)线性相关和线性无关的证明方法中较为常用的方法是利用秩和定义来证明。 2)此外,线性相关和线性无关的证明常常也会用到反证法。 3&…

简单水印通过python去除

简单水印通过python去除 先看效果,如果效果不是你需要的就可以不用浪费时间。 注意:这种主要还是对应的文字在水印上方的情况,同时最好不要有渐变水印否则可能最后输出的图片的水印还会有所残留,不过还是学习使用,相信…

DOS(Disk Operating System,磁盘操作系统)常用指令

目录 背景: 早期探索: DOS之父: 发展历程: 常用指令: 进入命令: 操作1.进入和回退: 操作2.增、删: 操作3.其它: 总结: 背景: 早期探索: DOS(Disk Operating System,磁盘操作系统)在…

【Web】PolarCTF2024秋季个人挑战赛wp

EZ_Host 一眼丁真命令注入 payload: ?host127.0.0.1;catf* 序列一下 exp: <?phpclass Polar{public $lt;public $b; } $pnew Polar(); $p->lt"system"; $p->b"tac /f*"; echo serialize($p);payload: xO:5:"Polar":2:{s:2:"…

VSCode C++ Tasks.json基本信息介绍

前言 上文介绍了VSCode在Windows环境下如果创建C项目和编译多个文件项目&#xff0c;但是只是粗略的说明了一下Tasks.json文件。今天对tasks.json进行进一步的了解。 内容 Tasks文件 {"version": "2.0.0","tasks": [{"type": &quo…

C#进阶-基于雪花算法的订单号设计与实现

在现代电商系统和分布式系统中&#xff0c;高效地生成全局唯一的订单号是一个关键需求。订单号不仅需要唯一性&#xff0c;还需要具备一定的趋势递增性&#xff0c;以满足数据库索引和排序的需求。本文将介绍如何在C#中使用雪花算法&#xff08;Snowflake&#xff09;设计和实现…