IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> unity AB包体大小优化 -> 正文阅读

[游戏开发]unity AB包体大小优化

1模型Texture 贴图优化

项目AB里面,角色的数量比较多,从而角色模型的AB体积也比较庞大,角色模型一方面要控制好顶点 三角面数量,另一方面就是模型贴图资源优化,美术做资源的时候,为了追求质量,经常会把模型贴图搞的特别大,就拿我们现在这个项目来说吧,美术用的的模型贴图都是4096*4096的,到了unity里面,肯定不能这样搞要,4096你让低端机还玩个鸟呀,所以限定了贴图 Max Size为2048。
原始的资源格式如下,一个模型里面包含3张这样格式的贴图,分别是uv贴图,法线贴图,和一张灰度贴图,打出来的AB资源是5.7M。
在这里插入图片描述
在这里插入图片描述
考虑到我们的项目是RPG 回合制游戏,角色只有在战斗场景中会存在多个,而且距离摄像机距离没有特别明显的变动,所以就建议去掉MipMap。然后打包AB,资源大小为4.1M。
接下来要处理的就是贴图压缩了,我这里主要测试了Android的,由于贴图都不带透明通道,所以UV贴图和灰度贴图采用了RGBCrunchedETC格式,法线贴图采用了RGBCompressETC4bit 方式,AB资源大小为2M
如果法线贴图也按照RGBCrunchedETC方式进行压缩的话,AB资源大小可以降到1.4M,这个根据根据显示效果而定。
还有就是 Max Size的设置,最好是根据模型显示效果进行分类控制

2TimeLine 资源优化

项目战斗技能,我们采用的是TimeLine制作的,今天查看了一下TimeLine资源,一个TimeLine 的prefab资源,竟然达到了1M多,有个别的都2M了,我的天呢,你们这群美术小伙伴都对TimeLine做了啥,于是我就用AssetBundleBorswer工具分析了一下00003号角色大招的 TimeLine AB资源,不看不知道,一看吓一跳,这些都是什么鬼呀。为什么会有一堆其他角色的animation和timeline资源。
在这里插入图片描述
然后我看了一眼 unity 编辑器中的原始资源
在这里插入图片描述
在这里插入图片描述
资源好像也没啥毛病,那这一堆乱七八糟的资源是从哪里来的,既然AB认为他们有引用关系,那肯定还是有问题的,只能从meta文件中在进行求证了。这次倒是没有让我失望,在m_SceneBindings属性下面看到了这一堆被引用的资源,无语了。
在这里插入图片描述
然后回到unity,开启debug 模式,重新审视资源,怪我太年轻,不懂人心险恶,差点儿就被你蒙混过关了,这106个资源引用,打出来的AB能小吗,哎!
在这里插入图片描述
我实际需要的资源,只有关闭debug模式的时候,显示的这三个资源呀
在这里插入图片描述
我该肿么办,既然是码农,那就发挥出搬砖精神吧,度娘度娘,我要搬砖了,然后度娘给了我一个链接
https://zhuanlan.zhihu.com/p/396526134
果然是同道中人,感谢楼主帮我们趟坑,然后复制粘贴,一个修改工具就做好了

    [MenuItem("Assets/CleanUpPlayableBind")]
    private static void CleanUpPlayableBind()
    {
        GameObject gob = Selection.activeGameObject as GameObject;
        if (gob != null) {
            var playable = gob.GetComponent<PlayableDirector>();
            CleanUpBind(playable);
        }
    }

    public static void CleanUpBind(PlayableDirector playable)
    {
        if (playable == null) return;
        Dictionary<UnityEngine.Object, UnityEngine.Object> bindings = new Dictionary<UnityEngine.Object, UnityEngine.Object>();
        foreach (var pb in playable.playableAsset.outputs)
        {
            var key = pb.sourceObject;
            var value = playable.GetGenericBinding(key);
            if (!bindings.ContainsKey(key) && key != null)
            {
                bindings.Add(key, value);
            }
        }

        var dirSO = new UnityEditor.SerializedObject(playable);
        var sceneBindings = dirSO.FindProperty("m_SceneBindings");
        for (var i = sceneBindings.arraySize - 1; i >= 0; i--)
        {
            var binding = sceneBindings.GetArrayElementAtIndex(i);
            var key = binding.FindPropertyRelative("key");
            if (key.objectReferenceValue == null || !bindings.ContainsKey(key.objectReferenceValue))
                sceneBindings.DeleteArrayElementAtIndex(i);
        }
        dirSO.ApplyModifiedProperties();
    }

搞定!顺便提一嘴,TimeLine用到的animaton资源,记得压缩它们的浮点数精度值和scale通道,通常情况下我们用不到这么高的精度,之前打AB比较大,除了多余的资源,animation也没有压缩。最终这个AB资源从1.6M,被我降低到了132k,Nice。
在这里插入图片描述
在这里插入图片描述

Animation优化

核心代码

    [MenuItem("Assets/AnimationClipChange/Directory/降低精度并删除Scale通道")]
    private static void ChangeAnimFloatAndScaleDirectory()
    {
        Object gob = Selection.activeObject;
        string path = AssetDatabase.GetAssetPath(gob);
        if (Directory.Exists(path))
        {
            string[] udids = AssetDatabase.FindAssets("t:AnimationClip", new string[] { path });
            int index = 0;
            foreach (var item in udids)
            {
                string itemPath = AssetDatabase.GUIDToAssetPath(item);
                EditorUtility.DisplayProgressBar("执行中..." + path, itemPath, (float)index / udids.Length);

                AnimationClip clip = AssetDatabase.LoadAssetAtPath<AnimationClip>(itemPath);
                optmizeAnimationFloatAndScale(clip);
                index++;
            }
            EditorUtility.ClearProgressBar();
            Debug.LogColor("AnimationClip change float finish!");
            AssetDatabase.SaveAssets();
            AssetDatabase.Refresh();
        }
        else
        {
            Debug.LogError("select target is not a directory:" + path);
        }

    }


    static void optmizeAnimationFloatAndScale(AnimationClip targetAnimClip)
    {
        optmizeAnimationScaleCurve(targetAnimClip);
        optmizeAnimationFloat(targetAnimClip);
        Resources.UnloadUnusedAssets();
        
    }


    /// <summary>
    /// 优化浮点数精度
    /// </summary>
    static AnimationClip optmizeAnimationFloat(AnimationClip clip)
    {
        //浮点数精度压缩到f3
        AnimationClipCurveData[] curves = null;
        curves = AnimationUtility.GetAllCurves(clip);
        Keyframe key;
        Keyframe[] keyFrames;
        for (int ii = 0; ii < curves.Length; ++ii)
        {
            AnimationClipCurveData curveDate = curves[ii];
            if (curveDate.curve == null || curveDate.curve.keys == null)
            {
                //Debug.LogWarning(string.Format("AnimationClipCurveData {0} don't have curve; Animation name {1} ", curveDate, animationPath));
                continue;
            }
            keyFrames = curveDate.curve.keys;
            for (int i = 0; i < keyFrames.Length; i++)
            {
                key = keyFrames[i];
                key.value = float.Parse(key.value.ToString("f3"));
                key.inTangent = float.Parse(key.inTangent.ToString("f3"));
                key.outTangent = float.Parse(key.outTangent.ToString("f3"));
                keyFrames[i] = key;
            }
            curveDate.curve.keys = keyFrames;
            clip.SetCurve(curveDate.path, curveDate.type, curveDate.propertyName, curveDate.curve);
        }

        return clip;
    }

    /// <summary>
    /// 优化scale曲线
    /// </summary>
    static AnimationClip optmizeAnimationScaleCurve(AnimationClip clip)
    {
        //去除scale曲线
        foreach (EditorCurveBinding theCurveBinding in AnimationUtility.GetCurveBindings(clip))
        {
            string name = theCurveBinding.propertyName.ToLower();
            if (name.Contains("scale"))
            {
                AnimationUtility.SetEditorCurve(clip, theCurveBinding, null);
            }
        }

        return clip;
    }
  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2021-08-24 15:52:55  更:2021-08-24 15:53:45 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/15 16:34:19-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码