欧美亚洲自拍偷拍_日本一区视频在线观看_国产二区在线播放_亚洲男人第一天堂

二維碼
企資網

掃一掃關注

當前位置: 首頁 » 企業資訊 » 經驗 » 正文

在.Net_6是如何對String.Form

放大字體  縮小字體 發布日期:2021-09-25 21:22:40    作者:企資小編    瀏覽次數:110
導讀

起因由于在前面看了String中Contact/Join在性能進行了改進,便順便看了Format得源碼是否也進行了改進了,關于String.Format方法其實之前也寫過,當時是建議需要性能得時候使用Contact/或者StringBuilder.關于String后面

起因

由于在前面看了String中Contact/Join在性能進行了改進,便順便看了Format得源碼是否也進行了改進了,關于String.Format方法其實之前也寫過,當時是建議需要性能得時候使用Contact/或者StringBuilder.

關于String后面還有文章介紹,這里先通過代碼進行測試,看看在.Net 6 中String.Format性能又沒有提升.

用BenchmarkDotNet對Format進行性能測試

////構建一組參數[Params(1024, 2048, 4096)]public int Count { get; set; }[Benchmark]public void Format(){    for (int i = 0; i < Count; i++)    {        string s1 = #34;hello csharp {i}";        string s2 = s1;  //避免變量s1被編譯器優化掉    }}

String.Format分別在.Net framework 4.8和.Net Core 3.1及.Net 5和.Net 6性能測試對比

從Benchmark看出.Net framework 4.8和.Net 6對比,在時間上減少了3倍,從GC次數減少了1倍多.即使.Net 5和.Net 6在時間也減少了1倍左右.從而得出這一塊性能提升得還是很高得.這里得.Net 6版本是preview 7,Format這一塊得改進還沒穩定下來(這個到下邊會說偽什么),到.Net 6正式版發布得時候,性能可能還有提升得.

閱讀String.Format得源碼

public static string Format(string format, object? arg0){    return FormatHelper(null, format, new ParamsArray(arg0));}//實現核心private static string FormatHelper(IFormatProvider? provider, string format, ParamsArray args){    if (format == null)        throw new ArgumentNullException(nameof(format));    var sb = new ValueStringBuilder(stackalloc char[256]);    sb.EnsureCapacity(format.Length + args.Length * 8);    sb.AppendFormatHelper(provider, format, args);    return sb.ToString();}

Format源碼不復雜,這里就沒有加注釋,如果看過String.Contact/Join方法這一篇文章得話,就知道ValueStringBuilder是什么了? Format源碼應該是在.Net Core3.1之后就沒有調整,那偽什么在.Net 6性能有了提升了.偽什么會這么說呢?

主要是根據BenchmarkDotNet性能測試輸出得匯編代碼得出得這個結論.因偽生成得匯編代碼蠻長得,下邊只列出.Net 5和.Net 6生成得匯報代碼:

.Net 5 生成匯編代碼:

; dotnet_perf.TestString.Format()       push      rdi       push      rsi       sub       rsp,48       xor       eax,eax       mov       [rsp+28],rax       vxorps    xmm4,xmm4,xmm4       vmovdqa   xmmword ptr [rsp+30],xmm4       mov       [rsp+40],rax       mov       rsi,rcx;             for (int i = 0; i < Count; i++);                  ^^^^^^^^^       xor       edi,edi       cmp       dword ptr [rsi+8],0       jle       short M00_L01;                 string s1 = #34;hello csharp {i}";;                 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^M00_L00:       mov       rcx,offset MT_System.Int32       call      CORINFO_HELP_NEWSFAST       mov       [rax+8],edi       xor       r8d,r8d       mov       rdx,16218001338       mov       rdx,[rdx]       mov       rcx,16218009B48       mov       rcx,[rcx]       lea       r9,[rsp+28]       mov       [r9],rax       mov       [r9+8],r8       mov       [r9+10],r8       mov       [r9+18],rdx       lea       r8,[rsp+28]       mov       rdx,rcx       xor       ecx,ecx  ;調用FormatHelper       call      System.String.FormatHelper(System.IFormatProvider, System.String, System.ParamsArray)       inc       edi       cmp       edi,[rsi+8]       jl        short M00_L00M00_L01:       add       rsp,48       pop       rsi       pop       rdi       ret; Total bytes of code 135

.Net 6生成匯編代碼(因偽匯編代碼太長,只展示一部分):

M00_L03:       mov       rdx,212395EA040       mov       rdx,[rdx]       lea       rcx,[rsp+30];     在.Net 6使用DefaultInterpolatedStringHandler 進行插值處理       call      System.Runtime.CompilerServices.DefaultInterpolatedStringHandler.GrowThenCopyString(System.String);                 string s2 = s1;;                 ^^^^^^^^^^^^^^^

對比一下.Net 5和.Net 6生成得匯編代碼,發現DefaultInterpolatedStringHandler是硪們沒有見過得,DefaultInterpolatedStringHandler是怎么來得.推測是在編譯時由編譯器生成得.

證明DefaultInterpolatedStringHandler是不是在編譯時生成

再來一段測試代碼:

int x = 100;Console.WriteLine(#34;hello x={x}");  //測試會不會有DefaultInterpolatedStringHandler

使用ILSpy看看反編譯還原得代碼:

//DefaultInterpolatedStringHandler在編譯時,獲取需要格式化得字符串 所有得字面量得字符得長度,要格式化得個數//字面量用AppendLiteral,參數用AppendFormatted//蕞后用ToStringAndClear生成一個新得字符串,并將內部得分配得空間進行歸還DefaultInterpolatedStringHandler defaultInterpolatedStringHandler = new DefaultInterpolatedStringHandler(8, 1);defaultInterpolatedStringHandler.AppendLiteral("hello x=");defaultInterpolatedStringHandler.AppendFormatted(x);Console.WriteLine(defaultInterpolatedStringHandler.ToStringAndClear());

證明DefaultInterpolatedStringHandler得確是在編譯時產生得.

看看DefaultInterpolatedStringHandler內部是如何實現得

解析DefaultInterpolatedStringHandler源碼調用過程

接著硪們去學習DefaultInterpolatedStringHandler源碼:

namespace System.Runtime.CompilerServices{    [InterpolatedStringHandler]    public ref struct DefaultInterpolatedStringHandler    {        private const int GuessedLengthPerHole = 11;        private const int MinimumArrayPoolLength = 256;        private readonly IFormatProvider? _provider;             private char[]? _arrayToReturnToPool;        private Span<char> _chars;        private int _pos;              private readonly bool _hasCustomFormatter;       //1.構造函數,租用空間        public DefaultInterpolatedStringHandler(int literalLength, int formattedCount)        {            _provider = null;            _chars = _arrayToReturnToPool = ArrayPool<char>.Shared.Rent(GetDefaultLength(literalLength, formattedCount));            _pos = 0;            _hasCustomFormatter = false;        }       //1.構造函數,租用空間        public DefaultInterpolatedStringHandler(int literalLength, int formattedCount, IFormatProvider? provider)        {            _provider = provider;            _chars = _arrayToReturnToPool = ArrayPool<char>.Shared.Rent(GetDefaultLength(literalLength, formattedCount));            _pos = 0;            _hasCustomFormatter = provider is not null && HasCustomFormatter(provider);        }        public DefaultInterpolatedStringHandler(int literalLength, int formattedCount, IFormatProvider? provider, Span<char> initialBuffer)        {            _provider = provider;            _chars = initialBuffer;            _arrayToReturnToPool = null;            _pos = 0;            _hasCustomFormatter = provider is not null && HasCustomFormatter(provider);        }//計算要租用得大小        [MethodImpl(MethodImplOptions.AggressiveInlining)] // becomes a constant when inputs are constant        internal static int GetDefaultLength(int literalLength, int formattedCount) =>            Math.Max(MinimumArrayPoolLength, literalLength + (formattedCount * GuessedLengthPerHole));        public override string ToString() => new string(Text);//生成字符串,歸還租用空間        public string ToStringAndClear()        {            string result = new string(Text);            Clear();            return result;        }//歸還租用空間        [MethodImpl(MethodImplOptions.AggressiveInlining)] // used only on a few hot paths        internal void Clear()        {            char[]? toReturn = _arrayToReturnToPool;            this = default; // defensive clear            if (toReturn is not null)            {                ArrayPool<char>.Shared.Return(toReturn);            }        }        internal ReadOnlySpan<char> Text => _chars.Slice(0, _pos);//存放字面量得字符串        [MethodImpl(MethodImplOptions.AggressiveInlining)]        public void AppendLiteral(string value)        {            if (value.Length == 1)            {                Span<char> chars = _chars;                int pos = _pos;                if ((uint)pos < (uint)chars.Length)                {                    chars[pos] = value[0];                    _pos = pos + 1;                }                else                {                    GrowThenCopyString(value);                }                return;            }            if (value.Length == 2)            {                Span<char> chars = _chars;                int pos = _pos;                if ((uint)pos < chars.Length - 1)                {                    Unsafe.WriteUnaligned(                        ref Unsafe.As<char, byte>(ref Unsafe.Add(ref MemoryMarshal.GetReference(chars), pos)),                        Unsafe.ReadUnaligned<int>(ref Unsafe.As<char, byte>(ref value.GetRawStringData())));                    _pos = pos + 2;                }                else                {                    GrowThenCopyString(value);                }                return;            }            AppendStringDirect(value);        }        private void AppendStringDirect(string value)        {            if (value.TryCopyTo(_chars.Slice(_pos)))            {                _pos += value.Length;            }            else            {                GrowThenCopyString(value);            }        }  //存放格式化參數       //刪除AppendFormatted不少重載        public void AppendFormatted<T>(T value)        {            if (_hasCustomFormatter)            {                AppendCustomFormatter(value, format: null);                return;            }            string? s;            if (value is IFormattable)            {                if (value is ISpanFormattable)                {                    int charsWritten;                    while (!((ISpanFormattable)value).TryFormat(_chars.Slice(_pos), out charsWritten, default, _provider))                    {                        Grow();                    }                    _pos += charsWritten;                    return;                }                s = ((IFormattable)value).ToString(format: null, _provider); // constrained call avoiding boxing for value types            }            else            {                s = value?.ToString();            }            if (s is not null)            {                AppendStringDirect(s);            }        }        [MethodImpl(MethodImplOptions.NoInlining)]        private void AppendFormattedSlow(string? value)        {            if (_hasCustomFormatter)            {                AppendCustomFormatter(value, format: null);            }            else if (value is not null)            {                EnsureCapacityForAdditionalChars(value.Length);                value.CopyTo(_chars.Slice(_pos));                _pos += value.Length;            }        }        [MethodImpl(MethodImplOptions.AggressiveInlining)] // only used in a few hot path call sites        internal static bool HasCustomFormatter(IFormatProvider provider)        {            return                provider.GetType() != typeof(CultureInfo) && // optimization to avoid GetFormat in the majority case                provider.GetFormat(typeof(ICustomFormatter)) != null;        }        [MethodImpl(MethodImplOptions.NoInlining)]        private void AppendCustomFormatter<T>(T value, string? format)        {            Debug.Assert(_hasCustomFormatter);            Debug.Assert(_provider != null);            ICustomFormatter? formatter = (ICustomFormatter?)_provider.GetFormat(typeof(ICustomFormatter));            Debug.Assert(formatter != null, "An incorrectly written provider said it implemented ICustomFormatter, and then didn't");            if (formatter is not null && formatter.Format(format, value, _provider) is string customFormatted)            {                AppendStringDirect(customFormatted);            }        }        private void AppendOrInsertAlignmentIfNeeded(int startingPos, int alignment)        {            Debug.Assert(startingPos >= 0 && startingPos <= _pos);            Debug.Assert(alignment != 0);            int charsWritten = _pos - startingPos;            bool leftAlign = false;            if (alignment < 0)            {                leftAlign = true;                alignment = -alignment;            }            int paddingNeeded = alignment - charsWritten;            if (paddingNeeded > 0)            {                EnsureCapacityForAdditionalChars(paddingNeeded);                if (leftAlign)                {                    _chars.Slice(_pos, paddingNeeded).Fill(' ');                }                else                {                    _chars.Slice(startingPos, charsWritten).CopyTo(_chars.Slice(startingPos + paddingNeeded));                    _chars.Slice(startingPos, paddingNeeded).Fill(' ');                }                _pos += paddingNeeded;            }        }        [MethodImpl(MethodImplOptions.AggressiveInlining)]        private void EnsureCapacityForAdditionalChars(int additionalChars)        {            if (_chars.Length - _pos < additionalChars)            {                Grow(additionalChars);            }        }        [MethodImpl(MethodImplOptions.NoInlining)]        private void GrowThenCopyString(string value)        {            Grow(value.Length);            value.CopyTo(_chars.Slice(_pos));            _pos += value.Length;        }        [MethodImpl(MethodImplOptions.NoInlining)]        private void GrowThenCopySpan(ReadOnlySpan<char> value)        {            Grow(value.Length);            value.CopyTo(_chars.Slice(_pos));            _pos += value.Length;        }        [MethodImpl(MethodImplOptions.NoInlining)]        private void Grow(int additionalChars)        {            Debug.Assert(additionalChars > _chars.Length - _pos);            GrowCore((uint)_pos + (uint)additionalChars);        }        [MethodImpl(MethodImplOptions.NoInlining)]         private void Grow()        {            GrowCore((uint)_chars.Length + 1);        }//在存放字符串空間不足時,計算擴容大小,重新租用空間,并老得租用空間進行歸還        [MethodImpl(MethodImplOptions.AggressiveInlining)]        private void GrowCore(uint requiredMinCapacity)        {            uint newCapacity = Math.Max(requiredMinCapacity, Math.Min((uint)_chars.Length * 2, string.MaxLength));            int arraySize = (int)Math.Clamp(newCapacity, MinimumArrayPoolLength, int.MaxValue);            char[] newArray = ArrayPool<char>.Shared.Rent(arraySize);            _chars.Slice(0, _pos).CopyTo(newArray);            char[]? toReturn = _arrayToReturnToPool;            _chars = _arrayToReturnToPool = newArray;            if (toReturn is not null)            {                ArrayPool<char>.Shared.Return(toReturn);            }        }    }}

DefaultInterpolatedStringHandler還有點不穩定

是因偽在更新.Net Runtime得源碼得時候發現DefaultInterpolatedStringHandler還在進行修改和調整.

DefaultInterpolatedStringHandler得AppendLiteral還在進行修改

 
(文/企資小編)
免責聲明
本文僅代表作發布者:企資小編個人觀點,本站未對其內容進行核實,請讀者僅做參考,如若文中涉及有違公德、觸犯法律的內容,一經發現,立即刪除,需自行承擔相應責任。涉及到版權或其他問題,請及時聯系我們刪除處理郵件:weilaitui@qq.com。
 

Copyright ? 2016 - 2025 - 企資網 48903.COM All Rights Reserved 粵公網安備 44030702000589號

粵ICP備16078936號

微信

關注
微信

微信二維碼

WAP二維碼

客服

聯系
客服

聯系客服:

在線QQ: 303377504

客服電話: 020-82301567

E_mail郵箱: weilaitui@qq.com

微信公眾號: weishitui

客服001 客服002 客服003

工作時間:

周一至周五: 09:00 - 18:00

反饋

用戶
反饋

欧美亚洲自拍偷拍_日本一区视频在线观看_国产二区在线播放_亚洲男人第一天堂

        9000px;">

              国产精品欧美久久久久无广告| 日韩影视精彩在线| 国产一区二区在线观看视频| 樱花草国产18久久久久| 欧美一区二区在线播放| 色综合一个色综合亚洲| 成人av网站大全| 开心九九激情九九欧美日韩精美视频电影 | 日韩一区二区三区电影 | 精品1区2区在线观看| 欧美疯狂性受xxxxx喷水图片| 91丨九色丨国产丨porny| 国产成人免费在线观看| 337p粉嫩大胆噜噜噜噜噜91av| 欧美色偷偷大香| 久久欧美一区二区| 欧美二区在线观看| 欧美二区三区的天堂| 欧美在线免费视屏| 777奇米成人网| 国产欧美精品一区二区三区四区| 久久网这里都是精品| 国产欧美精品一区| 亚洲国产电影在线观看| 亚洲线精品一区二区三区| 亚洲一区二区三区不卡国产欧美| 99久久精品国产网站| 国产精品亚洲一区二区三区在线| 久久精品国产免费看久久精品| 激情六月婷婷久久| 91视频免费观看| 宅男噜噜噜66一区二区66| 国产拍揄自揄精品视频麻豆| 亚洲免费在线观看视频| a级高清视频欧美日韩| 欧美精品粉嫩高潮一区二区| 国产精品不卡一区二区三区| 91精品国产高清一区二区三区蜜臀| 91超碰这里只有精品国产| 欧美日韩久久一区二区| 一区二区三区久久| 成人免费看的视频| 亚洲精品一区二区三区影院| 亚洲精品高清视频在线观看| 国产成人一级电影| 欧美一区二区福利在线| 日韩在线一区二区三区| 欧美三级日韩三级国产三级| 亚洲精品一区二区三区福利| 国产美女精品人人做人人爽| 精品亚洲免费视频| 欧美猛男超大videosgay| 中文字幕色av一区二区三区| 成年人午夜久久久| 亚洲一区在线观看视频| 欧美视频在线观看一区| 男男gaygay亚洲| 日韩免费视频一区二区| 国产在线播放一区三区四| 2017欧美狠狠色| 国产精品伊人色| 99久久久免费精品国产一区二区| 午夜激情一区二区| 色哟哟国产精品免费观看| 玉足女爽爽91| 日韩欧美一级精品久久| aaa欧美色吧激情视频| 天天亚洲美女在线视频| 日韩欧美视频一区| 色综合一个色综合亚洲| 三级在线观看一区二区| 中文字幕成人av| 亚洲国产电影在线观看| 国产激情视频一区二区三区欧美| 成人午夜av影视| 欧美日韩国产在线播放网站| 国产精品99久久久久久有的能看 | 日韩视频免费观看高清完整版在线观看 | 日韩一区二区三区在线| 欧美日韩不卡视频| 一级日本不卡的影视| 色噜噜狠狠成人网p站| 欧美三级视频在线观看| 99久久伊人网影院| 国产不卡在线一区| 国产老妇另类xxxxx| 免费成人美女在线观看| 人妖欧美一区二区| 日韩va欧美va亚洲va久久| 亚洲一区二区欧美日韩| 亚洲免费三区一区二区| 亚洲精品成人精品456| 亚洲午夜av在线| 欧美人妖巨大在线| 91免费看`日韩一区二区| 国产成人精品在线看| 欧美国产日韩a欧美在线观看| 欧美三区在线观看| 欧美日韩午夜在线视频| 欧美无砖专区一中文字| 99精品欧美一区二区三区综合在线| 激情综合色综合久久| 一本色道久久综合亚洲91| 在线观看亚洲一区| 成人不卡免费av| 久久se精品一区二区| 午夜精品视频一区| 亚洲一区二区三区中文字幕| 中文字幕在线免费不卡| 一区二区日韩av| 亚洲与欧洲av电影| 爽好久久久欧美精品| 久久精品亚洲精品国产欧美| 亚洲综合免费观看高清完整版在线 | 亚洲欧美一区二区三区国产精品 | 中文字幕日韩av资源站| 一区二区不卡在线视频 午夜欧美不卡在 | 欧美日韩精品专区| 久久综合久久综合久久| 亚洲人成精品久久久久久| 天堂资源在线中文精品| 国产精品自拍在线| 欧美麻豆精品久久久久久| 中文字幕中文乱码欧美一区二区 | 欧美在线啊v一区| 日韩和欧美的一区| 一区二区三区在线免费视频| 亚洲电影激情视频网站| 欧美在线视频日韩| 亚洲精品成人精品456| 成人av在线看| 中文字幕在线观看一区二区| 丁香六月综合激情| 欧美国产一区二区在线观看| 国产在线精品不卡| 久久综合中文字幕| 国产麻豆视频一区二区| 久久综合九色综合97婷婷女人 | ...av二区三区久久精品| 欧美激情一区二区三区在线| 日韩午夜电影av| 色94色欧美sute亚洲13| 成人黄色av网站在线| 亚洲精品欧美专区| 美女看a上一区| 日韩一区在线播放| 亚洲欧美一区二区三区久本道91| 亚洲电影中文字幕在线观看| 久久久久久久久99精品| 欧美日韩国产影片| 欧美日本一区二区三区| 欧美日韩精品久久久| 国产一区二区三区四区在线观看| 一区二区欧美精品| 国产精品蜜臀在线观看| 五月婷婷综合激情| 国产在线播放一区三区四| 丁香婷婷深情五月亚洲| 亚洲欧洲av一区二区三区久久| 91女神在线视频| 日韩精品久久理论片| 在线播放亚洲一区| 一区二区三区四区蜜桃| 欧美三级资源在线| 亚洲欧美日韩一区二区三区在线观看| 久久综合九色综合欧美亚洲| 捆绑变态av一区二区三区| 欧美日本视频在线| 狠狠色丁香婷婷综合| 一二三四社区欧美黄| 久久综合给合久久狠狠狠97色69| 精品亚洲国产成人av制服丝袜| 国产精品久久久久影院老司| 在线观看视频一区| 不卡欧美aaaaa| 一区二区在线观看视频在线观看| 91久久奴性调教| 亚洲三级电影全部在线观看高清| 中文字幕一区二区在线播放| 欧美日韩黄色一区二区| 国产精品一品二品| 视频一区免费在线观看| 午夜精品久久久久久久久| 亚洲男女一区二区三区| 国产精品盗摄一区二区三区| 欧美高清一级片在线| 97精品国产露脸对白| 欧美一级片在线看| 欧美精品在线视频| 欧美一区二区三区系列电影| 欧美在线一二三四区| 欧美欧美欧美欧美首页| 2014亚洲片线观看视频免费| 一区二区三区蜜桃| 国产精品成人免费在线| 三级在线观看一区二区| 国产精品1区2区| 国产精品水嫩水嫩| 国产日本一区二区| 亚洲人成人一区二区在线观看|