日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當(dāng)前位置:首頁 > 科技  > 軟件

Linq 查詢的結(jié)果會開辟新的內(nèi)存嗎?

來源: 責(zé)編: 時間:2023-10-27 17:22:15 421觀看
導(dǎo)讀一:背景1. 講故事圖片昨天群里有位朋友問:linq 查詢的結(jié)果會開辟新的內(nèi)存嗎?如果開了,那是對原序列集里面元素的深拷貝還是僅僅拷貝其引用?其實這個問題我覺得問的挺好,很多初學(xué) C# 的朋友或多或少都有這樣的疑問,甚至有 3,4

一:背景

1. 講故事

圖片圖片9pL28資訊網(wǎng)——每日最新資訊28at.com

昨天群里有位朋友問:linq 查詢的結(jié)果會開辟新的內(nèi)存嗎?如果開了,那是對原序列集里面元素的深拷貝還是僅僅拷貝其引用?9pL28資訊網(wǎng)——每日最新資訊28at.com

其實這個問題我覺得問的挺好,很多初學(xué) C# 的朋友或多或少都有這樣的疑問,甚至有 3,4 年工作經(jīng)驗的朋友可能都不是很清楚,這就導(dǎo)致在寫代碼的時候總是會畏手畏腳,還會莫名的揪心這樣玩的話內(nèi)存會不會暴漲暴跌,這一篇我就用 windbg 來幫助朋友徹底分析一下。9pL28資訊網(wǎng)——每日最新資訊28at.com

二:尋找答案

1. 一個小案例

這位老弟提到了是深拷貝還是淺拷貝,本意就是想問:linq 一個引用類型集合 到底會怎樣? 這里我先模擬一個集合,代碼如下:9pL28資訊網(wǎng)——每日最新資訊28at.com

class Program    {        static void Main(string[] args)        {            var personList = new List<Person>() {                                              new Person() { Name="jack", Age=20 },                                              new Person() { Name="elen",Age=25,  },                                              new Person() {  Name="john", Age=22 }                                            };            var query = personList.Where(m => m.Age > 20).ToList();            Console.WriteLine($"query.count={query.Count}");            Console.ReadLine();        }    }    class Person    {        public string Name { get; set; }        public int Age { get; set; }    }

圖片圖片9pL28資訊網(wǎng)——每日最新資訊28at.com

2. 真的是深copy嗎?

如果用 windbg 的話,就非常簡單了,假設(shè)是深copy 的話,那么 query 之后,托管堆上就會有 5個 Person,那是不是這樣呢?用 !dumpheap -stat -type Person 到托管堆驗證一下即可。9pL28資訊網(wǎng)——每日最新資訊28at.com

0:000> !dumpheap -stat -type PersonStatistics:              MT    Count    TotalSize Class Name00007ff7f27c3528        1           64 System.Func`2[[ConsoleApp5.Person, ConsoleApp5],[System.Boolean, System.Private.CoreLib]]00007ff7f27c2b60        2           64 System.Collections.Generic.List`1[[ConsoleApp5.Person, ConsoleApp5]]00007ff7f27c9878        1           72 System.Linq.Enumerable+WhereListIterator`1[[ConsoleApp5.Person, ConsoleApp5]]00007ff7f27c7a10        3          136 ConsoleApp5.Person[]00007ff7f27c2ad0        3           96 ConsoleApp5.Person

從最后一行輸出可以看到: ConsoleApp5.Person 的 Count=3,也就表明沒有所謂的深copy,如果你還不信的話,可以在 query 中修改某一個Person的Age,看看原始的 personList 集合是不是同步更新,修改代碼如下:9pL28資訊網(wǎng)——每日最新資訊28at.com

static void Main(string[] args)        {            var personList = new List<Person>() {                                              new Person() { Name="jack", Age=20 },                                              new Person() { Name="elen",Age=25,  },                                              new Person() {  Name="john", Age=22 }                                            };            var query = personList.Where(m => m.Age > 20).ToList();            //故意修改 Age=25 為  Age=100;             query[0].Age = 100;            Console.WriteLine($"query[0].Age={query[0].Age}, personList[2].Age={personList[1].Age}");            Console.ReadLine();        }

圖片圖片9pL28資訊網(wǎng)——每日最新資訊28at.com

從截圖來看更加驗證了 并沒有所謂的 深copy 一說。9pL28資訊網(wǎng)——每日最新資訊28at.com

3. 真的是 copy 引用嗎?

要驗證是不是 copy 引用,最粗暴的方法就是看看 query 這個數(shù)組在 托管堆上的存儲行態(tài)就明白了,同樣你也可以借助 windbg 去驗證一下,先到線程棧去找 query 變量,然后用 da 命令 對 query 進(jìn)行打印。9pL28資訊網(wǎng)——每日最新資訊28at.com

0:000> !clrstack -lOS Thread Id: 0x809c (0)        Child SP               IP Call Site000000E143D7E9B0 00007ff7f26f18be ConsoleApp5.Program.Main(System.String[]) [E:/net5/ConsoleApp5/ConsoleApp5/Program.cs @ 20]    LOCALS:        0x000000E143D7EA38 = 0x00000218266aab70        0x000000E143D7EA30 = 0x00000218266aad980:000> !do 0x00000218266aad98Name:        System.Collections.Generic.List`1[[ConsoleApp5.Person, ConsoleApp5]]MethodTable: 00007ff7f27b2b60EEClass:     00007ff7f27abad0Size:        32(0x20) bytesFile:        C:/Program Files/dotnet/shared/Microsoft.NETCore.App/3.1.9/System.Private.CoreLib.dllFields:              MT    Field   Offset                 Type VT     Attr            Value Name0000000000000000  4001c35        8              SZARRAY  0 instance 00000218266aadb8 _items00007ff7f26bb1f0  4001c36       10         System.Int32  1 instance                2 _size00007ff7f26bb1f0  4001c37       14         System.Int32  1 instance                2 _version0000000000000000  4001c38        8              SZARRAY  0   static dynamic statics NYI                 s_emptyArray0:000> !da 00000218266aadb8Name:        ConsoleApp5.Person[]MethodTable: 00007ff7f27b7a10EEClass:     00007ff7f26b6580Size:        56(0x38) bytesArray:       Rank 1, Number of elements 4, Type CLASSElement Methodtable: 00007ff7f27b2ad0[0] 00000218266aac00[1] 00000218266aac20[2] null[3] null

從最后四行代碼可以看出數(shù)組有 4 個格子,前2個格子放的是內(nèi)存地址,后兩個都是 null,可能有些朋友會問,query 不是 2 條記錄嗎?怎么會有 4 個格子呢?這是因為 query 是 List 結(jié)構(gòu),而 List 底層用的是數(shù)組,默認(rèn)以 4 個格子起步,不信的話翻一下 List 原代碼即可。9pL28資訊網(wǎng)——每日最新資訊28at.com

public class List<T>    {        private void EnsureCapacity(int min)        {            if (_items.Length < min)            {                int num = (_items.Length == 0) ? 4 : (_items.Length * 2);   //默認(rèn) 4 個大小                if ((uint)num > 2146435071u)                {                    num = 2146435071;                }                if (num < min)                {                    num = min;                }                Capacity = num;            }        }    }

如果你想進(jìn)一步查看數(shù)組中前兩個元素 00000218266aac00, 00000218266aac20 指向的是什么,可以用 !do 打印一下即可。9pL28資訊網(wǎng)——每日最新資訊28at.com

0:000> !do 00000218266aac00Name:        ConsoleApp5.PersonMethodTable: 00007ff7f27b2ad0EEClass:     00007ff7f27c2a00Size:        32(0x20) bytesFile:        E:/net5/ConsoleApp5/ConsoleApp5/bin/Debug/netcoreapp3.1/ConsoleApp5.dllFields:              MT    Field   Offset                 Type VT     Attr            Value Name00007ff7f2771e18  4000001        8        System.String  0 instance 00000218266aab30 <Name>k__BackingField00007ff7f26bb1f0  4000002       10         System.Int32  1 instance               25 <Age>k__BackingField0:000> !do 00000218266aac20Name:        ConsoleApp5.PersonMethodTable: 00007ff7f27b2ad0EEClass:     00007ff7f27c2a00Size:        32(0x20) bytesFile:        E:/net5/ConsoleApp5/ConsoleApp5/bin/Debug/netcoreapp3.1/ConsoleApp5.dllFields:              MT    Field   Offset                 Type VT     Attr            Value Name00007ff7f2771e18  4000001        8        System.String  0 instance 00000218266aab50 <Name>k__BackingField00007ff7f26bb1f0  4000002       10         System.Int32  1 instance               22 <Age>k__BackingField

到這里為止,我覺得回答這位朋友的疑問應(yīng)該是沒有問題了,不過這里既然說到了集合中的引用類型,不得不說一下集合中的值類型又會是怎么樣的?9pL28資訊網(wǎng)——每日最新資訊28at.com

三:集合中的值類型是什么樣的copy方式

1. 使用 windbg 驗證

有了上面的基礎(chǔ),驗證這個問題的答案就簡單了,先上測試代碼9pL28資訊網(wǎng)——每日最新資訊28at.com

static void Main(string[] args)        {            var list = new List<int>() { 1, 2, 3, 4, 5, 6, 7,8,9,10 };            var query = list.Where(m => m > 5).ToList();            Console.ReadLine();        }

然后直接把整個數(shù)組內(nèi)容打印出來9pL28資訊網(wǎng)——每日最新資訊28at.com

// list0:000> !DumpArray /d 0000019687c8aba8Name:        System.Int32[]MethodTable: 00007ff7f279f090EEClass:     00007ff7f279f010Size:        88(0x58) bytesArray:       Rank 1, Number of elements 16, Type Int32Element Methodtable: 00007ff7f26cb1f0[0] 0000019687c8abb8[1] 0000019687c8abbc[2] 0000019687c8abc0[3] 0000019687c8abc4[4] 0000019687c8abc8[5] 0000019687c8abcc[6] 0000019687c8abd0[7] 0000019687c8abd4[8] 0000019687c8abd8[9] 0000019687c8abdc[10] 0000019687c8abe0[11] 0000019687c8abe4[12] 0000019687c8abe8[13] 0000019687c8abec[14] 0000019687c8abf0[15] 0000019687c8abf4// query0:000> !DumpArray /d 0000019687c8ae68Name:        System.Int32[]MethodTable: 00007ff7f279f090EEClass:     00007ff7f279f010Size:        56(0x38) bytesArray:       Rank 1, Number of elements 8, Type Int32Element Methodtable: 00007ff7f26cb1f0[0] 0000019687c8ae78[1] 0000019687c8ae7c[2] 0000019687c8ae80[3] 0000019687c8ae84[4] 0000019687c8ae88[5] 0000019687c8ae8c[6] 0000019687c8ae90[7] 0000019687c8ae94

仔細(xì)對比 list 和 query 的數(shù)組呈現(xiàn),發(fā)現(xiàn)有兩點好玩的信息:9pL28資訊網(wǎng)——每日最新資訊28at.com

  • 值類型和引用類型一樣,數(shù)組中都是存放地址的。
  • 值類型數(shù)組中的所有格子都被填滿,不像引用類型數(shù)組中還有 null 的情況。

接下來的問題是,數(shù)組中每個元素的地址到底指向了誰,可以挑出每個數(shù)組的 0 號元素地址,用 dp 命令看一看:9pL28資訊網(wǎng)——每日最新資訊28at.com

//list0:000> dp 0000019687c8abb800000196`87c8abb8  00000002`00000001 00000004`0000000300000196`87c8abc8  00000006`00000005 00000008`0000000700000196`87c8abd8  0000000a`00000009 00000000`00000000//query0:000> dp 0000019687c8ae7800000196`87c8ae78  00000007`00000006 00000009`0000000800000196`87c8ae88  00000000`0000000a 00000000`00000000

看到?jīng)]有,原來地址上面存放的都是數(shù)字值,深copy無疑哈。9pL28資訊網(wǎng)——每日最新資訊28at.com

四:總結(jié)

以上所有的分析可以得出:引用類型數(shù)組是引用copy,值類型數(shù)組是深copy,有時候背誦得來的東西總是容易忘記,只有實操驗證才能真正的刻骨銘心!9pL28資訊網(wǎng)——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-15458-0.htmlLinq 查詢的結(jié)果會開辟新的內(nèi)存嗎?

聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 徹底理解C語言中的指針

下一篇: 訪問者模式:對象結(jié)構(gòu)的元素處理

標(biāo)簽:
  • 熱門焦點
  • Find N3入網(wǎng):最高支持16+1TB

    OPPO將于近期登場的Find N3折疊屏目前已經(jīng)正式入網(wǎng),型號為PHN110。本次Find N3在外觀方面相比前兩代有很大的變化,不再是小號的橫向折疊屏,而是跟別的廠商一樣采用了較為常見的
  • 5月安卓手機好評榜:魅族20 Pro奪冠

    性能榜和性價比榜之后,我們來看最后的安卓手機好評榜,數(shù)據(jù)來源安兔兔評測,收集時間2023年5月1日至5月31日,僅限國內(nèi)市場。第一名:魅族20 Pro好評率:97.50%不得不感慨魅族老品牌還
  • 5月iOS設(shè)備好評榜:iPhone 14僅排第43?

    來到新的一月,安兔兔的各個榜單又重新匯總了數(shù)據(jù),像安卓陣營的榜單都有著比較大的變動,不過iOS由于設(shè)備的更新?lián)Q代并沒有那么快,所以相對來說變化并不大,特別是iOS好評榜,老款設(shè)
  • 線程通訊的三種方法!通俗易懂

    線程通信是指多個線程之間通過某種機制進(jìn)行協(xié)調(diào)和交互,例如,線程等待和通知機制就是線程通訊的主要手段之一。 在 Java 中,線程等待和通知的實現(xiàn)手段有以下幾種方式:Object 類下
  • 2023年,我眼中的字節(jié)跳動

    此時此刻(2023年7月),字節(jié)跳動從未上市,也從未公布過任何官方的上市計劃;但是這并不妨礙它成為中國最受關(guān)注的互聯(lián)網(wǎng)公司之一。從2016-17年的抖音強勢崛起,到2018年的&ldquo;頭騰
  • 華為將推出盤古數(shù)字人大模型 可幫助用戶12小時完成數(shù)字人生成

    在今日舉行的2023年華為云數(shù)字文娛AI創(chuàng)新峰會上,華為云全球Marketing與銷售服務(wù)總裁石冀琳表示,華為云將在后續(xù)推出盤古數(shù)字人大模型,可幫助用戶12小
  • 2納米決戰(zhàn)2025

    集微網(wǎng)報道 從三強爭霸到四雄逐鹿,2nm的廝殺聲已然隱約傳來。無論是老牌勁旅臺積電、三星,還是誓言重回先進(jìn)制程領(lǐng)先地位的英特爾,甚至初成立不久的新
  • OPPO K11搭載長壽版100W超級閃充:26分鐘充滿100%

    據(jù)此前官方宣布,OPPO將于7月25日也就是今天下午14:30舉辦新品發(fā)布會,屆時全新的OPPO K11將正式與大家見面,將主打旗艦影像,和同檔位競品相比,其最大的賣
  • 聯(lián)想的ThinkBook Plus下一版曝光,鍵盤旁邊塞個平板

    ThinkBook Plus 是聯(lián)想的一個特殊筆記本類別,它在封面放入了一塊墨水屏,也給人留下了較為深刻的印象。據(jù)有人爆料,聯(lián)想的下一款 ThinkBook Plus 可能更特殊,它
Top 主站蜘蛛池模板: 黔西| 什邡市| 老河口市| 浮梁县| 永胜县| 福海县| 基隆市| 灵山县| 南开区| 镇雄县| 丰宁| 永清县| 宁德市| 寻甸| 张家川| 耒阳市| 西畴县| 普兰店市| 仁化县| 乌鲁木齐县| 信宜市| 永州市| 临桂县| 临武县| 马龙县| 桐梓县| 资中县| 台山市| 淳安县| 北安市| 香格里拉县| 沧州市| 新津县| 松阳县| 海兴县| 南安市| 屏边| 凉城县| 洱源县| 建湖县| 南宁市|