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

6 齐次坐标模块(homogen.rs)

homogen.rs代码定义了一个名为 HomogeneousVector 的结构体,它是用于表示三维空间中的齐次向量。齐次向量常用于计算机图形学和几何学中,特别是在处理投影和变换时。下面是对这段代码的详细解释和一些关键的代码片段分析:

一、homogen.rs文件源码

use crate::point::{Point2D, Point3D};
use crate::vector::{Vector2D, Vector3D};use crate::num::{One, Zero};#[cfg(feature = "bytemuck")]
use bytemuck::{Pod, Zeroable};
use core::cmp::{Eq, PartialEq};
use core::fmt;
use core::hash::Hash;
use core::marker::PhantomData;
use core::ops::Div;
#[cfg(feature = "serde")]
use serde;/// Homogeneous vector in 3D space.
#[repr(C)]
pub struct HomogeneousVector<T, U> {pub x: T,pub y: T,pub z: T,pub w: T,#[doc(hidden)]pub _unit: PhantomData<U>,
}impl<T: Copy, U> Copy for HomogeneousVector<T, U> {}impl<T: Clone, U> Clone for HomogeneousVector<T, U> {fn clone(&self) -> Self {HomogeneousVector {x: self.x.clone(),y: self.y.clone(),z: self.z.clone(),w: self.w.clone(),_unit: PhantomData,}}
}#[cfg(feature = "serde")]
impl<'de, T, U> serde::Deserialize<'de> for HomogeneousVector<T, U>
whereT: serde::Deserialize<'de>,
{fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>whereD: serde::Deserializer<'de>,{let (x, y, z, w) = serde::Deserialize::deserialize(deserializer)?;Ok(HomogeneousVector {x,y,z,w,_unit: PhantomData,})}
}#[cfg(feature = "serde")]
impl<T, U> serde::Serialize for HomogeneousVector<T, U>
whereT: serde::Serialize,
{fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>whereS: serde::Serializer,{(&self.x, &self.y, &self.z, &self.w).serialize(serializer)}
}#[cfg(feature = "arbitrary")]
impl<'a, T, U> arbitrary::Arbitrary<'a> for HomogeneousVector<T, U>
whereT: arbitrary::Arbitrary<'a>,
{fn arbitrary(u: &mut arbitrary::Unstructured<'a>) -> arbitrary::Result<Self> {let (x, y, z, w) = arbitrary::Arbitrary::arbitrary(u)?;Ok(HomogeneousVector {x,y,z,w,_unit: PhantomData,})}
}#[cfg(feature = "bytemuck")]
unsafe impl<T: Zeroable, U> Zeroable for HomogeneousVector<T, U> {}#[cfg(feature = "bytemuck")]
unsafe impl<T: Pod, U: 'static> Pod for HomogeneousVector<T, U> {}impl<T, U> Eq for HomogeneousVector<T, U> where T: Eq {}impl<T, U> PartialEq for HomogeneousVector<T, U>
whereT: PartialEq,
{fn eq(&self, other: &Self) -> bool {self.x == other.x && self.y == other.y && self.z == other.z && self.w == other.w}
}impl<T, U> Hash for HomogeneousVector<T, U>
whereT: Hash,
{fn hash<H: core::hash::Hasher>(&self, h: &mut H) {self.x.hash(h);self.y.hash(h);self.z.hash(h);self.w.hash(h);}
}impl<T, U> HomogeneousVector<T, U> {/// Constructor taking scalar values directly.#[inline]pub const fn new(x: T, y: T, z: T, w: T) -> Self {HomogeneousVector {x,y,z,w,_unit: PhantomData,}}
}impl<T: Copy + Div<T, Output = T> + Zero + PartialOrd, U> HomogeneousVector<T, U> {/// Convert into Cartesian 2D point.////// Returns `None` if the point is on or behind the W=0 hemisphere.#[inline]pub fn to_point2d(self) -> Option<Point2D<T, U>> {if self.w > T::zero() {Some(Point2D::new(self.x / self.w, self.y / self.w))} else {None}}/// Convert into Cartesian 3D point.////// Returns `None` if the point is on or behind the W=0 hemisphere.#[inline]pub fn to_point3d(self) -> Option<Point3D<T, U>> {if self.w > T::zero() {Some(Point3D::new(self.x / self.w,self.y / self.w,self.z / self.w,))} else {None}}
}impl<T: Zero, U> From<Vector2D<T, U>> for HomogeneousVector<T, U> {#[inline]fn from(v: Vector2D<T, U>) -> Self {HomogeneousVector::new(v.x, v.y, T::zero(), T::zero())}
}impl<T: Zero, U> From<Vector3D<T, U>> for HomogeneousVector<T, U> {#[inline]fn from(v: Vector3D<T, U>) -> Self {HomogeneousVector::new(v.x, v.y, v.z, T::zero())}
}impl<T: Zero + One, U> From<Point2D<T, U>> for HomogeneousVector<T, U> {#[inline]fn from(p: Point2D<T, U>) -> Self {HomogeneousVector::new(p.x, p.y, T::zero(), T::one())}
}impl<T: One, U> From<Point3D<T, U>> for HomogeneousVector<T, U> {#[inline]fn from(p: Point3D<T, U>) -> Self {HomogeneousVector::new(p.x, p.y, p.z, T::one())}
}impl<T: fmt::Debug, U> fmt::Debug for HomogeneousVector<T, U> {fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {f.debug_tuple("").field(&self.x).field(&self.y).field(&self.z).field(&self.w).finish()}
}#[cfg(test)]
mod homogeneous {use super::HomogeneousVector;use crate::default::{Point2D, Point3D};#[test]fn roundtrip() {assert_eq!(Some(Point2D::new(1.0, 2.0)),HomogeneousVector::from(Point2D::new(1.0, 2.0)).to_point2d());assert_eq!(Some(Point3D::new(1.0, -2.0, 0.1)),HomogeneousVector::from(Point3D::new(1.0, -2.0, 0.1)).to_point3d());}#[test]fn negative() {assert_eq!(None,HomogeneousVector::<f32, ()>::new(1.0, 2.0, 3.0, 0.0).to_point2d());assert_eq!(None,HomogeneousVector::<f32, ()>::new(1.0, -2.0, -3.0, -2.0).to_point3d());}
}

二、结构体定义

#[repr(C)]
pub struct HomogeneousVector<T, U> {pub x: T,pub y: T,pub z: T,pub w: T,#[doc(hidden)]pub _unit: PhantomData<U>,
}
  • #[repr©] 属性确保了结构体在内存中的布局是连续的,这对于与C语言接口或者特定的内存对齐需求很有用。
  • T 是向量的坐标类型,它可以是任何数值类型,比如 f32 或 f64。
  • U 是一个类型参数,通过 PhantomData 被引入,通常用于表示一些与结构体相关的额外信息,比如单位或维度,但不占用实际的内存空间。
  • _unit 字段被标记为文档隐藏,意味着在生成的文档中不会显示这个字段。

三、实现特性

  • Copy 和 Clone:由于 HomogeneousVector 持有的是泛型 T,只有当 T 实现 Copy 或 Clone 时,HomogeneousVector 才能相应地实现 Copy 或 Clone。
  • serde 序列化/反序列化:当启用了 serde 功能时,HomogeneousVector 可以被序列化和反序列化,前提是它的类型参数 T 也支持 serde。
  • From 实现:提供了从 Vector2D、Vector3D、Point2D 和 Point3D 到 HomogeneousVector 的转换方法。这些转换方法将源数据转换为齐次坐标形式,例如,二维点转换为齐次坐标时,w 分量被设置为 1,而 z 分量(对于2D点来说不存在)被设置为 0。

四、示例代码片段分析

从 Vector2D 转换到 HomogeneousVector

impl<T: Zero, U> From<Vector2D<T, U>> for HomogeneousVector<T, U> {#[inline]fn from(v: Vector2D<T, U>) -> Self {HomogeneousVector::new(v.x, v.y, T::zero(), T::zero())}
}

这里假设 Vector2D 是一个二维向量结构体,T::zero() 返回类型 T 的零值。此实现将二维向量的 x 和 y 分量转换为齐次向量的前两个分量,并将 z 和 w 分量都设置为零。

调试输出

impl<T: fmt::Debug, U> fmt::Debug for HomogeneousVector<T, U> {fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {f.debug_tuple("").field(&self.x).field(&self.y).field(&self.z).field(&self.w).finish()}
}

当 T 实现了 fmt::Debug 时,HomogeneousVector 也可以被格式化输出,这通常用于调试目的。

五、总结

homogen.rs代码展示了如何在 Rust 中定义和实现一个泛型结构体,以及如何利用 Rust 的类型系统和特性(如 PhantomData、条件编译 #[cfg()]、特性标志等)来增强代码的功能性和灵活性。HomogeneousVector 的设计使其能够灵活地与不同的数值类型和额外的类型信息一起工作,同时支持序列化和从其他几何类型转换。

相关文章:

6 齐次坐标模块(homogen.rs)

homogen.rs代码定义了一个名为 HomogeneousVector 的结构体&#xff0c;它是用于表示三维空间中的齐次向量。齐次向量常用于计算机图形学和几何学中&#xff0c;特别是在处理投影和变换时。下面是对这段代码的详细解释和一些关键的代码片段分析&#xff1a; 一、homogen.rs文件…...

007 JSON Web Token

文章目录 https://doc.hutool.cn/pages/jwt/#jwt%E4%BB%8B%E7%BB%8D JWT是一种用于双方之间安全传输信息的简洁的、URL安全的令牌标准。这个标准由互联网工程任务组(IETF)发表&#xff0c;定义了一种紧凑且自包含的方式&#xff0c;用于在各方之间作为JSON对象安全地传输信息。…...

Cursor 背后的技术栈:从 VS Code 到 AI 集成

引言 在当今快速发展的软件开发领域&#xff0c;开发者工具正在经历一场由人工智能&#xff08;AI&#xff09;驱动的革命。Cursor 作为一款新兴的智能编程助手&#xff0c;凭借其强大的 AI 能力和高效的开发体验&#xff0c;迅速吸引了大量开发者的关注。Cursor 不仅继承了 V…...

【Python蓝桥杯备赛宝典】

文章目录 一、基础数据结构1.1 链表1.2 队列1.3 栈1.4 二叉树1.5 堆二、基本算法2.1 算法复杂度2.2 尺取法2.3 二分法2.4 三分法2.5 倍增法和ST算法2.6 前缀和与差分2.7 离散化2.8 排序与排列2.9 分治法2.10贪心法1.接水时间最短问题2.糖果数量有限问题3.分发时间最短问题4.采摘…...

前端js高级25.1.30

原型&#xff1a;函数的组成结构 通过这个图我们需要知道。 假设我们创建了一个Foo函数。 规则&#xff1a;Function.protoType是函数显示原型。__proto__是隐式对象。 Function、Object、Foo函数的__proto__指向了Function.protoType说明。这三个都依托function函数来创建。…...

【后端开发】字节跳动青训营之性能分析工具pprof

性能分析工具pprof 一、测试程序介绍二、pprof工具安装与使用2.1 pprof工具安装2.2 pprof工具使用 资料链接&#xff1a; 项目代码链接实验指南pprof使用指南 一、测试程序介绍 package mainimport ("log""net/http"_ "net/http/pprof" // 自…...

云原生(五十二) | DataGrip软件使用

文章目录 DataGrip软件使用 一、DataGrip基本使用 二、软件界面介绍 三、附件文件夹到项目中 四、DataGrip设置 五、SQL执行快捷键 DataGrip软件使用 一、DataGrip基本使用 1. 软件界面介绍 2. 附加文件夹到项目中【重要】 3. DataGrip配置 快捷键使用&#xff1a;C…...

FreeRTOS学习 --- 任务调度

开启任务调度器 作用&#xff1a;用于启动任务调度器&#xff0c;任务调度器启动后&#xff0c; FreeRTOS 便会开始进行任务调度 该函数内部实现&#xff0c;如下&#xff1a; 1、创建空闲任务&#xff08;优先级最低&#xff09; 2、如果使能软件定时器&#xff0c;则创建定…...

2025年人工智能技术:Prompt与Agent的发展趋势与机遇

文章目录 一、Prompt与Agent的定义与区别(一)定义(二)区别二、2025年Prompt与Agent的应用场景(一)Prompt的应用场景(二)Agent的应用场景三、2025年Prompt与Agent的适合群体(一)Prompt适合的群体(二)Agent适合的群体四、2025年Prompt与Agent的发展机遇(一)Prompt的…...

区块链 智能合约安全 | 回滚攻击

视频教程在我主页简介和专栏里 目录&#xff1a; 智能合约安全 回滚攻击 总结 智能合约安全 回滚攻击 回滚攻击的本质是”耍赖” 举一个简单的例子,两个人玩石头剪刀布,输了的给对方10块钱,现在A输了,A说这把不算,重来 放在Solidity中,require()函数会检测其中的条件是否满…...

【Go语言圣经】第六节:方法

第六章&#xff1a;方法 6.1 方法声明 在函数声明时&#xff0c;在其名字之前放上一个变量&#xff0c;这就是声明了变量对应类型的一个方法&#xff0c;相当于为这种类型定义了一个独占的方法。 下例为 Point 类型声明了计算两个点之间距离的方法&#xff1a; package mai…...

【JavaEE进阶】图书管理系统 - 壹

目录 &#x1f332;序言 &#x1f334;前端代码的引入 &#x1f38b;约定前后端交互接口 &#x1f6a9;接口定义 &#x1f343;后端服务器代码实现 &#x1f6a9;登录接口 &#x1f6a9;图书列表接口 &#x1f384;前端代码实现 &#x1f6a9;登录页面 &#x1f6a9;…...

TensorFlow 简单的二分类神经网络的训练和应用流程

展示了一个简单的二分类神经网络的训练和应用流程。主要步骤包括&#xff1a; 1. 数据准备与预处理 2. 构建模型 3. 编译模型 4. 训练模型 5. 评估模型 6. 模型应用与部署 加载和应用已训练的模型 1. 数据准备与预处理 在本例中&#xff0c;数据准备是通过两个 Numpy 数…...

docker安装Redis:docker离线安装Redis、docker在线安装Redis、Redis镜像下载、Redis配置、Redis命令

一、镜像下载 1、在线下载 在一台能连外网的linux上执行docker镜像拉取命令 docker pull redis:7.4.0 2、离线包下载 两种方式&#xff1a; 方式一&#xff1a; -&#xff09;在一台能连外网的linux上安装docker执行第一步的命令下载镜像 -&#xff09;导出 # 导出镜像…...

Retrieval-Augmented Generation for Large Language Models: A Survey——(1)Overview

Retrieval-Augmented Generation for Large Language Models: A Survey——(1)Overview 文章目录 Retrieval-Augmented Generation for Large Language Models: A Survey——(1)Overview1. Introduction&Abstract1. LLM面临的问题2. RAG核心三要素3. RAG taxonomy 2. Overv…...

LabVIEW透镜多参数自动检测系统

在现代制造业中&#xff0c;提升产品质量检测的自动化水平是提高生产效率和准确性的关键。本文介绍了一个基于LabVIEW的透镜多参数自动检测系统&#xff0c;该系统能够在单一工位上完成透镜的多项质量参数检测&#xff0c;并实现透镜的自动搬运与分选&#xff0c;极大地提升了检…...

什么是Maxscript?为什么要学习Maxscript?

MAXScript是Autodesk 3ds Max的内置脚本语言,它是一种与3dsMax对话并使3dsMax执行某些操作的编程语言。它是一种脚本语言,这意味着您不需要编译代码即可运行。通过使用一系列基于文本的命令而不是使用UI操作,您可以完成许多使用UI操作无法完成的任务。 Maxscript是一种专有…...

Redis|前言

文章目录 什么是 Redis&#xff1f;Redis 主流功能与应用 什么是 Redis&#xff1f; Redis&#xff0c;Remote Dictionary Server&#xff08;远程字典服务器&#xff09;。Redis 是完全开源的&#xff0c;使用 ANSIC 语言编写&#xff0c;遵守 BSD 协议&#xff0c;是一个高性…...

128周二复盘(164)学习任天堂

1.设计相关 研究历史上某些武器数值&#xff0c;对一些设定进行参数修改。兼顾真实性&#xff0c;合理性&#xff0c;娱乐性。 学习宫本茂游戏思想&#xff0c;简单有趣-重玩性&#xff0c;风格化个性化-反拟真。对堆难度与内容的反思。 后续将学习岩田聪以及别的任天堂名人的…...

LeetCode:63. 不同路径 II

跟着carl学算法&#xff0c;本系列博客仅做个人记录&#xff0c;建议大家都去看carl本人的博客&#xff0c;写的真的很好的&#xff01; 代码随想录 LeetCode&#xff1a;63. 不同路径 II 给定一个 m x n 的整数数组 grid。一个机器人初始位于 左上角&#xff08;即 grid[0][0]…...

VUE之组件通信(一)

1、props 概述&#xff1a;props是使用频率最高的一种通信方式&#xff0c;常用与&#xff1a;父<——>子。 若 父传子&#xff1a;属性值是非函数。若 子传父&#xff1a;属性值是函数。 父组件&#xff1a; <template><div class"father">&l…...

ESP32-S3模组上跑通esp32-camera(41)

接前一篇文章:ESP32-S3模组上跑通esp32-camera(40) 一、OV5640初始化 2. 相机初始化及图像传感器配置 上一回继续对reset函数的后一段代码进行解析。为了便于理解和回顾,再次贴出reset函数源码,在components\esp32-camera\sensors\ov5640.c中,如下: static int reset…...

本地部署DeepSeek R1:打造专属私人AI助手指南

在当今人工智能蓬勃发展的浪潮中&#xff0c;DeepSeek R1模型的本地部署为用户带来了全新的体验。它不仅能够保障数据隐私&#xff0c;还具备与商业AI模型相媲美的出色性能。随着计算能力的不断提升以及开源AI社区的日益壮大&#xff0c;用户如今可以在本地运行高性能AI模型&am…...

Redis-布隆过滤器

文章目录 布隆过滤器的特点:实践布隆过滤器应用 布隆过滤器的特点: 就可以把布隆过滤器理解为一个set集合&#xff0c;我们可以通过add往里面添加元素&#xff0c;通过contains来判断是否包含某个元素。 布隆过滤器是一个很长的二进制向量和一系列随机映射函数。 可以用来检索…...

OpenCV 版本不兼容导致的问题

问题和解决方案 今天运行如下代码&#xff0c;发生了意外的错误&#xff0c;代码如下&#xff0c;其中输入的 frame 来自于 OpenCV 开启数据流的读取 """ cap cv2.VideoCapture(RTSP_URL) print("链接视频流完成") while True:ret, frame cap.rea…...

【视频+图文详解】HTML基础3-html常用标签

图文教程 html常用标签 常用标签 1. 文档结构 <!DOCTYPE html>&#xff1a;声明HTML文档类型。<html>&#xff1a;定义HTML文档的根元素。<head>&#xff1a;定义文档头部&#xff0c;包含元数据。<title>&#xff1a;设置网页标题&#xff0c;浏览…...

【B站保姆级视频教程:Jetson配置YOLOv11环境(五)Miniconda安装与配置】

Jetson配置YOLOv11环境&#xff08;5&#xff09;Miniconda安装与配置 文章目录 0. Anaconda vs Miniconda in Jetson1. 下载Miniconda32. 安装Miniconda33. 换源3.1 conda 换源3.2 pip 换源 4. 创建环境5. 设置默认启动环境 0. Anaconda vs Miniconda in Jetson Jetson 设备资…...

【PLL】杂散生成和调制

时钟生成 --》 数字系统 --》峰值抖动频率生成 --》无线系统 --》 频谱纯度、 周期信号的相位不确定性 随机抖动&#xff08;random jitter, RJ&#xff09;确定性抖动&#xff08;deterministic jitter,DJ&#xff09; 时域频域随机抖动积分相位噪声确定性抖动边带 杂散生成和…...

游戏引擎 Unity - Unity 启动(下载 Unity Editor、生成 Unity Personal Edition 许可证)

Unity Unity 首次发布于 2005 年&#xff0c;属于 Unity Technologies Unity 使用的开发技术有&#xff1a;C# Unity 的适用平台&#xff1a;PC、主机、移动设备、VR / AR、Web 等 Unity 的适用领域&#xff1a;开发中等画质中小型项目 Unity 适合初学者或需要快速上手的开…...

侯捷 C++ 课程学习笔记:深入理解 C++ 核心技术与实战应用

目录 引言 第一章&#xff1a;C 基础回顾 1.1 C 的历史与发展 1.2 C 的核心特性 1.3 C 的编译与执行 第二章&#xff1a;面向对象编程 2.1 类与对象 2.2 构造函数与析构函数 2.3 继承与多态 第三章&#xff1a;泛型编程与模板 3.1 函数模板 3.2 类模板 3.3 STL 容器…...