Unity字符串性能问题

前言

分享一些通过书籍和网络学到的知识

每次动态创建一个string,C#都会在堆内存分配一个内存用来分配字符串,因为C#没有对字符串的缓存机制,会导致每次连接、切割、组合的时候都会申请新的内存,并且抛弃原来的内存,等待GC,而GC又会消耗很多CPU空间,例如对于a = "ax"; c = "b" + a + "c",会造成一定程度的性能浪费。因为这里的+操作符是在运行时进行的,而不是在编译时。所以,每次+操作都会创建一个新的字符串对象,而不是改变原来的对象。这样就会产生两个无用的中间对象,"bax"和"baxc",占用内存空间,如果不注意的话就会导致程序卡顿逐步增加,降低运行效率,有几种办法提高字符串效率,注意StringFormat和用+拼接还有$语法糖都会有一定效率问题,提高效率还是尽量用StringBuilder,但是为了可读性字符串比较短而且低频没必要的话就语法糖就好。

id-字符串字典池

自己建一个Key-字符串的缓存池,拼的时候从池子里面找减少GC

Dict<int,string>

复制代码
Dictionary<int,string> strCache;
string strName = null;
if(!strCache.TryGetValue(id, out strName))
{
ResData resData = GetDataById(ID);
string strName = "This is " + resData,Name;
strCache.Add(id,strName);
}
return strName;

做一个定长的字符串池,用指针来拼接字符串

将指针指向目标字符串,把strA和strB分别拼在前置位和后置位,减少内存的分配和释放次数

复制代码
public unsafe string Concat(string strA, string strB)  
{  
    int a_length = strA.Length;  
    int b_length = strB.Length;  
    int sum_length = a_length + b_length;  
    string strResult = null;  
    if (!cacheStr.TryGetValue(sum_length, out strResult))  
    // 如果不存在 sum_length 长度的缓存字符串,那么直接连接后存入缓存  
    {  
        strResult = strA + strB;        cacheStr.Add(sum_length, strResult);  
        return strResult;  
    }    //字符串再利用  
    fixed (char* strA_ptr = strA)  
    {        fixed (char* strB_ptr = strB)  
        {            fixed (char* strResult_ptr = strResult)  
            {                memcopy((byte*)strResult_ptr, (byte*)strA_ptr, a_length * sizeof(char));  
                memcopy((byte*)strResult_ptr+a_length,(byte*)strB_ptr,b_length * sizeof(char));  
            }        }    }    return strResult;  
}

memcopy函数

复制代码
public unsafe void memcopy(byte* dest, byte* src, int len)  
{  
    while((--len)>=0)  
    {dest[len] = src[len];}}

StringBuilder

同样可以解决多次创建字符串反复拼接时候的GC问题,StringBuilder是可变的,也就是说它可以在原有的字符串上进行修改,而不需要创建新的字符串对象,这样可以节省内存空间和提高效率。StringBuilder支持链式调用,可以方便地进行多个字符串操作,例如builder.Append("a").Append("b").Append("c")。StringBuilder可以动态地调整其容量,当字符串长度超过当前容量时,它会自动扩展容量,而不会抛出异常。

缺点是tringBuilder是线程不安全的,也就是说在多线程环境下,如果多个线程同时对同一个StringBuilder对象进行修改,可能会导致数据不一致或者错误。StringBuilder的ToString方法需要遍历其内部的字符数组,将其拼接成一个新的字符串对象,这个过程可能会消耗一定的时间和资源。StringBuilder只会在toString时创建一次字符串,相比于之前的多次创建字符串给CPU性能压力减少很大。

相关推荐
叶帆9 天前
【YFIOs】用C#开发硬件之设备上云
开发语言·unity·c#
久数君9 天前
AI三维建模工具“造形家”:地理场景三维化的高效解决方案
unity·glb·ai算法·ai三维建模工具·地图框选·造形家·城市建筑模型
会思考的猴子10 天前
Unity VFX 属性 Postion 和 TargetPostion
unity
hai31524754310 天前
九章编程法 · 猜数字游戏 (GW-BASIC 重构版) *
人工智能·microsoft·游戏引擎·游戏程序
心前阳光10 天前
Unity资源导入之自动化资源导入
unity·自动化·游戏引擎
心前阳光10 天前
Unity之2021.3.45f2c1发布安卓程序遇到的问题
android·unity·游戏引擎
纪纯10 天前
PicoVR Unity Integration SDK 3.4 常用交互API
unity·游戏引擎·vr·pico
龙智DevSecOps解决方案10 天前
3A 游戏优化技术栈:如何打通引擎级分析工具与 DevOps 持续集成管线?
unity·性能优化·游戏开发·技术美术·perforce·unrealengine
葛兰岱尔10 天前
从 SolidWorks 到 Three.js,从 Inventor 到 Unity——制造业CAD模型“几何-语义一体化“转换,不再是天方夜谭!
开发语言·javascript·unity
鼎艺创新科技10 天前
三维电子沙盘中OSGB倾斜摄影数据的加载与渲染
游戏引擎·cocos2d