詳解C# Protobuf如何做到0分配內(nèi)存的序列化
題目很簡單, 就是IMessage對象怎么變成Byte[]
答案1:
msg.ToByteArray()
這肯定不符合我們的要求
答案2:
using var memoryStream = new MemoryStream(); using var codedOutputStream = new CodedOutputStream(memoryStream); msg.WriteTo(codedOutputStream); codedOutputStream.Flush(); memoryStream.ToArray();
這里面memoryStream, codedOutputStream, 還有ToArray都產(chǎn)生了一個(gè)對象, MemoryStream內(nèi)部還會多產(chǎn)生一個(gè)byte[]對象
不符合要求
答案3:
有人說你可以給MemoryStream傳遞一個(gè)byte[] slice, 讓MemoryStream直接用byte[]
var bytes = new byte[msg.CalculateSize()]; using var memoryStream = new MemoryStream(); using var codedOutputStream = new CodedOutputStream(memoryStream); msg.WriteTo(codedOutputStream); codedOutputStream.Flush();
這次消息直接被序列化到bytes里面去了, 但是memoryStream對象, codecOutputStream還有memoryStream內(nèi)部的byte[]都還在, 我就序列化了一個(gè)對象, 卻產(chǎn)生了3個(gè)垃圾對象
所以, 來仔細(xì)看看CodedOutputStream類:
/// <summary> /// Creates a new CodedOutputStream that writes directly to the given /// byte array. If more bytes are written than fit in the array, /// OutOfSpaceException will be thrown. /// </summary> public CodedOutputStream(byte[] flatArray) : this(flatArray, 0, flatArray.Length) { } /// <summary> /// Creates a new CodedOutputStream that writes directly to the given /// byte array slice. If more bytes are written than fit in the array, /// OutOfSpaceException will be thrown. /// </summary> private CodedOutputStream(byte[] buffer, int offset, int length) { this.output = null; this.buffer = buffer; this.position = offset; this.limit = offset + length; leaveOpen = true; // Simple way of avoiding trying to dispose of a null reference }
提供了一個(gè)byte[]的構(gòu)造函數(shù), 但是沒提供slice的構(gòu)造函數(shù), 好在有一個(gè)私有的構(gòu)造函數(shù)
答案4:
這邊就不寫代碼了, 大概意思就是通過反射私有構(gòu)造函數(shù)來構(gòu)造一個(gè)CodedOutputStream對象, 來省掉MemoryStream和他內(nèi)部的byte[]
現(xiàn)在離答案已經(jīng)比較接近了
那我們的問題是, 能不能連CodedOutputStream也省掉呢?
答案5來了:
經(jīng)過仔細(xì)觀察, 發(fā)現(xiàn)這個(gè)類沒有使用Stream的情況下, 就只需要修改buffer, limit, 和position幾個(gè)成員就行了, 雖然是private成員, 但是C#還是能修改
下來立馬實(shí)踐
delegate void ClearCodedOutputStream(CodedOutputStream stream, byte[] buffer, int offset, int count); static ClearCodedOutputStream ResetCodedOutputStream; static CodedOutputStream codedOutputStream = new CodedOutputStream(new byte[10]); static unsafe void Encode(IMessage msg, byte[] buffer) { ResetCodedOutputStream(codedOutputStream, buffer, 0, buffer.Length); msg.WriteTo(codedOutputStream); codedOutputStream.Flush(); } static Action<T, TValue> MakeSetter<T, TValue>(FieldInfo field) { DynamicMethod m = new DynamicMethod( "setter", typeof(void), new Type[] { typeof(T), typeof(TValue) }, typeof(Program)); ILGenerator cg = m.GetILGenerator(); cg.Emit(OpCodes.Ldarg_0); cg.Emit(OpCodes.Ldarg_1); cg.Emit(OpCodes.Stfld, field); cg.Emit(OpCodes.Ret); return (Action<T, TValue>)m.CreateDelegate(typeof(Action<T, TValue>)); } static void Main(string[] args) { var bufferField = typeof(CodedOutputStream).GetField("buffer", BindingFlags.NonPublic | BindingFlags.Instance); var limitField = typeof(CodedOutputStream).GetField("limit", BindingFlags.NonPublic | BindingFlags.Instance); var positionField = typeof(CodedOutputStream).GetField("position", BindingFlags.NonPublic | BindingFlags.Instance); var setLimit = MakeSetter<CodedOutputStream, int>(limitField); var setPosition = MakeSetter<CodedOutputStream, int>(positionField); var setBuffer = MakeSetter<CodedOutputStream, byte[]>(bufferField); ResetCodedOutputStream = (stream, buffer, offset, length) => { //this.buffer = buffer; //this.position = offset; //this.limit = offset + length; setBuffer(stream, buffer); setPosition(stream, offset); setLimit(stream, offset + length); }; var buffer = new byte[msg.CalculateSize()]; Encode(msg, buffer); }
這個(gè)實(shí)例代碼里面, 用了一個(gè)static的全局CodedOutputStream, 真正用的時(shí)候, 肯定要保證線程安全.
所以接下來的問題是:
1. 如何保證CodedOutputStream對象線程安全
2. 如何把var buffer = new byte[msg.CalculateSize()];這個(gè)也省掉
這倆問題就留給讀者思考.
Github: http://github.com/egmkang
到此這篇關(guān)于詳解C# Protobuf如何做到0分配內(nèi)存的序列化的文章就介紹到這了,更多相關(guān)C# Protobuf 序列化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C#中序列化實(shí)現(xiàn)深拷貝,實(shí)現(xiàn)DataGridView初始化刷新的方法
下面小編就為大家?guī)硪黄狢#中序列化實(shí)現(xiàn)深拷貝,實(shí)現(xiàn)DataGridView初始化刷新的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-02-02C#預(yù)定義數(shù)據(jù)類型之值類型和引用類型介紹
這篇文章主要介紹了C#預(yù)定義數(shù)據(jù)類型之值類型和引用類型介紹,本文著重講解了引用類型中的object(對象)類型和string(字符串)類型,需要的朋友可以參考下2015-03-03DevExpress之SplashScreen用法實(shí)例
這篇文章主要介紹了DevExpress中SplashScreen的用法,對于C#初學(xué)者有很好的參考借鑒價(jià)值,需要的朋友可以參考下2014-08-08基于C#?實(shí)現(xiàn)劉謙春晚魔術(shù)(示例代碼)
劉謙春晚魔術(shù)是一個(gè)讓人嘆為觀止的魔術(shù)表演,其中涉及到了數(shù)學(xué)、編程和創(chuàng)意的結(jié)合,看了春晚魔術(shù)的朋友們,是不是好奇春晚劉謙的魔術(shù)是怎么變的,本文分享C#?實(shí)現(xiàn)劉謙春晚魔術(shù)示例代碼,一起看看吧2024-02-02WPF+SkiaSharp實(shí)現(xiàn)自繪投籃小游戲
這篇文章主要介紹了如何利用WPF+SkiaSharp實(shí)現(xiàn)自繪投籃小游戲。此案例主要是針對光線投影法碰撞檢測功能的示例,順便做成了一個(gè)小游戲,很簡單,但是,效果卻很不錯(cuò),感興趣的可以動(dòng)手嘗試一下2022-08-08