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

【图形学】TA之路-矩阵应用平移-旋转-大小

矩阵应用:在 Unity 中,Transform 和矩阵之间的关系非常密切。Transform 组件主要用于描述和控制一个物体在三维空间中的位置、旋转和缩放,而这些操作背后实际上都是通过矩阵来实现的

1. Transform 组件与矩阵的关系

Transform 组件包含以下三个核心属性:

  • Position:物体的位置。
  • Rotation:物体的旋转(通常以四元数或欧拉角表示)。
  • Scale:物体的缩放。

Unity 在计算一个物体在世界空间中的最终位置、旋转和缩放时,会将这些属性组合成一个 4x4 的变换矩阵

2. 变换矩阵 (Transformation Matrix)

变换矩阵是一个 4x4 矩阵,它可以表示一个物体在三维空间中的所有变换(平移、旋转、缩放)。这个矩阵通常被称为世界矩阵(World Matrix)或模型矩阵(Model Matrix)。

在 Unity 中,这个矩阵由 TransformlocalToWorldMatrix 属性表示,用于将物体的本地坐标转换到世界坐标系中。

3. Unity 中矩阵的运算过程

当 Unity 计算一个物体在场景中的最终位置时,会依次应用以下矩阵变换:

  • 缩放矩阵:描述物体的缩放变换。
  • 旋转矩阵:描述物体的旋转变换。
  • 平移矩阵:描述物体的位置变换。

4. Transform 矩阵的实际应用

当 Unity 需要将一个物体的本地坐标转换为世界坐标时,它会使用 localToWorldMatrix 进行转换。相反,如果需要将世界坐标转换为物体的本地坐标,Unity 会使用 worldToLocalMatrix

原理解析 

当一个物体的位置、旋转、缩放分别为 {0, 0, 0}{90, 90, 90}{1, 1, 1} 时,它的变换矩阵代表了如何将物体从其本地坐标系变换到世界坐标系。

1. 平移矩阵(Translation Matrix)

位置为 {0, 0, 0},意味着物体位于世界坐标系的原点(没有平移)。因此,平移矩阵是一个单位矩阵:

平移公式
 

2. 旋转矩阵(Rotation Matrix)

旋转为 {90°, 90°, 90°},意味着物体在每个轴上都旋转了 90 度。这个旋转由三个矩阵(分别绕 X 轴、Y 轴、Z 轴的旋转)组合而成。

  • 绕 X 轴旋转 90 度:

     

X轴旋转公式 

 

  • 绕 Y 轴旋转 90 度:

     

Y轴旋转公式  

 

  • 绕 Z 轴旋转 90 度:

     

 Z轴旋转公式  

  

最终的旋转矩阵是这些矩阵的乘积:

Rotation Matrix=Rotation Matrix (X)×Rotation Matrix (Y)×Rotation Matrix (Z)


3. 缩放矩阵(Scale Matrix)
缩放为 {1, 1, 1},意味着物体在所有方向上都没有缩放。缩放矩阵也是单位矩阵

缩放公式 

 

4. 组合变换矩阵

最终的变换矩阵是平移、旋转和缩放矩阵的组合。顺序为缩放 -> 旋转 -> 平移,因此组合矩阵为:

Transformation Matrix=Translation Matrix×Rotation Matrix×Scale Matrix

源码示例(个人写的,基本是这原理吧)

using System;
using System.Collections;
using System.Collections.Generic;
using System.Text;
using UnityEngine;
namespace Ethan
{public class Matrix4x4{private float[,] elements;// 构造函数public Matrix4x4(){elements = new float[4, 4];}// 用于初始化的构造函数public Matrix4x4(float[,] elements){this.elements = elements;}// 单位矩阵public static Matrix4x4 Identity(){return new Matrix4x4(new float[,]{{1, 0, 0, 0},{0, 1, 0, 0},{0, 0, 1, 0},{0, 0, 0, 1}});}// 零矩阵public static Matrix4x4 Zero(){return new Matrix4x4(new float[4, 4]);}// 矩阵乘法public static Matrix4x4 operator *(Matrix4x4 a, Matrix4x4 b){Matrix4x4 result = Zero();for (int i = 0; i < 4; i++){for (int j = 0; j < 4; j++){result.elements[i, j] = 0;for (int k = 0; k < 4; k++){result.elements[i, j] += a.elements[i, k] * b.elements[k, j];}}}return result;}// 平移矩阵public static Matrix4x4 Translate(float x, float y, float z){return new Matrix4x4(new float[,]{{1, 0, 0, x},{0, 1, 0, y},{0, 0, 1, z},{0, 0, 0, 1}});}// 绕X轴旋转矩阵public static Matrix4x4 RotateX(float angle){float rad = Mathf.Deg2Rad * angle;return new Matrix4x4(new float[,]{{1, 0, 0, 0},{0, Mathf.Cos(rad), -Mathf.Sin(rad), 0},{0, Mathf.Sin(rad), Mathf.Cos(rad), 0},{0, 0, 0, 1}});}// 绕Y轴旋转矩阵public static Matrix4x4 RotateY(float angle){float rad = Mathf.Deg2Rad * angle;return new Matrix4x4(new float[,]{{Mathf.Cos(rad), 0, Mathf.Sin(rad), 0},{0, 1, 0, 0},{-Mathf.Sin(rad), 0, Mathf.Cos(rad), 0},{0, 0, 0, 1}});}// 绕Z轴旋转矩阵public static Matrix4x4 RotateZ(float angle){float rad = Mathf.Deg2Rad * angle;return new Matrix4x4(new float[,]{{Mathf.Cos(rad), -Mathf.Sin(rad), 0, 0},{Mathf.Sin(rad), Mathf.Cos(rad), 0, 0},{0, 0, 1, 0},{0, 0, 0, 1}});}// 缩放矩阵public static Matrix4x4 Scale(float x, float y, float z){return new Matrix4x4(new float[,]{{x, 0, 0, 0},{0, y, 0, 0},{0, 0, z, 0},{0, 0, 0, 1}});}// 矩阵与向量相乘public static Vector3 MultiplyPoint(Matrix4x4 m, Vector3 point){float x = m.elements[0, 0] * point.x + m.elements[0, 1] * point.y + m.elements[0, 2] * point.z + m.elements[0, 3];float y = m.elements[1, 0] * point.x + m.elements[1, 1] * point.y + m.elements[1, 2] * point.z + m.elements[1, 3];float z = m.elements[2, 0] * point.x + m.elements[2, 1] * point.y + m.elements[2, 2] * point.z + m.elements[2, 3];return new Vector3(x, y, z);}// 显示矩阵内容public override string ToString(){string result = "";for (int i = 0; i < 4; i++){for (int j = 0; j < 4; j++){result += elements[i, j] + "\t";}result += "\n";}return result;}}
}

调用示例
 

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Vector3 = Ethan.Vector3;
using Matrix4x4 = Ethan.Matrix4x4;
public class Game : MonoBehaviour
{void Start(){// 创建变换矩阵Matrix4x4 rotation = Matrix4x4.RotateY(45); // 绕Y轴旋转45度Matrix4x4 translation = Matrix4x4.Translate(10, 0, 5); // 平移Matrix4x4 scale = Matrix4x4.Scale(2, 2, 2); // 缩放// 组合变换Matrix4x4 transformation = translation * rotation * scale;// 应用到一个点Vector3 originalPoint = new Vector3(1, 1, 1);Vector3 transformedPoint = Matrix4x4.MultiplyPoint(transformation, originalPoint);// 输出结果Debug.Log("初始点: " + originalPoint);Debug.Log("转换点: " + transformedPoint);Debug.Log("变换矩阵:\n" + transformation);}}

相关文章:

【图形学】TA之路-矩阵应用平移-旋转-大小

矩阵应用&#xff1a;在 Unity 中&#xff0c;Transform 和矩阵之间的关系非常密切。Transform 组件主要用于描述和控制一个物体在三维空间中的位置、旋转和缩放&#xff0c;而这些操作背后实际上都是通过矩阵来实现的 1. Transform 组件与矩阵的关系 Transform 组件包含以下…...

Spring 循环依赖解决方案

文章目录 1. 循环依赖的产生2. 循环依赖的解决模型3. 基于setter/Autowired 的循环依赖1_编写测试代码2_初始化 Cat3_初始化 Person4_ 回到 Cat 的创建流程5_小结 4. 基于构造方法的循环依赖5. 基于原型 Bean 的循环依赖6. 引人AOP的额外设计7. 总结 IOC 容器初始化bean对象的逻…...

可视化大屏:如何get到领导心目中的“科技感”?

你如果问领导可视化大屏需要什么风格的&#xff0c;领导大概率说科技感的&#xff0c;然后你就去做了&#xff0c;结果被劈了一顿&#xff0c;什么原因&#xff1f;因为你没有get到领导心目中描述的科技感。 一、为什么都喜欢科技感 科技感在可视化大屏设计中具有以下好处&am…...

基于Python的金融数据采集与分析的设计与实现

基于Python的金融数据采集与分析的设计与实现 “Design and Implementation of Financial Data Collection and Analysis based on Python” 完整下载链接:基于Python的金融数据采集与分析的设计与实现 文章目录 基于Python的金融数据采集与分析的设计与实现摘要第一章 绪论1…...

使用Sanic和SSE实现实时股票行情推送

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐&#xff1a;「storm…...

redis散列若干记录

字典 redis本身使用字典结构管理数据 redis使用hash表实现字典结构 使用了什么hash算法 使用SipHash算法&#xff0c;该算法能有效防止Hash表碰撞&#xff0c;并有不错的性能 hash冲突怎么解决 使用链表法解决hash冲突 hash表如何扩容 渐进式扩容&#xff0c;不会引起线程长期阻…...

Java面试八股之什么是STOMP协议

什么是STOMP协议 STOMP&#xff08;Simple Text Oriented Messaging Protocol&#xff09;是一种为消息队列和事件驱动架构设计的轻量级协议&#xff0c;主要用于在消息中间件之间进行消息交换。它的设计原则是简单、跨平台和易于实现&#xff0c;这使得STOMP成为许多实时应用…...

【自用】Python爬虫学习(一):爬虫基础与四个简单案例

Python爬虫学习&#xff08;一&#xff09; 基础知识四个简单的爬虫案列1.使用urlopen获取百度首页并保存2.获取某翻译单词翻译候选结果3.获取某网页中的书名与价格4.获取某瓣排名前250的电影名称 基础知识 对于一个网页&#xff0c;浏览器右键可以查看页面源代码&#xff0c;…...

[python]uiautomation.WindowControl函数用法

Python UIAutomation 窗口控件 介绍 在本文中&#xff0c;我们将探讨Python UIAutomation库以及如何使用它来控制和自动化Windows应用程序。我们将介绍UIAutomation的基础知识及其功能&#xff0c;并提供代码示例来演示其用法。 什么是UI自动化&#xff1f; UIAutomation是一个…...

学习记录第二十七天

进程 wait函数 功能 等待子进程结束&#xff1a;父进程调用wait函数后&#xff0c;会暂停执行&#xff0c;直到它的某个子进程结束。收集子进程状态&#xff1a;当子进程结束时&#xff0c;wait函数会返回子进程的终止状态&#xff0c;包括是正常终止还是被信号终止等信息。…...

servlet的执行顺序

执行的时候Tomcat先初始化 然后调用 server 根据server来回调请求方式下面会追入源码解释 package com.haogu.servlet;import javax.servlet.ServletConfig; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.…...

Go语言 类封装和绑定方法

本篇文章主要内容为Go语言类相关操作&#xff1a;封装和绑定方法介绍及示例。 目录 封装 绑定方法 类方法形参 指针形参 设置类方法参数 指针与非指针区别 总结 封装 go语言支持类的操作&#xff0c;但是没有class关键字&#xff0c;使用struct来模拟类。 示例如下&am…...

DirectShow过滤器开发-写WAV音频文件过滤器

下载本过滤器DLL 本过滤器将PCM音频流&#xff0c;或ADPCM&#xff0c;IEEE_FLOAT&#xff0c;ALAW&#xff0c;MULAW&#xff0c;GSM610音频流写入WAV音频文件。 写WAV音频文件过滤器信息 过滤器名称&#xff1a;写WAV 过滤器GUID&#xff1a;{CF704A9C-0C67-4712-BA33-DD0A…...

php根据截止时间计算剩余的时间,并且在剩余时间不足1天时仅显示小时数

//获取政策库文章public function getIndexZckList(){$fl_id = input(fl_id);if(empty(...

Docker最佳实践进阶(一):Dockerfile介绍使用

大家好,上一个系列我们使用docker安装了一系列的基础服务,但在实际开发过程中这样一个个的安装以及繁杂命令不仅仅浪费时间,更是容易遗忘,下面我们进行Docker的进阶教程,帮助我们更快速的部署和演示项目。 一、什么是Dockerfile? Dockerfile 是一个文本文件,其中包含了…...

Anything in Any Scene:无缝融入任何场景,实现逼真视频对象插入技术

人工智能咨询培训老师叶梓 转载标明出处 现实世界的视频捕获虽然因其真实性而宝贵&#xff0c;但常常受限于长尾分布的问题&#xff0c;即常见场景过度呈现&#xff0c;而关键的罕见场景却鲜有记录。这导致了所谓的"分布外问题"&#xff0c;在模拟复杂环境光线、几何…...

安卓开发中的AppCompat框架|安卓系统|安卓应用|兼容性|UI组件|核心组件|ActionBar|Fragment|最佳实践|框架|移动开发|移动应用

目录 1. 什么是AppCompat框架 1.1 AppCompat的起源 1.2 AppCompat的重要性 2. AppCompat框架的核心组件 2.1 AppCompatActivity 2.2 AppCompat主题 2.3 AppCompat Widgets 3. 在项目中使用AppCompat框架 3.1 添加依赖项 3.2 应用AppCompat主题 4. AppCompat的高级功…...

React使用useRef ts 报错

最近在写自己的React项目&#xff0c;我在使用useRef钩子函数的时候发现 TS2322: Type MutableRefObject<HTMLDivElement | undefined> is not assignable to type LegacyRef<HTMLDivElement> | undefined Type MutableRefObject<HTMLDivElement | undefined&g…...

python-信息交互-pyautogui

python-信息交互-pyautogui 一: pyautogui1> waht?2> 功能分类3> 概念及作用二: 通用功能1> function all2> function 注释三: 鼠标控制1> mouse functions2> mouse functions demo3> mouse drag demo四: keyboard控制1> keyboard functions2> …...

flink1.18 编译遇到的问题

1. flink-runtime-web编译失败 源码编译时一直卡在 [INFO] Running ‘npm ci --cache-max0 --no-save’ in 处理方法&#xff1a; 修改flink-runtime-web/pom.xml文件 将<arguments>ci --cache-max0 --no-save ${npm.proxy}</arguments> 替换为&#xff1a;<a…...

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站&#xff0c;会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后&#xff0c;网站没有变化的情况。 不熟悉siteground主机的新手&#xff0c;遇到这个问题&#xff0c;就很抓狂&#xff0c;明明是哪都没操作错误&#x…...

Xshell远程连接Kali(默认 | 私钥)Note版

前言:xshell远程连接&#xff0c;私钥连接和常规默认连接 任务一 开启ssh服务 service ssh status //查看ssh服务状态 service ssh start //开启ssh服务 update-rc.d ssh enable //开启自启动ssh服务 任务二 修改配置文件 vi /etc/ssh/ssh_config //第一…...

Admin.Net中的消息通信SignalR解释

定义集线器接口 IOnlineUserHub public interface IOnlineUserHub {/// 在线用户列表Task OnlineUserList(OnlineUserList context);/// 强制下线Task ForceOffline(object context);/// 发布站内消息Task PublicNotice(SysNotice context);/// 接收消息Task ReceiveMessage(…...

Java如何权衡是使用无序的数组还是有序的数组

在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...

STM32F4基本定时器使用和原理详解

STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...

sqlserver 根据指定字符 解析拼接字符串

DECLARE LotNo NVARCHAR(50)A,B,C DECLARE xml XML ( SELECT <x> REPLACE(LotNo, ,, </x><x>) </x> ) DECLARE ErrorCode NVARCHAR(50) -- 提取 XML 中的值 SELECT value x.value(., VARCHAR(MAX))…...

成都鼎讯硬核科技!雷达目标与干扰模拟器,以卓越性能制胜电磁频谱战

在现代战争中&#xff0c;电磁频谱已成为继陆、海、空、天之后的 “第五维战场”&#xff0c;雷达作为电磁频谱领域的关键装备&#xff0c;其干扰与抗干扰能力的较量&#xff0c;直接影响着战争的胜负走向。由成都鼎讯科技匠心打造的雷达目标与干扰模拟器&#xff0c;凭借数字射…...

中医有效性探讨

文章目录 西医是如何发展到以生物化学为药理基础的现代医学&#xff1f;传统医学奠基期&#xff08;远古 - 17 世纪&#xff09;近代医学转型期&#xff08;17 世纪 - 19 世纪末&#xff09;​现代医学成熟期&#xff08;20世纪至今&#xff09; 中医的源远流长和一脉相承远古至…...

MFC 抛体运动模拟:常见问题解决与界面美化

在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...

基于IDIG-GAN的小样本电机轴承故障诊断

目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) ​梯度归一化(Gradient Normalization)​​ (2) ​判别器梯度间隙正则化(Discriminator Gradient Gap Regularization)​​ (3) ​自注意力机制(Self-Attention)​​ 3. 完整损失函数 二…...