Unity顶点优化:UV Splits与Smoothing Splits消除技巧
一、顶点分裂问题概述
1. 什么是顶点分裂
顶点分裂(Vertex Splits)是3D渲染中常见的性能问题,当模型需要为同一顶点位置存储不同属性值时,会创建多个顶点副本。主要分为两类:
-
UV Splits:由UV不连续引起
-
Smoothing Splits:由硬边/法线不连续引起
- 对惹,这里有一个游戏开发交流小组,希望大家可以点击进来一起交流一下开发经验呀
2. 性能影响
| 分裂类型 | 顶点数增幅 | 典型影响 |
|---|---|---|
| UV Splits | 2-5倍 | 增加网格数据量,降低批处理效率 |
| Smoothing Splits | 3-8倍 | 增加顶点着色器计算负载 |
二、诊断工具与技术
1. 内置诊断方法
// 获取网格顶点统计数据
void AnalyzeMesh(Mesh mesh) {Debug.Log($"原始顶点数: {mesh.vertexCount}");Debug.Log($"子网格数: {mesh.subMeshCount}");// 使用Mesh.GetOriginal...方法检测分裂Vector3[] origVertices = mesh.vertices;Vector3[] actualVertices = new Vector3[mesh.vertexCount];mesh.GetVertices(actualVertices);float splitRatio = (float)actualVertices.Length / origVertices.Length;Debug.Log($"顶点分裂比例: {splitRatio:0.0}x");
}
2. 专业工具推荐
-
Unity Profiler:分析渲染批次和顶点数
-
Mesh Inspector插件:可视化显示分裂位置
-
RenderDoc:捕获帧调试顶点数据
三、UV Splits消除技巧
1. UV布局优化原则
-
最小化UV岛数量:减少切割线
-
保持UV连续:避免UV坐标突变
-
合理利用UV空间:减少重叠
2. 自动UV优化脚本
using UnityEditor;public class UVOptimizer : AssetPostprocessor {void OnPreprocessModel() {ModelImporter importer = (ModelImporter)assetImporter;// UV优化设置importer.generateSecondaryUV = true;importer.secondaryUVAngleDistortion = 88;importer.secondaryUVAreaDistortion = 15;importer.secondaryUVHardAngle = 88;importer.secondaryUVPackMargin = 0.003f;}
}
3. 运行时UV重映射
// 顶点着色器中动态计算UV
v2f vert (appdata v) {v2f o;o.uv = v.uv;// 简单UV展开算法float2 sphereUV = float2(atan2(v.normal.z, v.normal.x) / (2.0 * PI) + 0.5,asin(v.normal.y) / PI + 0.5);// 根据需求混合UVo.uv = lerp(o.uv, sphereUV, _UVRemapFactor);return o;
}
四、Smoothing Splits消除技巧
1. 法线平滑技术
// 法线平滑算法
Vector3[] SmoothNormals(Mesh mesh) {Vector3[] vertices = mesh.vertices;Vector3[] normals = mesh.normals;Dictionary<Vector3, List<int>> vertexMap = new Dictionary<Vector3, List<int>>();// 建立顶点位置到索引的映射for(int i=0; i<vertices.Length; i++) {if(!vertexMap.ContainsKey(vertices[i])) {vertexMap[vertices[i]] = new List<int>();}vertexMap[vertices[i]].Add(i);}// 平滑法线foreach(var pair in vertexMap) {Vector3 avgNormal = Vector3.zero;foreach(int index in pair.Value) {avgNormal += normals[index];}avgNormal = avgNormal.normalized;foreach(int index in pair.Value) {normals[index] = avgNormal;}}return normals;
}
2. 硬边标记优化
// 使用顶点颜色标记硬边
v2f vert (appdata_full v) {v2f o;// 硬边检测阈值float edgeFactor = smoothstep(_HardEdgeThreshold-0.1, _HardEdgeThreshold+0.1, v.color.r);// 混合法线o.normal = lerp(v.normal, normalize(cross(ddx(v.vertex), ddy(v.vertex)), edgeFactor);return o;
}
五、高级优化策略
1. 顶点缓存优化
// 重新排序顶点缓存
void OptimizeVertexCache(Mesh mesh) {Mesh optimizedMesh = new Mesh();// 使用Unity内置优化optimizedMesh.vertices = mesh.vertices;optimizedMesh.triangles = mesh.triangles;optimizedMesh.Optimize();optimizedMesh.OptimizeIndexBuffers();optimizedMesh.OptimizeReorderVertexBuffer();// 计算优化率float optimizationRate = (float)mesh.vertexCount / optimizedMesh.vertexCount;Debug.Log($"顶点缓存优化率: {optimizationRate:0.0}x");
}
2. 顶点属性压缩
// 使用半精度存储顶点属性
struct appdata_compressed {float3 vertex : POSITION;half3 normal : NORMAL;half4 tangent : TANGENT;half2 uv : TEXCOORD0;
};
六、性能对比数据
| 优化技术 | 顶点数减少 | 帧率提升 | 适用场景 |
|---|---|---|---|
| UV布局优化 | 35-60% | 15-25% | 静态模型 |
| 法线平滑 | 40-70% | 20-30% | 有机模型 |
| 顶点缓存优化 | 10-20% | 5-15% | 所有模型 |
| 属性压缩 | 0% | 3-8% | 移动端 |
七、完整工作流示例
-
预处理阶段
void PreprocessModel(string path) {ModelImporter importer = ModelImporter.GetAtPath(path) as ModelImporter;// 基础设置importer.optimizeMesh = true;importer.keepQuads = false;importer.weldVertices = true;// 法线计算importer.importNormals = ModelImporterNormals.Calculate;importer.normalCalculationMode = ModelImporterNormalCalculationMode.AreaAndAngleWeighted;importer.normalSmoothingAngle = 60;// UV优化importer.generateSecondaryUV = true;importer.secondaryUVPackMargin = 0.003f;importer.SaveAndReimport();
}
-
运行时优化
IEnumerator RuntimeOptimization(GameObject model) {MeshFilter mf = model.GetComponent<MeshFilter>();if(mf == null) yield break;// 异步加载后优化while(mf.sharedMesh == null) {yield return null;}Mesh optimizedMesh = Instantiate(mf.sharedMesh);optimizedMesh.name = mf.sharedMesh.name + "_Optimized";// 执行优化流程Vector3[] smoothedNormals = SmoothNormals(optimizedMesh);optimizedMesh.normals = smoothedNormals;optimizedMesh = OptimizeVertexCache(optimizedMesh);mf.sharedMesh = optimizedMesh;
}
八、实用工具推荐
-
Unity官方工具
-
Mesh.Optimize方法
-
Model Importer中的优化选项
-
-
第三方插件
-
Mesh Baker:合并和优化网格
-
Simplygon:自动LOD生成
-
Maya/Blender:专业的UV展开工具
-
通过综合应用这些技术,开发者可以显著减少顶点数量,提升渲染性能,特别是在移动设备和VR应用中效果尤为明显。建议在项目早期建立优化流程,避免后期大规模返工。
相关文章:
Unity顶点优化:UV Splits与Smoothing Splits消除技巧
一、顶点分裂问题概述 1. 什么是顶点分裂 顶点分裂(Vertex Splits)是3D渲染中常见的性能问题,当模型需要为同一顶点位置存储不同属性值时,会创建多个顶点副本。主要分为两类: UV Splits:由UV不连续引起 Smoothing Splits&#…...
第五十三章 Spring之假如让你来写Boot——环境篇
Spring源码阅读目录 第一部分——IOC篇 第一章 Spring之最熟悉的陌生人——IOC 第二章 Spring之假如让你来写IOC容器——加载资源篇 第三章 Spring之假如让你来写IOC容器——解析配置文件篇 第四章 Spring之假如让你来写IOC容器——XML配置文件篇 第五章 Spring之假如让你来写…...
Router [Continuation Settings]
楼上网络CMCC-Wmew,楼下接收不到,可能因为喜好弱,再弄一台路由器中转一下 Router [Continuation Settings] 路由器中续设置 到这里这台K3的路由器设置完成了,作为转发,中续,她还需要设置上游路由器&#…...
Zookeeper中的Zxid是如何设计的
想获取更多高质量的Java技术文章?欢迎访问Java技术小馆官网,持续更新优质内容,助力技术成长 Java技术小馆官网https://www.yuque.com/jtostring Zookeeper中的Zxid是如何设计的 如果你们之前学习过 ZooKeeper,你们可能已经了解…...
蓝桥云客 岛屿个数
0岛屿个数 - 蓝桥云课 问题描述 小蓝得到了一副大小为 MN 的格子地图,可以将其视作一个只包含字符 0(代表海水)和 1(代表陆地)的二维数组,地图之外可以视作全部是海水,每个岛屿由在上/下/左/右…...
深度学习篇---paddleocr正则化提取
文章目录 前言一、代码总述&介绍1.1导入必要的库1.1.1cv21.1.2re1.1.3paddleocr 1.2初始化PaddleOCR1.3打开摄像头1.4使用 PaddleOCR 进行识别1.5定义正则表达式模式1.6打印提取结果1.7异常处理 二、正则表达式2.1简介2.2常用正则表达式模式及原理2.2.1. 快递单号模式2.2.2…...
Android 蓝牙/Wi-Fi通信协议之:低功耗蓝牙(BLE 4.0+)介绍
介绍:蓝牙通信协议详解 1. 蓝牙协议分层 Android主要支持**经典蓝牙(Bluetooth Classic)和低功耗蓝牙(BLE)**两种模式: 经典蓝牙(BT 2.1/3.0) 低功耗蓝牙(BLE 4.0&…...
流影---开源网络流量分析平台(四)(分析引擎部署)
目录 功能介绍 部署过程 一、安装依赖环境 二、源码编译部署 三、运行环境配置 四、运行配置 功能介绍 本章我将继续安装流影的分析引擎组件首先,ly_analyser是流影的威胁行为分析引擎,读取netflow v9格式的数据作为输入,运行各种威胁行…...
31天Python入门——第14天:异常处理
你好,我是安然无虞。 文章目录 异常处理1. Python异常2. 异常捕获try-except语句捕获所有的异常信息获取异常对象finally块 3. raise语句4. 自定义异常5. 函数调用里面产生的异常补充练习 异常处理 1. Python异常 Python异常指的是在程序执行过程中发生的错误或异…...
浅析Android Jetpack ACC之LiveData
一、Android Jetpack简介 Android官网对Jetpack的介绍如下: Jetpack is a suite of libraries to help developers follow best practices, reduce boilerplate code, and write code that works consistently across Android versions and devices so that develo…...
【区块链安全 | 第十五篇】类型之值类型(二)
文章目录 值类型有理数和整数字面量(Rational and Integer Literals)字符串字面量和类型(String Literals and Types)Unicode 字面量(Unicode Literals)十六进制字面量(Hexadecimal Literals&am…...
深度学习篇---模型训练评估参数
文章目录 前言一、Precision(精确率)1.1定义1.2意义1.3数值接近11.4数值再0.5左右1.5数值接近0 二、Recall(召回率)2.1定义2.2意义2.3数值接近12.4数值在0.5左右2.5数值接近0 三、Accuracy(准确率)3.1定义3…...
SQL Server 可用性组自动种子设定失败问题
目录标题 SQL Server 可用性组自动种子设定失败问题笔记一、问题背景二、错误日志分析错误信息错误代码与分析 三、自动种子设定概述(同上,无需修改)四、解决步骤1. 备份主数据库2. 在辅助副本上恢复数据库3. 重新启动自动种子设定 SQL Serve…...
02 相机标定相关坐标系
标定相关坐标系 一共四个坐标系 图像像素坐标系: u-v,图像左上角为原点图像物理坐标系: x-y,图像中心为原点...
Ubuntu修改用户名
修改用户名: 1.CTRL ALT T 快捷键打开终端,输入‘sudo su’ 转为root用户。 2.输入‘ gredit /etc/passwd ’,修改用户名,只修改用户名,后面的全名、目录等不修改。 3.输入 ‘ gedit /etc/shadow ’ 和 ‘ gedit /etc/…...
Windows 系统下多功能免费 PDF 编辑工具详解
IceCream PDF Editor是一款极为实用且操作简便的PDF文件编辑工具,它完美适配Windows操作系统。其用户界面设计得十分直观,哪怕是初次接触的用户也能快速上手。更为重要的是,该软件具备丰富多样的强大功能,能全方位满足各类PDF编辑…...
UE学习记录part11
第14节 breakable actors 147 destructible meshes a geometry collection is basically a set of static meshes that we get after we fracture a mesh. 几何体集合基本上是我们在断开网格后获得的一组静态网格。 选中要破碎的网格物品,创建集合 可以选择不同的…...
Redis-07.Redis常用命令-集合操作命令
一.集合操作命令 SADD key member1 [member2]: sadd set1 a b c d sadd set1 a 0表示没有添加成功,因为集合中已经有了这个元素了,因此无法重复添加。 SMEMBERS key: smembers set1 SCARD key: scard set1 SADD key member1 …...
vscode 源代码管理
https://code.visualstudio.com/updates/v1_92#_source-control 您可以通过切换 scm.showHistoryGraph 设置来禁用传入/传出更改的图形可视化。...
arm64位FFmpeg与X264库
参考链接: https://blog.csdn.net/gitblog_09700/article/details/142945092...
iOS审核被拒:Missing privacy manifest 第三方库添加隐私声明文件
问题: iOS提交APP审核被拒,苹果开发者网页显示二进制错误,收到的邮件显示的详细信息如下图: 分析: 从上面信息能看出第三方SDK库必须要包含一个隐私文件,去第三方库更新版本。 几经查询资料得知,苹果在…...
用mkdocs写文档#自动更新github-page
https://wuyisheng.github.io/blog 背景是上一篇博客 使用mkdocs,最后提及可以部署github page。这里说明下怎么自动部署。 当然,这篇博客主要的目的还是提供下github page的链接 :) 我是这样做的: step 1: pip3 i…...
【LeetCode Solutions】LeetCode 101 ~ 105 题解
CONTENTS LeetCode 101. 对称二叉树(简单)LeetCode 102. 二叉树的层序遍历(中等)LeetCode 103. 二叉树的锯齿形层序遍历(中等)LeetCode 104. 二叉树的最大深度(简单)LeetCode 105. 从…...
Orpheus-TTS 介绍,新一代开源文本转语音
Orpheus-TTS 是由 Canopy Labs 团队于2025年3月19日发布的开源文本转语音(TTS)模型,其技术突破集中在超低延迟、拟人化情感表达与实时流式生成三大领域。以下从技术架构、核心优势、应用场景、对比分析、开发背景及最新进展等多维度展开深入解…...
Java数据结构-栈和队列
目录 1. 栈(Stack) 1.1 概念 1.2 栈的使用 1.3 栈的模拟实现 1.4 栈的应用场景 1. 改变元素的序列 2. 将递归转化为循环 3. 括号匹配 4. 逆波兰表达式求值 5. 出栈入栈次序匹配 6. 最小栈 1.5 概念区分 2. 队列(Queue) 2.1 概念 2.2 队列的使用 2.3 队列模拟实…...
MySQL中的CREATE TABLE LIKE和CREATE TABLE SELECT
MySQL中的CREATE TABLE LIKE和CREATE TABLE SELECT CREATE TABLE LIKECREATE TABLE SELECT CREATE TABLE LIKE CREATE TABLE ... LIKE可以用来复制表结构,源表上的索引和约束也会复制。CREATE TABLE ... LIKE不能复制表数据。CREATE TABLE ... LIKE只能复制基表&…...
权重衰减-笔记
《动手学深度学习》-4.5-笔记 权重衰减就像给模型“勒紧裤腰带”,不让它太贪心、不让它学太多。 你在学英语单词,别背太多冷门单词,只背常见的就行,这样考试时更容易拿分。” —— 这其实就是在“限制你学的内容复杂度”。 在…...
Hyperliquid 遇袭「拔网线」、Polymarket 遭治理攻击「不作为」,从双平台危机看去中心化治理的进化阵痛
作者:Techub 热点速递 撰文:Glendon,Techub News 继 3 月 12 日「Hyperliquid 50 倍杠杆巨鲸」引发的 Hyperliquid 清算事件之后,3 月 26 日 晚间,Hyperliquid 再次遭遇了一场针对其流动性和治理模式的「闪电狙击」。…...
软考笔记6——结构化开发方法
第六章节——结构化开发方法 结构化开发方法 第六章节——结构化开发方法一、系统分析与设计概述1. 系统分析概述2. 系统设计的基本原理3. 系统总体结构设计 二、结构化分析方法1. 结构化分析方法概述2. 数据流图(DFD)3. 数据字典 三、结构化设计方法(了解ÿ…...
一种C# Winform的UI处理
效果 圆角 阴影 突出按钮 说明 这是一种另类的处理,不是多层窗口 也不是WPF 。这种方式的特点是比较简单,例如圆角、阴影、按钮等特别容易修改过。其实就是html css DirectXForm。 在VS中如下 圆角和阴影 然后编辑这个窗体的Html模板,…...
