Direct3D网格(一)

news2025/1/2 3:35:56

创建网格

我们可以用D3DXCreateMeshFVF函数创建一个"空"网格对象 ,空网格对象是指我们指定了网格的面片总数和顶点总数,然后由该函数为顶点缓存索引缓存属性缓存分配大小合适的内存,之后即可手工填入网格数据。

HRESULT WINAPI D3DXCreateMeshFVF(
	DWORD NumFaces,
	DWORD NumVertices,
	DWORD Options,
	DWORD FVF,
	LPDIRECT3DDEVICE9 pD3DDevice,
	LPD3DXMESH* ppMesh
);

NumFaces:网格将具有的面片总数,该值必须大于0

NumVertices:网格将具有的顶点总数,该值必须大于0

Options:创建网格时所使用的创建标记,枚举D3DXMESH,一些常用的标记如下
D3DXMESH_32BIT  网格将使用32位索引
D3DXMESH_MANAGED  网格数据将被存储于托管内存池中
D3DXMESH_WEITEONLY  指定网格数据为只读

FVF:存储在该网格中的顶点的灵活顶点格式

pDevice:设备指针

ppMesh:所创建的网格对象的指针

也可以用函数D3DXCreateMesh函数来创建空网格,在该函数中并未指定FVF,而是用一个D3DVERTEXELEMENT9类型的结构数组来描述顶点数据的布局方式。

HRESULT WINAPI D3DXCreateMesh(
	DWORD NumFaces,
	DWORD NumVertices,
	DWORD Options,
	CONST D3DVERTEXELEMENT9 *pDeclaration,
	LPDIRECT3DDEVICE9 pD3DDevice,
	LPD3DXMESH* ppMesh
);

HRESULT WINAPI D3DXDeclaratorFromFVF(
	DWORD FVF,
	D3DVERTEXELEMENT9 pDeclarator[MAX_FVF_DECL_SIZE]
);

ID3DXBaseMesh获取几何信息接口

//获取
HRESULT ID3DXMesh::GetVertexBuffer(LPDIRECT3DVERTEXBUFFER9* ppVB);			//获取顶点缓存(用于存储网格顶点)
HRESULT ID3DXMesh::GetIndexBuffer(LPDIRECT3DINDEXBUFFER9* ppIB);			//获得索引缓存(决定顶点应以何种组合方式构成网格的三角形单元)
//锁定缓存,进行读写操作,注意这些方法锁定的是整个顶点缓存或索引缓存
//Flags参数描述了如何进行锁定,函数返回是ppData参返回指向被锁定的内存的指针的地址
HRESULT ID3DXMesh::LockVertexBuffer(DWORD Flags, BYTE** ppData);
HRESULT ID3DXMesh::LockIndexBuffer(DWORD Flags, BYTE** ppData);
//解锁
HRESULT ID3DXMesh::UnlockVertexBuffer();
HRESULT ID3DXMesh::UnlockIndexBuffer();
//获取几何信息的其他接口
DWORD GetFVF();					//返回描述顶点格式的DWORD
DWORD GetNumVertices();			//返回顶点缓存中的顶点个数
DWORD GetNumBytesPerVertex();	//返回每个顶点所占的字节数
DWORD GetNumFaces();			//返回网格中(三角形)面片的个数
DWORD GetOptions();             //返回网格的标记组合

子集和属性缓存

一个网格由一个或多个子集组成,一个子集是网格中一组可用相同属性进行绘制的三角形单元,这里属性是指材质、纹理和绘制状态

为了区分不同子集,我们为每个子集指定一个唯一的非负整数值,该值可为DWORD类型所能容纳的任何非负整数,网格中的每个三角形单元都被赋予了一个属性ID,该ID指定了该三角形单元所属的子集

这些三角形单元的属性ID被存储在网格的属性缓存中,该属性缓存实际上是一个DWORD类型的数组,由于每个面片在属性缓存中都有对应项,所以属性缓存中元素的个数与网格中面片的个数完全相等,而且属性缓存中那些项与在索引缓存中定义的三角形单元是一一对应 的,即属性缓存中的第i项对应于索引缓存中的第i个三角形,三角形单元i是有索引缓存中如下3个索引定义的。

A=i\cdot 3      B=i\cdot 3+1     C=i\cdot 3+2

属性缓存的锁定与解锁
ID3DXMesh::LockAttributeBuffer(DWORD Flags, DWORD** ppData);
ID3DXMesh::UnlockAttributeBuffer();

绘制

ID3DXMesh接口提供了方法DrawSubset用于绘制三角形单元,参数AttribId为指定的某个子集,若要绘制整个网格,必须绘制该网格的所有子集,比较方便的方法为将各子集的属性ID依次指定为0,1,2...n-1,n为子集的总数,每个子集都有一个对应的材质和纹理数组,这样通过索引i就可找到对应的材质和纹理。

ID3DXMesh::DrawSubset(DWORD AttribId);

for (int i = 0; i < numSubsets; ++i)
{
	Device->SetMaterial(mtrls[i]);
	Device->SetTexture(0, textures[i]);
	Mesh->DrawSubset(i);
}

网格优化

为了更高效的绘制一个网格,我们可对该网格中的顶点和索引进行重组,这个重组的过程称为网格优化OptimizeInplace

HRESULT ID3DXMesh::OptimizeInplace(
	DWORD Flags,
	CONST DWORD* pAdjacencyIn,
	DWORD* pAdjacencyOut,
	DWORD* pFaceRemap,
	LPD3DXBUFFER *ppVertexRemap
);

Flags:优化选项标记,通知该方法所要实施的优化方案
D3DXMESHOPT_COMPACT  从网格中移除那些无用顶点和索引
D3DXMESHOPT_ATTSORT  依据属性对各三角形单元进行排序,并生成一个属性表,这样可使DrawSubset获得更高的绘制效率
D3DXMESHOPT_VERTEXCACHE  提高顶点高速缓存的命中率
D3DXMESHOPT_STRIPORDER  对索引进行重组,以使三角形单元条带尽可能的长
D3DXMESHOPT_IGNOREVERTS  仅对索引进行优化,忽略顶点
标记D3DXMESHOPT_VERTEXCACHE和D3DXMESHOPT_STRIPORDER不允许被被同时使用

pAdjacencyIn:指向未经优化的网格的邻接数组的指针

pAdjacencyOut:指向一个DWORD类型数组的指针,该数组被填充了经优化后的网格的邻接信息,该数组的维数必须为GetNumFaces()*3(每个三角形单元有3个边,所有邻接信息倍数为3倍),如果不需要该信息,可将参数赋为0

pFaceRemap:指向一个DWORD类型数组的指针,该数组填充了网格面片的重绘信息,该数组的维数应为GetNumFaces(),当对一个网格面实施优化后,其面片在索引缓存中可能发生了移动,该面片重绘信息表明了原始面片所被移动到的新位置,即pFaceRemap中的第i项保存了表示第i个原始面片被移动到哪里的面片索引,如果不需要该信息可将该参数赋为0

ppVertexRemap:指向ID3DXMesh对象指针的地址,该对象中保存了顶点的重绘信息,该缓存所包含的顶点数应为GetNumVertices(),当网格面经过优化之后,其顶点在索引缓存中的位置可能发生变动,顶点重绘信息表明了原始顶点移动到的新位置,即ppVertexRemap中的第i项保存了表示第i个原始顶点被移动到哪里的顶点索引,如果不需要该值,将该参数赋为0

DWORD adjacencyInfo[Mesh->GetNumFaces() * 3];
Mesh->GenerateAdjacency(0.0f, adjacencyInfo);
DWORD optimizedAdjacencyInfo[Mesh->GetNumFaces() * 3];
Mesh->OptimizeInplace(D3DXMESHOPT_ATTRSORT | D3DXMESHOPT_COMPACT | D3DXMESHOPT_VERTEXCACHE, adjacencyInfo, optimizedAdjacencyInfo, 0, 0);

OptimizeInplace功能类似的另一个方法是Optimize该方法将输出调用该方法的网格对象优化后的版本,但是网格对象本身不会发生改变

HRESULT ID3DXMesh::Optimize(
	DWORD Flags,
	CONST DWORD* pAdjacencyIn,
	DWORD* pAdjacencyOut,
	DWORD* pFaceRemap,
	LPD3DXBUFFER *ppVertexRemap,
	LPD3DXMESH* ppOptMesh
);

属性表

如果一个网格对象在优化处理时使用了D3DXMESHOPT_ATTRSORT标记,则构成该网格的三角形面片就会依据其属性进行排序,这样属于特定子集的三角形面片就会被保存在顶点缓存或索引缓存中的一个连续存储空间

除了可对面片进行排序外,D3DXMESHOPT_ATTRSORT优化选项还将创建一个属性表,该属性表是一个D3DXATTRIBUTERANGE类型的结构数组,属性表中每一项都对应于网格的一个子集,并指定了该子集中面片的几何信息被存储在顶点缓存或索引缓存的哪一个存储块中。

typedef struct _D3DXATTRIBUTERANGE
{
    DWORD AttribId;
    DWORD FaceStart;
    DWORD FaceCount;
    DWORD VertexStart;
    DWORD VertexCount;
} D3DXATTRIBUTERANGE;

AttribId:子集的Id
FaceStart:一个大小为FaceStart*3的偏移量,表明了该子集的三角形单元在索引缓存中的起始位置
FaceCount:该子集中面片(三角形单元)的总数
VertexStart:一个表明了与子集相关的顶点在顶点缓存中起始位置的偏移量
VertexCount:该子集中的顶点总数

属性表的访问与设置

可以使用GetAttributeTable方法来访问一个网格面的属性表,该方法完成了俩项工作:返回属性表中的属性个数和属性数据填充D3DXATTRIBUTERANGE类型的结构数组。要想获取属性表中的元素个数,可将方法第一个参数取为0

HRESULT ID3DXMesh::GetAttributeTable(
	D3DXATTRIBUTERANGE *pAttribTable,
	DWORD* pAttribTableSize
);

DWORD numSubsets=0;
Mesh->GetAttributeTable(0,&numSubsets);

//一旦得到了属性表中的元素个数,就可以用属性数据填充结构数组
D3DXATTRIBUTERANGE table = new D3DXATTRIBUTERANGE[numSubSets];
Mesh->GetAttributeTable(table,&numSubsets);

可以使用SetAttributeTable对属性表进行设置

D3DXATTRIBUTERANGE attributeTable[12];
//填充该数组结构....

//设定属性表共有12个子集
Mesh->SetAttributeTable(attributeTable,12);

邻接信息

对于某些网格运算(如网格优化),需要知道对任意给定三角形面片,哪些面片与其邻接,这些邻接信息都存储在网格的邻接数组中。邻接数组的类型为DWORD,其每一项都包含了一个标识网格中某个三角形面片的索引,如果邻接数组中某一项等于ULONG MAX=4294967295,则表明网格中某一特定边没有邻接面片,我们也可将该项赋为-1来表示此种情形。

邻接数组的维数必须为GetNumFaces()*3,网格中每个三角形面片都有3个可能的邻接面片,可用GenerateAdjacency函数来输出邻接信息。

HRESULT ID3DXMesh::GenerateAdjacency(
	FLOAT Epsilon,
	DWORD* pAdjacency
);

DWORD adjacencyInfo[Mesh->GetNumFaces()*3];
Mesh->GenerateAdjacency(0.001f,adjacencyInfo);

fEpsilon:一个很小的正数值,指定了在某种距离度量下,俩个点接近到何种程度方可认为这俩点为同一点,例如俩点间距离小于该值,认为俩点为同一点
pAdjacency:指向一个DWORD类型的数组的指针,该数组中存储了邻接信息

克隆

有时需要生成网格数据的一个副本可用CloneMeshFVF方式来实现,该方法允许目标网格采用与原网格不同的创建选项和灵活顶点格式。

HRESULT ID3DXMesh::CloneMeshFVF(
	DWORD Options,
	DWORD FVF, 
	LPDIRECT3DDEVICE9 pD3DDevice, 
	LPD3DXMESH* ppCloneMesh
);

ID3DXMesh* clone = 0;
Mesh->CloneMeshFVF(Mesh->GetOptions(), D3DFVF_XYZ | D3DFVF_NORMAL, Device, &clone);

Options:创建某网格副本时的创建标记或标记组合,枚举D3DXMESH
D3DXMESH_32BIT  网格将使用32位索引
D3DXMESH_MANAGED  网格数据将被存储于托管内存池中
D3DXMESH_WRITEONLY 指定网格数据为只读
D3DXMESH_DYNAMIC  网格缓存将使用动态内存

FVF:所要创建的克隆网格的灵活顶点格式

pDevice:设备指针

ppCloneMesh:输出所创建的克隆网格

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

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

相关文章

供水管网监测系统

随着城市人口的不断增长和经济的快速发展&#xff0c;供水管网的安全和可靠性变得尤为重要。在过去&#xff0c;供水管网的监测往往是依靠人工巡查&#xff0c;这种方式不仅费时费力&#xff0c;而且容易出现疏漏和盲区。然而&#xff0c;随着科技的进步&#xff0c;供水管网监…

【算法|动态规划No.16】leetcode931. 下降路径最小和

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 &#x1f354;本专栏旨在提高自己算法能力的同时&#xff0c;记录一下自己的学习过程&#xff0c;希望…

智能售后工单系统是什么?智能工单系统有什么用?

对于传统的客服问题处理机制中&#xff0c;如果使用智能工单详细记录客户的问题以及解决问题的全过程&#xff0c;可以有效地帮助客服中心或业务管理层从总体层面了解客户的问题&#xff0c;给出有效的解决方案以及处理问题的紧急程度。从每个服务分支到问题&#xff0c;基于大…

361154-30-5|点击糖化学试剂Ac4ManNAz

产品简介&#xff1a;N-叠氮乙酰基甘露糖胺-三酰化&#xff08;AC4MANAZ&#xff09;可用作标记试剂&#xff0c;点击糖化学试剂&#xff0c;叠氮化物基团允许它与炔烃反应&#xff0c;是一种含叠氮的代谢糖蛋白标记试剂&#xff0c;叠氮化物修饰的蛋白质可以通过与炔烃反应检测…

基于springboot实现自习室预订系统的设计与实现项目【项目源码+论文说明】分享

基于springboot实现自习室预订系统的设计与实现演示 摘要 在网络高速发展的时代&#xff0c;众多的软件被开发出来&#xff0c;给学生带来了很大的选择余地&#xff0c;而且人们越来越追求更个性的需求。在这种时代背景下&#xff0c;学院只能以学生为导向&#xff0c;所以自习…

软件工程师都应该知道的10个定律

一、海勒姆法则 内容 当一个 API 有足够多的用户&#xff0c;你在契约中承诺了什么并不重要&#xff1a;系统中所有看得见的行为都会有某个人依赖…… 案例 现在有两个系统A和B&#xff0c;B的一个接口返回一个列表。A系统的开发人员发现返回的列表都是按照ID正向排序的。本…

SSH版本信息可被获取

漏洞描述 Name SSH版本信息可被获取 Description SSH服务允许远程攻击者获得ssh的具体信息&#xff0c;如版本号等等。这可能为攻击者发动进一步攻击提供帮助。 CVE No. CVE-1999-0634 分析结果 该问题不属于漏洞&#xff0c;不存在安全风险。SSH协议是一种安全协议&am…

基于Springboot实现幼儿园管理系统项目【项目源码+论文说明】

基于Springboot实现幼儿园管理系统演示 摘要 随着科学技术的飞速发展&#xff0c;各行各业都在努力与现代先进技术接轨&#xff0c;通过科技手段提高自身的优势&#xff1b;对于幼儿园管理系统当然也不能排除在外&#xff0c;随着网络技术的不断成熟&#xff0c;带动了幼儿园管…

【PickerView案例13-应用程序对象介绍 Objective-C语言】

一、应用程序对象介绍: 1.应用程序对象介绍: 应用程序介绍: 应用程序介绍: 应用程序介绍: 应用程序启动,本身这一过程,不是应用程序启动就完事儿了, 它有一些比较细节的东西,比如说: 1)info.plist以及pch文件 2)UIApplication对象 这个呢,我们都是分开的去说,…

C语言进阶文件操作

本章重点 1. 为什么使用文件 2. 什么是文件 3. 文件的打开和关闭 4. 文件的顺序读写 5. 文件的随机读写 6. 文本文件和二进制文件 7. 文件读取结束的判定 8. 文件缓冲区 目录 1. 为什么使用文件 2. 什么是文件 2.1 程序文件 2.2 数据文件 2.3 文件名 3. 文…

分布式软件架构——服务端缓存的三种属性

服务端缓存 在透明多级分流系统中&#xff0c;我们以流量从客户端中发出开始&#xff0c;以流量到达服务器集群中真正处理业务的节点结束。一起探索了在这个过程中与业务无关的一些通用组件&#xff0c;包括DNS、CDN、客户端缓存&#xff0c;等等。 实际上&#xff0c;服务端缓…

Docker搭建Redis cluster集群

redis常用的三种集群方式是&#xff1a;主从复制&#xff0c;sentinel 哨兵模式&#xff0c;cluster模式&#xff0c;本节我想详细记录下cluster集群的动手部署方式。cluster是比较主流的方式&#xff0c;优缺点可以百度查下。 Redis-Cluster采用无中心结构&#xff0c;每个节…

Hadoop启动缺失ResourceManager

Hadoop启动缺失ResourceManager 查日志是resourcemanager空指针&#xff0c;java.lang.NullPointerException 1、进入根目录的tmp文件夹 cd /tmp2、删除hadoop-yarn-root rm -rf hadoop-yarn-root3、重启hadoop就好啦&#xff01;&#xff01;&#xff01;&#xff01;&…

三网运营商大数据精准营销客源信息提取的原理是什么?

现在的这个互联网时代&#xff0c;越来越多的人从事运营商大数据这个行业&#xff0c;相信大家都很好奇运营商大数据的绝对优势究竟是什么&#xff0c;它究竟是有什么魅力&#xff0c;能够让那么多的人为它所倾倒&#xff0c;下面我们就来看一下它究竟有什么优势吧。 1.对用户的…

解析硬件连通性测试的重要性及测试方法

在现代科技世界中&#xff0c;硬件设备的复杂性和多样性已经达到了前所未有的水平。无论是计算机、智能手机、物联网设备还是嵌入式系统&#xff0c;各种硬件组件的协同工作对于设备的正常运行至关重要。硬件连通性测试是确保这些组件相互配合无误的重要步骤。 一、硬件连通性测…

[nltk_data] Error loading stopwords: <urlopen error [WinError 10054]

报错提示&#xff1a; >>> import nltk >>> nltk.download(stopwords) 按照提示执行后 [nltk_data] Error loading stopwords: <urlopen error [WinError 10054] 找到路径C:\\Users\\EDY\\nltk_data&#xff0c;如果没有nltk_data文件夹&#xff0c;在…

嵌入式养成计划-30-网络编程----多点通信--单播--广播--组播

六十六、多点通信 66.1 网络属性相关函数 getsockoptsetsockopt 功能&#xff1a;获取/设置网络属性; 原型&#xff1a;#include <sys/types.h> /* See NOTES */#include <sys/socket.h>int getsockopt(int sockfd, int level, int optname, void *opt…

12大自媒体推广平台整理

1、百家号&#xff1a;权重很高&#xff0c;百度端排名效果极好&#xff0c;即便是行业核心词&#xff0c;也可以轻松进入前三页。 2、头条号&#xff1a;即时展示效果较好&#xff0c;文章质量越高&#xff0c;被推送的范围越广。 3、微信公众号&#xff1a;更适合做私域流量…

STM32实战项目——WIFI远程开关灯

前言 其实WIFI开关灯在几个月前就想做了&#xff0c;但是对于没有云平台调试经验的我&#xff0c;一开始有些摸不着头脑&#xff0c;所以就搁置了。十一假期与老同学聊天时了解到他也在做一个远程开关灯的小项目&#xff0c;所以就重新开始了WIFI远程开关灯的小项目。 本文使用…

你不会还在人工巡检UPS设备吧!?大神技巧学起来!

在现代生活和商业中&#xff0c;依赖电力已经成为不可或缺的一部分。因此&#xff0c;UPS监控成为了保障业务连续性和设备运行的关键要素。 UPS监控的目标是提供对电力系统的全面可视化和控制&#xff0c;以及在电力故障或其他问题发生时采取即时行动。 客户案例 企业数据中心…