| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661 | using System;using System.Collections.Generic;/* 介绍:    C# 0GC字符串补充方案。结合gstring与CString两者特点(向这两个方案的作者致敬),只有一个文件,性能与使用方便性高于两者。 报告地址:    https://coh5.cn/p/1ace6338.html 使用方式:    1.Unity引擎将zstring.cs文件放于plugins目录下即可使用(不在plugins目录,则IOS打包或IL2CPP打包等FULLAOT方式编译不过),或者直接把结构体定义放入zstring类中;其余C#程序将zstring.cs直接放入工程使用即可。    2.(最佳性能)当update每帧刷新标签显示,或者大量UI飘字,或者该字符串是短时间使用的则使用如下方式:        using (zstring.Block())        {            uiText1.text=(zstring)"hello world"+" you";            uiText2.text=zstring.format("{0},{1}","hello","world");        }        此方式设置的string值位于浅拷贝缓存中,一定时间可能会改变,出作用域后正确性不予保证。     3.资源路径这种需要常驻的则需要intern一下在作用域外使用         using (zstring.Block())        {            zstring a="Assets/";            zstring b=a+"prefabs/"+"/solider.prefab";            prefabPath1=b.Intern();            prefabPath2=zstring.format("{0},{1}","hello","world").Intern();        }        此方式设置的string值位于深拷贝缓存中,游戏运行期间不会改变,可以在作用域外使用。    4.不可使用zstring作为类的成员变量,不建议在using作用域中写for循环,而是在for循环内using。    5.首次调用时会初始化类,分配各种空间,建议游戏启动时调用一次using(zstring.Block()){}    6.0GC。时间消耗上,短字符串处理,zstring比gstring时间少20%~30%,比原生慢。大字符串处理,zstring比gstring时间少70%~80%,接近原生string速度。    7.追求极限性能的话,核心函数可以用C++Dll中的 memcpy内存拷贝函数,性能提升10%~20%,一般没这个必要。    8.测试打开zstringTest工程,在Test脚本上勾选与不勾选bigStringTest下查看Profile性能。(同时对比了zstring,gstring,CString,还有王国纪元里的string)    9.据热心用户反应,IL2CPP 2017.4 在 Android上有字节对齐问题,换成2018就木有了。所以此时解决办法有三个:1.IL2CPP换成2018以上版本。 2.719行左右的memcpy函数换成循环一次拷贝一个字节。 3.不怕麻烦的话此处调用C语言的内存拷贝函数dll,即C语言<string.h>中的memcpy,这样性能也更高。    10.有事请联系 871041532@outlook.com 或 QQ(微信):871041532 */namespace Fort23.Core{    struct Byte8192    {        Byte4096 a1; Byte4096 a2;    }    struct Byte4096    {        Byte2048 a1; Byte2048 a2;    }    struct Byte2048    {        Byte1024 a1; Byte1024 a2;    }    struct Byte1024    {        Byte512 a1; Byte512 a2;    }    struct Byte512    {        Byte256 a1; Byte256 a2;    }    struct Byte256    {        Byte128 a1; Byte128 a2;    }    struct Byte128    {        Byte64 a1; Byte64 a2;    }    struct Byte64    {        Byte32 a1; Byte32 a2;    }    struct Byte32    {        Byte16 a1; Byte16 a2;    }    struct Byte16    {        Byte8 a1; Byte8 a2;    }    struct Byte8    {        long a1;    }    struct Byte4    {        int a1;    }    struct Byte2    {        short a;    }    struct Byte1    {        byte a;    }    public class zstring    {        static Queue<zstring>[] g_cache;//idx特定字符串长度,深拷贝核心缓存        static Dictionary<int, Queue<zstring>> g_secCache;//key特定字符串长度value字符串栈,深拷贝次级缓存        static Stack<zstring> g_shallowCache;//浅拷贝缓存        static Stack<zstring_block> g_blocks;//zstring_block缓存栈        static Stack<zstring_block> g_open_blocks;//zstring已经打开的缓存栈              static Dictionary<int, string> g_intern_table;//字符串intern表        public static zstring_block g_current_block;//zstring所在的block块        static List<int> g_finds;//字符串replace功能记录子串位置        static zstring[] g_format_args;//存储格式化字符串值        const int INITIAL_BLOCK_CAPACITY = 32;//gblock块数量          const int INITIAL_CACHE_CAPACITY = 128;//cache缓存字典容量  128*4Byte 500多Byte        const int INITIAL_STACK_CAPACITY = 48;//cache字典每个stack默认nstring容量        const int INITIAL_INTERN_CAPACITY = 256;//Intern容量        const int INITIAL_OPEN_CAPACITY = 5;//默认打开层数为5        const int INITIAL_SHALLOW_CAPACITY = 100;//默认50个浅拷贝用        const char NEW_ALLOC_CHAR = 'X';//填充char        private bool isShallow = false;//是否浅拷贝        [NonSerialized]        string _value;//值        [NonSerialized]        bool _disposed;//销毁标记        //不支持构造        private zstring()        {            throw new NotSupportedException();        }        //带默认长度的构造        private zstring(int length)        {            _value = new string(NEW_ALLOC_CHAR, length);        }        //浅拷贝专用构造        private zstring(string value, bool shallow)        {            if (!shallow)            {                throw new NotSupportedException();            }            _value = value;            isShallow = true;        }        static zstring()        {            Initialize(INITIAL_CACHE_CAPACITY,                       INITIAL_STACK_CAPACITY,                       INITIAL_BLOCK_CAPACITY,                       INITIAL_INTERN_CAPACITY,                       INITIAL_OPEN_CAPACITY,                       INITIAL_SHALLOW_CAPACITY                       );            g_finds = new List<int>(10);            g_format_args = new zstring[10];        }        //析构        private void dispose()        {            if (_disposed)                throw new ObjectDisposedException(this);            if (isShallow)//深浅拷贝走不同缓存            {                g_shallowCache.Push(this);            }            else            {                Queue<zstring> stack;                if (g_cache.Length > Length)                {                    stack = g_cache[Length];//取出valuelength长度的栈,将自身push进去                }                else                {                    stack = g_secCache[Length];                }                stack.Enqueue(this);            }            //memcpy(_value, NEW_ALLOC_CHAR);//内存拷贝至value            _disposed = true;        }        //由string获取相同内容zstring,深拷贝        private static zstring get(string value)        {            if (value == null)                return null;#if DBG            if (log != null)                log("Getting: " + value);#endif            var result = get(value.Length);            memcpy(dst: result, src: value);//内存拷贝            return result;        }        //由string浅拷贝入zstring        private static zstring getShallow(string value)        {            if (g_current_block == null)            {                throw new InvalidOperationException("nstring 操作必须在一个nstring_block块中。");            }            zstring result;            if (g_shallowCache.Count == 0)            {                result = new zstring(value, true);            }            else            {                result = g_shallowCache.Pop();                result._value = value;            }            result._disposed = false;            g_current_block.push(result);//zstring推入块所在栈            return result;        }        //将string加入intern表中        private static string __intern(string value)        {            int hash = value.GetHashCode();            if (g_intern_table.ContainsKey(hash))            {                return g_intern_table[hash];            }            else            {                string interned = new string(NEW_ALLOC_CHAR, value.Length);                memcpy(interned, value);                g_intern_table.Add(hash, interned);                return interned;            }        }        //手动添加方法        private static void getStackInCache(int index, out Queue<zstring> outStack)        {            int length = g_cache.Length;            if (length > index)//从核心缓存中取            {                outStack = g_cache[index];            }            else//从次级缓存中取            {                if (!g_secCache.TryGetValue(index, out outStack))                {                    outStack = new Queue<zstring>(INITIAL_STACK_CAPACITY);                    g_secCache[index] = outStack;                }            }        }        //获取特定长度zstring        private static zstring get(int length)        {            if (g_current_block == null || length <= 0)                throw new InvalidOperationException("zstring 操作必须在一个zstring_block块中。");            zstring result;            Queue<zstring> stack;            getStackInCache(length, out stack);            //从缓存中取Stack            if (stack.Count == 0)            {                result = new zstring(length);            }            else            {                result = stack.Dequeue();            }            result._disposed = false;            g_current_block.push(result);//zstring推入块所在栈            return result;        }        //value是10的次方数        private static int get_digit_count(long value)        {            int cnt;            for (cnt = 1; (value /= 10) > 0; cnt++) ;            return cnt;        }        //value是10的次方数        private static uint get_digit_count(uint value)        {            uint cnt;            for (cnt = 1; (value /= 10) > 0; cnt++) ;            return cnt;        }        //value是10的次方数        private static int get_digit_count(int value)        {            int cnt;            for (cnt = 1; (value /= 10) > 0; cnt++) ;            return cnt;        }        //获取char在input中start起往后的下标        private static int internal_index_of(string input, char value, int start)        {            return internal_index_of(input, value, start, input.Length - start);        }        //获取string在input中起始0的下标        private static int internal_index_of(string input, string value)        {            return internal_index_of(input, value, 0, input.Length);        }        //获取string在input中自0起始下标        private static int internal_index_of(string input, string value, int start)        {            return internal_index_of(input, value, start, input.Length - start);        }        //获取格式化字符串        private unsafe static zstring internal_format(string input, int num_args)        {            if (input == null)                throw new ArgumentNullException("value");            //新字符串长度            int new_len = input.Length;            for (int i = -3; ;)            {                i = internal_index_of(input, '{', i + 3);                if (i == -1)                {                    break;                }                new_len -= 3;                int arg_idx = input[i + 1] - '0';                zstring arg = g_format_args[arg_idx];                new_len += arg.Length;            }            zstring result = get(new_len);            string res_value = result._value;            int next_output_idx = 0;            int next_input_idx = 0;            int brace_idx = -3;            for (int i = 0, j = 0, x = 0; ; x++) // x < num_args            {                brace_idx = internal_index_of(input, '{', brace_idx + 3);                if (brace_idx == -1)                {                    break;                }                next_input_idx = brace_idx;                int arg_idx = input[brace_idx + 1] - '0';                string arg = g_format_args[arg_idx]._value;                if (brace_idx == -1)                    throw new InvalidOperationException("没有发现大括号{ for argument " + arg);                if (brace_idx + 2 >= input.Length || input[brace_idx + 2] != '}')                    throw new InvalidOperationException("没有发现大括号} for argument " + arg);                fixed (char* ptr_input = input)                {                    fixed (char* ptr_result = res_value)                    {                        for (int k = 0; i < new_len;)                        {                            if (j < brace_idx)                            {                                ptr_result[i++] = ptr_input[j++];                                ++next_output_idx;                            }                            else                            {                                ptr_result[i++] = arg[k++];                                ++next_output_idx;                                if (k == arg.Length)                                {                                    j += 3;                                    break;                                }                            }                        }                    }                }            }            next_input_idx += 3;            for (int i = next_output_idx, j = 0; i < new_len; i++, j++)            {                fixed (char* ptr_input = input)                {                    fixed (char* ptr_result = res_value)                    {                        ptr_result[i] = ptr_input[next_input_idx + j];                    }                }            }            return result;        }        //获取char在字符串中start开始的下标        private unsafe static int internal_index_of(string input, char value, int start, int count)        {            if (start < 0 || start >= input.Length)                // throw new ArgumentOutOfRangeException("start");                return -1;            if (start + count > input.Length)                return -1;            // throw new ArgumentOutOfRangeException("count=" + count + " start+count=" + start + count);            fixed (char* ptr_this = input)            {                int end = start + count;                for (int i = start; i < end; i++)                    if (ptr_this[i] == value)                        return i;                return -1;            }        }        //获取value在input中自start起始下标        private unsafe static int internal_index_of(string input, string value, int start, int count)        {            int input_len = input.Length;            if (start < 0 || start >= input_len)                throw new ArgumentOutOfRangeException("start");            if (count < 0 || start + count > input_len)                throw new ArgumentOutOfRangeException("count=" + count + " start+count=" + (start + count));            if (count == 0)                return -1;            fixed (char* ptr_input = input)            {                fixed (char* ptr_value = value)                {                    int found = 0;                    int end = start + count;                    for (int i = start; i < end; i++)                    {                        for (int j = 0; j < value.Length && i + j < input_len; j++)                        {                            if (ptr_input[i + j] == ptr_value[j])                            {                                found++;                                if (found == value.Length)                                    return i;                                continue;                            }                            if (found > 0)                                break;                        }                    }                    return -1;                }            }        }        //移除string中自start起始count长度子串        private unsafe static zstring internal_remove(string input, int start, int count)        {            if (start < 0 || start >= input.Length)                throw new ArgumentOutOfRangeException("start=" + start + " Length=" + input.Length);            if (count < 0 || start + count > input.Length)                throw new ArgumentOutOfRangeException("count=" + count + " start+count=" + (start + count) + " Length=" + input.Length);            if (count == 0)                return input;            zstring result = get(input.Length - count);            internal_remove(result, input, start, count);            return result;        }        //将src中自start起count长度子串复制入dst        private unsafe static void internal_remove(string dst, string src, int start, int count)        {            fixed (char* src_ptr = src)            {                fixed (char* dst_ptr = dst)                {                    for (int i = 0, j = 0; i < dst.Length; i++)                    {                        if (i >= start && i < start + count) // within removal range                            continue;                        dst_ptr[j++] = src_ptr[i];                    }                }            }        }        //字符串replace,原字符串,需替换子串,替换的新子串        private unsafe static zstring internal_replace(string value, string old_value, string new_value)        {            // "Hello, World. There World" | World->Jon =            // "000000000000000000000" (len = orig - 2 * (world-jon) = orig - 4            // "Hello, 00000000000000"            // "Hello, Jon00000000000"            // "Hello, Jon. There 000"            // "Hello, Jon. There Jon"            // "Hello, World. There World" | World->Alexander =            // "000000000000000000000000000000000" (len = orig + 2 * (alexander-world) = orig + 8            // "Hello, 00000000000000000000000000"            // "Hello, Alexander00000000000000000"            // "Hello, Alexander. There 000000000"            // "Hello, Alexander. There Alexander"            if (old_value == null)                throw new ArgumentNullException("old_value");            if (new_value == null)                throw new ArgumentNullException("new_value");            int idx = internal_index_of(value, old_value);            if (idx == -1)                return value;            g_finds.Clear();            g_finds.Add(idx);            // 记录所有需要替换的idx点            while (idx + old_value.Length < value.Length)            {                idx = internal_index_of(value, old_value, idx + old_value.Length);                if (idx == -1)                    break;                g_finds.Add(idx);            }            // calc the right new total length            int new_len;            int dif = old_value.Length - new_value.Length;            if (dif > 0)                new_len = value.Length - (g_finds.Count * dif);            else                new_len = value.Length + (g_finds.Count * -dif);            zstring result = get(new_len);            fixed (char* ptr_this = value)            {                fixed (char* ptr_result = result._value)                {                    for (int i = 0, x = 0, j = 0; i < new_len;)                    {                        if (x == g_finds.Count || g_finds[x] != j)                        {                            ptr_result[i++] = ptr_this[j++];                        }                        else                        {                            for (int n = 0; n < new_value.Length; n++)                                ptr_result[i + n] = new_value[n];                            x++;                            i += new_value.Length;                            j += old_value.Length;                        }                    }                }            }            return result;        }        //向字符串value中自start位置插入count长度的to_insertChar        private unsafe static zstring internal_insert(string value, char to_insert, int start, int count)        {            // "HelloWorld" (to_insert=x, start=5, count=3) -> "HelloxxxWorld"            if (start < 0 || start >= value.Length)                throw new ArgumentOutOfRangeException("start=" + start + " Length=" + value.Length);            if (count < 0)                throw new ArgumentOutOfRangeException("count=" + count);            if (count == 0)                return get(value);            int new_len = value.Length + count;            zstring result = get(new_len);            fixed (char* ptr_value = value)            {                fixed (char* ptr_result = result._value)                {                    for (int i = 0, j = 0; i < new_len; i++)                    {                        if (i >= start && i < start + count)                            ptr_result[i] = to_insert;                        else                            ptr_result[i] = ptr_value[j++];                    }                }            }            return result;        }        //向input字符串中插入to_insert串,位置为start        private unsafe static zstring internal_insert(string input, string to_insert, int start)        {            if (input == null)                throw new ArgumentNullException("input");            if (to_insert == null)                throw new ArgumentNullException("to_insert");            if (start < 0 || start >= input.Length)                throw new ArgumentOutOfRangeException("start=" + start + " Length=" + input.Length);            if (to_insert.Length == 0)                return get(input);            int new_len = input.Length + to_insert.Length;            zstring result = get(new_len);            internal_insert(result, input, to_insert, start);            return result;        }        //字符串拼接        private unsafe static zstring internal_concat(string s1, string s2)        {            int total_length = s1.Length + s2.Length;            zstring result = get(total_length);            fixed (char* ptr_result = result._value)            {                fixed (char* ptr_s1 = s1)                {                    fixed (char* ptr_s2 = s2)                    {                        memcpy(dst: ptr_result, src: ptr_s1, length: s1.Length, src_offset: 0);                        memcpy(dst: ptr_result, src: ptr_s2, length: s2.Length, src_offset: s1.Length);                    }                }            }            return result;        }        //将to_insert串插入src的start位置,内容写入dst        private unsafe static void internal_insert(string dst, string src, string to_insert, int start)        {            fixed (char* ptr_src = src)            {                fixed (char* ptr_dst = dst)                {                    fixed (char* ptr_to_insert = to_insert)                    {                        for (int i = 0, j = 0, k = 0; i < dst.Length; i++)                        {                            if (i >= start && i < start + to_insert.Length)                                ptr_dst[i] = ptr_to_insert[k++];                            else                                ptr_dst[i] = ptr_src[j++];                        }                    }                }            }        }        //将长度为count的数字插入dst中,起始位置为start,dst的长度需大于start+count        private unsafe static void longcpy(char* dst, long value, int start, int count)        {            int end = start + count;            for (int i = end - 1; i >= start; i--, value /= 10)                *(dst + i) = (char)(value % 10 + 48);        }        //将长度为count的数字插入dst中,起始位置为start,dst的长度需大于start+count        private unsafe static void intcpy(char* dst, int value, int start, int count)        {            int end = start + count;            for (int i = end - 1; i >= start; i--, value /= 10)                *(dst + i) = (char)(value % 10 + 48);        }        private static unsafe void _memcpy4(byte* dest, byte* src, int size)        {            /*while (size >= 32) {                // using long is better than int and slower than double                // FIXME: enable this only on correct alignment or on platforms                // that can tolerate unaligned reads/writes of doubles                ((double*)dest) [0] = ((double*)src) [0];                ((double*)dest) [1] = ((double*)src) [1];                ((double*)dest) [2] = ((double*)src) [2];                ((double*)dest) [3] = ((double*)src) [3];                dest += 32;                src += 32;                size -= 32;            }*/            while (size >= 16)            {                ((int*)dest)[0] = ((int*)src)[0];                ((int*)dest)[1] = ((int*)src)[1];                ((int*)dest)[2] = ((int*)src)[2];                ((int*)dest)[3] = ((int*)src)[3];                dest += 16;                src += 16;                size -= 16;            }            while (size >= 4)            {                ((int*)dest)[0] = ((int*)src)[0];                dest += 4;                src += 4;                size -= 4;            }            while (size > 0)            {                ((byte*)dest)[0] = ((byte*)src)[0];                dest += 1;                src += 1;                --size;            }        }        private static unsafe void _memcpy2(byte* dest, byte* src, int size)        {            while (size >= 8)            {                ((short*)dest)[0] = ((short*)src)[0];                ((short*)dest)[1] = ((short*)src)[1];                ((short*)dest)[2] = ((short*)src)[2];                ((short*)dest)[3] = ((short*)src)[3];                dest += 8;                src += 8;                size -= 8;            }            while (size >= 2)            {                ((short*)dest)[0] = ((short*)src)[0];                dest += 2;                src += 2;                size -= 2;            }            if (size > 0)            {                ((byte*)dest)[0] = ((byte*)src)[0];            }        }        //从src,0位置起始拷贝count长度字符串src到dst中        //private unsafe static void memcpy(char* dest, char* src, int count)        //{        //    // Same rules as for memcpy, but with the premise that         //    // chars can only be aligned to even addresses if their        //    // enclosing types are correctly aligned        //    superMemcpy(dest, src, count);        //    //if ((((int)(byte*)dest | (int)(byte*)src) & 3) != 0)//转换为byte指针        //    //{        //    //    if (((int)(byte*)dest & 2) != 0 && ((int)(byte*)src & 2) != 0 && count > 0)        //    //    {        //    //        ((short*)dest)[0] = ((short*)src)[0];        //    //        dest++;        //    //        src++;        //    //        count--;        //    //    }        //    //    if ((((int)(byte*)dest | (int)(byte*)src) & 2) != 0)        //    //    {        //    //        _memcpy2((byte*)dest, (byte*)src, count * 2);//转换为short*指针一次两个字节拷贝        //    //        return;        //    //    }        //    //}        //    //_memcpy4((byte*)dest, (byte*)src, count * 2);//转换为int*指针一次四个字节拷贝        //}        //--------------------------------------手敲memcpy-------------------------------------//        private static int m_charLen = sizeof(char);        private unsafe static void memcpy(char* dest, char* src, int count)        {            byteCopy((byte*)dest, (byte*)src, count * m_charLen);        }        private unsafe static void byteCopy(byte* dest, byte* src, int byteCount)        {            if (byteCount < 128)            {                goto g64;            }            else if (byteCount < 2048)            {                goto g1024;            }            while (byteCount >= 8192)            {                ((Byte8192*)dest)[0] = ((Byte8192*)src)[0];                dest += 8192;                src += 8192;                byteCount -= 8192;            }            if (byteCount >= 4096)            {                ((Byte4096*)dest)[0] = ((Byte4096*)src)[0];                dest += 4096;                src += 4096;                byteCount -= 4096;            }            if (byteCount >= 2048)            {                ((Byte2048*)dest)[0] = ((Byte2048*)src)[0];                dest += 2048;                src += 2048;                byteCount -= 2048;            }        g1024: if (byteCount >= 1024)            {                ((Byte1024*)dest)[0] = ((Byte1024*)src)[0];                dest += 1024;                src += 1024;                byteCount -= 1024;            }            if (byteCount >= 512)            {                ((Byte512*)dest)[0] = ((Byte512*)src)[0];                dest += 512;                src += 512;                byteCount -= 512;            }            if (byteCount >= 256)            {                ((Byte256*)dest)[0] = ((Byte256*)src)[0];                dest += 256;                src += 256;                byteCount -= 256;            }            if (byteCount >= 128)            {                ((Byte128*)dest)[0] = ((Byte128*)src)[0];                dest += 128;                src += 128;                byteCount -= 128;            }        g64: if (byteCount >= 64)            {                ((Byte64*)dest)[0] = ((Byte64*)src)[0];                dest += 64;                src += 64;                byteCount -= 64;            }            if (byteCount >= 32)            {                ((Byte32*)dest)[0] = ((Byte32*)src)[0];                dest += 32;                src += 32;                byteCount -= 32;            }            if (byteCount >= 16)            {                ((Byte16*)dest)[0] = ((Byte16*)src)[0];                dest += 16;                src += 16;                byteCount -= 16;            }            if (byteCount >= 8)            {                ((Byte8*)dest)[0] = ((Byte8*)src)[0];                dest += 8;                src += 8;                byteCount -= 8;            }            if (byteCount >= 4)            {                ((Byte4*)dest)[0] = ((Byte4*)src)[0];                dest += 4;                src += 4;                byteCount -= 4;            }            if (byteCount >= 2)            {                ((Byte2*)dest)[0] = ((Byte2*)src)[0];                dest += 2;                src += 2;                byteCount -= 2;            }            if (byteCount >= 1)            {                ((Byte1*)dest)[0] = ((Byte1*)src)[0];                dest += 1;                src += 1;                byteCount -= 1;            }        }        //-----------------------------------------------------------------------------------------//        //将字符串dst用字符src填充        private unsafe static void memcpy(string dst, char src)        {            fixed (char* ptr_dst = dst)            {                int len = dst.Length;                for (int i = 0; i < len; i++)                    ptr_dst[i] = src;            }        }        //将字符拷贝到dst指定index位置        private unsafe static void memcpy(string dst, char src, int index)        {            fixed (char* ptr = dst)                ptr[index] = src;        }        //将相同长度的src内容拷入dst        private unsafe static void memcpy(string dst, string src)        {            if (dst.Length != src.Length)                throw new InvalidOperationException("两个字符串参数长度不一致。");            fixed (char* dst_ptr = dst)            {                fixed (char* src_ptr = src)                {                    memcpy(dst_ptr, src_ptr, dst.Length);                }            }        }        //将src指定length内容拷入dst,dst下标src_offset偏移        private unsafe static void memcpy(char* dst, char* src, int length, int src_offset)        {            memcpy(dst + src_offset, src, length);        }        private unsafe static void memcpy(string dst, string src, int length, int src_offset)        {            fixed (char* ptr_dst = dst)            {                fixed (char* ptr_src = src)                {                    memcpy(ptr_dst + src_offset, ptr_src, length);                }            }        }        public class zstring_block : IDisposable        {            readonly Stack<zstring> stack;            internal zstring_block(int capacity)            {                stack = new Stack<zstring>(capacity);            }            internal void push(zstring str)            {                stack.Push(str);            }            internal IDisposable begin()//构造函数            {#if DBG                if (log != null)                    log("Began block");#endif                return this;            }            void IDisposable.Dispose()//析构函数            {#if DBG                if (log != null)                    log("Disposing block");#endif                while (stack.Count > 0)                {                    var str = stack.Pop();                    str.dispose();//循环调用栈中zstring的Dispose方法                }                zstring.g_blocks.Push(this);//将自身push入缓存栈                //赋值currentBlock                g_open_blocks.Pop();                if (g_open_blocks.Count > 0)                {                    zstring.g_current_block = g_open_blocks.Peek();                }                else                {                    zstring.g_current_block = null;                }            }        }        // Public API        #region         public static Action<string> Log = null;        public static uint DecimalAccuracy = 3; // 小数点后精度位数        //获取字符串长度        public int Length        {            get { return _value.Length; }        }        //类构造:cache_capacity缓存栈字典容量,stack_capacity缓存字符串栈容量,block_capacity缓存栈容量,intern_capacity缓存,open_capacity默认打开层数        public static void Initialize(int cache_capacity, int stack_capacity, int block_capacity, int intern_capacity, int open_capacity, int shallowCache_capacity)        {            g_cache = new Queue<zstring>[cache_capacity];            g_secCache = new Dictionary<int, Queue<zstring>>(cache_capacity);            g_blocks = new Stack<zstring_block>(block_capacity);            g_intern_table = new Dictionary<int, string>(intern_capacity);            g_open_blocks = new Stack<zstring_block>(open_capacity);            g_shallowCache = new Stack<zstring>(shallowCache_capacity);            for (int c = 0; c < cache_capacity; c++)            {                var stack = new Queue<zstring>(stack_capacity);                for (int j = 0; j < stack_capacity; j++)                    stack.Enqueue(new zstring(c));                g_cache[c] = stack;            }            for (int i = 0; i < block_capacity; i++)            {                var block = new zstring_block(block_capacity * 2);                g_blocks.Push(block);            }            for (int i = 0; i < shallowCache_capacity; i++)            {                g_shallowCache.Push(new zstring(null, true));            }        }        //using语法所用。从zstring_block栈中取出一个block并将其置为当前g_current_block,在代码块{}中新生成的zstring都将push入块内部stack中。当离开块作用域时,调用块的Dispose函数,将内栈中所有zstring填充初始值并放入zstring缓存栈。同时将自身放入block缓存栈中。(此处有个问题:使用Stack缓存block,当block被dispose放入Stack后g_current_block仍然指向此block,无法记录此block之前的block,这样导致zstring.Block()无法嵌套使用)        public static IDisposable Block()        {            if (g_blocks.Count == 0)                g_current_block = new zstring_block(INITIAL_BLOCK_CAPACITY * 2);            else                g_current_block = g_blocks.Pop();            g_open_blocks.Push(g_current_block);//新加代码,将此玩意压入open栈            return g_current_block.begin();        }        //将zstring value放入intern缓存表中以供外部使用        public string Intern()        {            //string interned = new string(NEW_ALLOC_CHAR, _value.Length);            //memcpy(interned, _value);            //return interned;            return __intern(_value);        }        //将string放入zstring intern缓存表中以供外部使用        public static string Intern(string value)        {            return __intern(value);        }        public static void Intern(string[] values)        {            for (int i = 0; i < values.Length; i++)                __intern(values[i]);        }        //下标取值函数        public char this[int i]        {            get { return _value[i]; }            set { memcpy(this, value, i); }        }        //获取hashcode        public override int GetHashCode()        {            return _value.GetHashCode();        }        //字面值比较        public override bool Equals(object obj)        {            if (obj == null)                return ReferenceEquals(this, null);            var gstr = obj as zstring;            if (gstr != null)                return gstr._value == this._value;            var str = obj as string;            if (str != null)                return str == this._value;            return false;        }        //转化为string        public override string ToString()        {            return _value;        }        //bool->zstring转换        public static implicit operator zstring(bool value)        {            return get(value ? "True" : "False");        }        // long - >zstring转换        public unsafe static implicit operator zstring(long value)        {            // e.g. 125            // first pass: count the number of digits            // then: get a zstring with length = num digits            // finally: iterate again, get the char of each digit, memcpy char to result            bool negative = value < 0;            value = Math.Abs(value);            int num_digits = get_digit_count(value);            zstring result;            if (negative)            {                result = get(num_digits + 1);                fixed (char* ptr = result._value)                {                    *ptr = '-';                    longcpy(ptr, value, 1, num_digits);                }            }            else            {                result = get(num_digits);                fixed (char* ptr = result._value)                    longcpy(ptr, value, 0, num_digits);            }            return result;        }        //int->zstring转换        public unsafe static implicit operator zstring(int value)        {            // e.g. 125            // first pass: count the number of digits            // then: get a zstring with length = num digits            // finally: iterate again, get the char of each digit, memcpy char to result            bool negative = value < 0;            value = Math.Abs(value);            int num_digits = get_digit_count(value);            zstring result;            if (negative)            {                result = get(num_digits + 1);                fixed (char* ptr = result._value)                {                    *ptr = '-';                    intcpy(ptr, value, 1, num_digits);                }            }            else            {                result = get(num_digits);                fixed (char* ptr = result._value)                    intcpy(ptr, value, 0, num_digits);            }            return result;        }        //float->zstring转换        public unsafe static implicit operator zstring(float value)        {            // e.g. 3.148            bool negative = value < 0;            if (negative) value = -value;            long mul = (long)Math.Pow(10, DecimalAccuracy);            long number = (long)(value * mul); // gets the number as a whole, e.g. 3148            int left_num = (int)(number / mul); // left part of the decimal point, e.g. 3            int right_num = (int)(number % mul); // right part of the decimal pnt, e.g. 148            int left_digit_count = get_digit_count(left_num); // e.g. 1            int right_digit_count = get_digit_count(right_num); // e.g. 3            //int total = left_digit_count + right_digit_count + 1; // +1 for '.'            int total = left_digit_count + (int)DecimalAccuracy + 1; // +1 for '.'            zstring result;            if (negative)            {                result = get(total + 1); // +1 for '-'                fixed (char* ptr = result._value)                {                    *ptr = '-';                    intcpy(ptr, left_num, 1, left_digit_count);                    *(ptr + left_digit_count + 1) = '.';                    int offest = (int)DecimalAccuracy - right_digit_count;                    for (int i = 0; i < offest; i++)                        *(ptr + left_digit_count + i + 1) = '0';                    intcpy(ptr, right_num, left_digit_count + 2 + offest, right_digit_count);                }            }            else            {                result = get(total);                fixed (char* ptr = result._value)                {                    intcpy(ptr, left_num, 0, left_digit_count);                    *(ptr + left_digit_count) = '.';                    int offest = (int)DecimalAccuracy - right_digit_count;                    for (int i = 0; i < offest; i++)                        *(ptr + left_digit_count + i + 1) = '0';                    intcpy(ptr, right_num, left_digit_count + 1 + offest, right_digit_count);                }            }            return result;        }        //string->zstring转换        public static implicit operator zstring(string value)        {            //return get(value);            return getShallow(value);        }        //string->zstring转换        public static zstring shallow(string value)        {            return getShallow(value);        }        //zstring->string转换        public static implicit operator string(zstring value)        {            return value._value;        }        //+重载        public static zstring operator +(zstring left, zstring right)        {            return internal_concat(left, right);        }        //==重载        public static bool operator ==(zstring left, zstring right)        {            if (ReferenceEquals(left, null))                return ReferenceEquals(right, null);            if (ReferenceEquals(right, null))                return false;            return left._value == right._value;        }        //!=重载        public static bool operator !=(zstring left, zstring right)        {            return !(left._value == right._value);        }        //转换为大写        public unsafe zstring ToUpper()        {            var result = get(Length);            fixed (char* ptr_this = this._value)            {                fixed (char* ptr_result = result._value)                {                    for (int i = 0; i < _value.Length; i++)                    {                        var ch = ptr_this[i];                        if (char.IsLower(ch))                            ptr_result[i] = char.ToUpper(ch);                        else                            ptr_result[i] = ptr_this[i];                    }                }            }            return result;        }        //转换为小写        public unsafe zstring ToLower()        {            var result = get(Length);            fixed (char* ptr_this = this._value)            {                fixed (char* ptr_result = result._value)                {                    for (int i = 0; i < _value.Length; i++)                    {                        var ch = ptr_this[i];                        if (char.IsUpper(ch))                            ptr_result[i] = char.ToLower(ch);                        else                            ptr_result[i] = ptr_this[i];                    }                }            }            return result;        }        //移除剪切        public zstring Remove(int start)        {            return Remove(start, Length - start);        }        //移除剪切        public zstring Remove(int start, int count)        {            return internal_remove(this._value, start, count);        }        //插入start起count长度字符        public zstring Insert(char value, int start, int count)        {            return internal_insert(this._value, value, start, count);        }        //插入start起字符串        public zstring Insert(string value, int start)        {            return internal_insert(this._value, value, start);        }        //子字符替换        public unsafe zstring Replace(char old_value, char new_value)        {            zstring result = get(Length);            fixed (char* ptr_this = this._value)            {                fixed (char* ptr_result = result._value)                {                    for (int i = 0; i < Length; i++)                    {                        ptr_result[i] = ptr_this[i] == old_value ? new_value : ptr_this[i];                    }                }            }            return result;        }        //子字符串替换        public zstring Replace(string old_value, string new_value)        {            return internal_replace(this._value, old_value, new_value);        }        //剪切start位置起后续子串        public zstring Substring(int start)        {            return Substring(start, Length - start);        }        //剪切start起count长度的子串        public unsafe zstring Substring(int start, int count)        {            if (start < 0 || start >= Length)                throw new ArgumentOutOfRangeException("start");            if (count > Length)                throw new ArgumentOutOfRangeException("count");            zstring result = get(count);            fixed (char* src = this._value)            fixed (char* dst = result._value)                memcpy(dst, src + start, count);            return result;        }        //子串包含判断        public bool Contains(string value)        {            return IndexOf(value) != -1;        }        //字符包含判断        public bool Contains(char value)        {            return IndexOf(value) != -1;        }        //子串第一次出现位置        public int LastIndexOf(string value)        {            int idx = -1;            int last_find = -1;            while (true)            {                idx = internal_index_of(this._value, value, idx + value.Length);                last_find = idx;                if (idx == -1 || idx + value.Length >= this._value.Length)                    break;            }            return last_find;        }        //字符第一次出现位置        public int LastIndexOf(char value)        {            int idx = -1;            int last_find = -1;            while (true)            {                idx = internal_index_of(this._value, value, idx + 1);                last_find = idx;                if (idx == -1 || idx + 1 >= this._value.Length)                    break;            }            return last_find;        }        //字符第一次出现位置        public int IndexOf(char value)        {            return IndexOf(value, 0, Length);        }        //字符自start起第一次出现位置        public int IndexOf(char value, int start)        {            return internal_index_of(this._value, value, start);        }        //字符自start起count长度内,        public int IndexOf(char value, int start, int count)        {            return internal_index_of(this._value, value, start, count);        }        //子串第一次出现位置        public int IndexOf(string value)        {            return IndexOf(value, 0, Length);        }        //子串自start位置起,第一次出现位置        public int IndexOf(string value, int start)        {            return IndexOf(value, start, Length - start);        }        //子串自start位置起,count长度内第一次出现位置        public int IndexOf(string value, int start, int count)        {            return internal_index_of(this._value, value, start, count);        }        //是否以某字符串结束        public unsafe bool EndsWith(string postfix)        {            if (postfix == null)                throw new ArgumentNullException("postfix");            if (this.Length < postfix.Length)                return false;            fixed (char* ptr_this = this._value)            {                fixed (char* ptr_postfix = postfix)                {                    for (int i = this._value.Length - 1, j = postfix.Length - 1; j >= 0; i--, j--)                        if (ptr_this[i] != ptr_postfix[j])                            return false;                }            }            return true;        }        //是否以某字符串开始        public unsafe bool StartsWith(string prefix)        {            if (prefix == null)                throw new ArgumentNullException("prefix");            if (this.Length < prefix.Length)                return false;            fixed (char* ptr_this = this._value)            {                fixed (char* ptr_prefix = prefix)                {                    for (int i = 0; i < prefix.Length; i++)                        if (ptr_this[i] != ptr_prefix[i])                            return false;                }            }            return true;        }        //获取某长度字符串缓存数量        public static int GetCacheCount(int length)        {            Queue<zstring> stack;            getStackInCache(length, out stack);            return stack.Count;        }        //自身+value拼接        public zstring Concat(zstring value)        {            return internal_concat(this, value);        }        //静态拼接方法簇        public static zstring Concat(zstring s0, zstring s1) { return s0 + s1; }        public static zstring Concat(zstring s0, zstring s1, zstring s2) { return s0 + s1 + s2; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3) { return s0 + s1 + s2 + s3; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3, zstring s4) { return s0 + s1 + s2 + s3 + s4; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3, zstring s4, zstring s5) { return s0 + s1 + s2 + s3 + s4 + s5; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3, zstring s4, zstring s5, zstring s6) { return s0 + s1 + s2 + s3 + s4 + s5 + s6; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3, zstring s4, zstring s5, zstring s6, zstring s7) { return s0 + s1 + s2 + s3 + s4 + s5 + s6 + s7; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3, zstring s4, zstring s5, zstring s6, zstring s7, zstring s8) { return s0 + s1 + s2 + s3 + s4 + s5 + s6 + s7 + s8; }        public static zstring Concat(zstring s0, zstring s1, zstring s2, zstring s3, zstring s4, zstring s5, zstring s6, zstring s7, zstring s8, zstring s9) { return s0 + s1 + s2 + s3 + s4 + s5 + s6 + s7 + s8 + s9; }        //静态格式化方法簇        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3, zstring arg4, zstring arg5, zstring arg6, zstring arg7, zstring arg8, zstring arg9)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            if (arg4 == null) throw new ArgumentNullException("arg4");            if (arg5 == null) throw new ArgumentNullException("arg5");            if (arg6 == null) throw new ArgumentNullException("arg6");            if (arg7 == null) throw new ArgumentNullException("arg7");            if (arg8 == null) throw new ArgumentNullException("arg8");            if (arg9 == null) throw new ArgumentNullException("arg9");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            g_format_args[4] = arg4;            g_format_args[5] = arg5;            g_format_args[6] = arg6;            g_format_args[7] = arg7;            g_format_args[8] = arg8;            g_format_args[9] = arg9;            return internal_format(input, 10);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3, zstring arg4, zstring arg5, zstring arg6, zstring arg7, zstring arg8)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            if (arg4 == null) throw new ArgumentNullException("arg4");            if (arg5 == null) throw new ArgumentNullException("arg5");            if (arg6 == null) throw new ArgumentNullException("arg6");            if (arg7 == null) throw new ArgumentNullException("arg7");            if (arg8 == null) throw new ArgumentNullException("arg8");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            g_format_args[4] = arg4;            g_format_args[5] = arg5;            g_format_args[6] = arg6;            g_format_args[7] = arg7;            g_format_args[8] = arg8;            return internal_format(input, 9);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3, zstring arg4, zstring arg5, zstring arg6, zstring arg7)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            if (arg4 == null) throw new ArgumentNullException("arg4");            if (arg5 == null) throw new ArgumentNullException("arg5");            if (arg6 == null) throw new ArgumentNullException("arg6");            if (arg7 == null) throw new ArgumentNullException("arg7");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            g_format_args[4] = arg4;            g_format_args[5] = arg5;            g_format_args[6] = arg6;            g_format_args[7] = arg7;            return internal_format(input, 8);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3, zstring arg4, zstring arg5, zstring arg6)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            if (arg4 == null) throw new ArgumentNullException("arg4");            if (arg5 == null) throw new ArgumentNullException("arg5");            if (arg6 == null) throw new ArgumentNullException("arg6");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            g_format_args[4] = arg4;            g_format_args[5] = arg5;            g_format_args[6] = arg6;            return internal_format(input, 7);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3, zstring arg4, zstring arg5)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            if (arg4 == null) throw new ArgumentNullException("arg4");            if (arg5 == null) throw new ArgumentNullException("arg5");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            g_format_args[4] = arg4;            g_format_args[5] = arg5;            return internal_format(input, 6);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3, zstring arg4)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            if (arg4 == null) throw new ArgumentNullException("arg4");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            g_format_args[4] = arg4;            return internal_format(input, 5);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2, zstring arg3)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            if (arg3 == null) throw new ArgumentNullException("arg3");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            g_format_args[3] = arg3;            return internal_format(input, 4);        }        public static zstring Format(string input, zstring arg0, zstring arg1, zstring arg2)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            if (arg2 == null) throw new ArgumentNullException("arg2");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            g_format_args[2] = arg2;            return internal_format(input, 3);        }        public static zstring Format(string input, zstring arg0, zstring arg1)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            if (arg1 == null) throw new ArgumentNullException("arg1");            g_format_args[0] = arg0;            g_format_args[1] = arg1;            return internal_format(input, 2);        }        public static zstring Format(string input, zstring arg0)        {            if (arg0 == null) throw new ArgumentNullException("arg0");            g_format_args[0] = arg0;            return internal_format(input, 1);        }        // 普通的float->string是隐式转换,小数点后只保留三位有效数字        // 对于更高精确度需求,隐式转换,可以修改静态变量DecimalAccuracy        // 显式转换使用此方法即可,函数结束DecimalAccuracy值和之前的一样        public static zstring FloatToZstring(float value, uint DecimalAccuracy)        {            uint oldValue = zstring.DecimalAccuracy;            zstring.DecimalAccuracy = DecimalAccuracy;            zstring target = (zstring)value;            zstring.DecimalAccuracy = oldValue;            return target;        }        //判空或长度        public static bool IsNullOrEmpty(zstring str)        {            return str == null || str.Length == 0;        }        //是否以value结束        public static bool IsPrefix(zstring str, string value)        {            return str.StartsWith(value);        }        //是否以value开始        public static bool isPostfix(zstring str, string postfix)        {            return str.EndsWith(postfix);        }        #endregion    }}
 |