成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

ArrayPool 源碼解讀之 byte[] 也能池化?

開發(fā) 前端
學(xué)習(xí)這其中的 池化架構(gòu)? 思想,對平時項目開發(fā)還是能提供一些靈感的,其次對那些一次性使用 byte[] 的場景,用池化是個非常不錯的方法,這也是我對朋友dump分析后提出的一個優(yōu)化思路。

一:背景

1. 講故事

最近在分析一個 dump 的過程中發(fā)現(xiàn)其在 gen2 和 LOH 上有不少size較大的free,仔細看了下,這些free生前大多都是模板引擎生成的html片段的byte[]數(shù)組,當(dāng)然這篇我不是來分析dump的,而是來聊一下,當(dāng)托管堆有很多l(xiāng)ength較大的 byte[] 數(shù)組時,如何讓內(nèi)存利用更高效,如何讓gc老先生壓力更小。

不知道大家有沒有發(fā)現(xiàn)在 .netcore 中增加了不少池化對象的東西,比如:ArrayPool,ObjectPool 等等,確實在某些場景下還是特別實用的,所以有必要對其進行較深入的理解。

二:ArrayPool 源碼分析

1. 一圖勝千言

在我花了將近一個小時的源碼閱讀之后,我畫了一張 ArrayPool 的池化圖,所謂:一圖在手,天下我有 。

圖片

有了這張圖,接下來再聊幾個概念并配上相應(yīng)源碼,我覺得應(yīng)該就差不多了。

2. 池化的架構(gòu)分級是什么樣的?

ArrayPool 是由若干個 Bucket 組成, 而 Bucket 又由若干個 buffer[] 數(shù)組組成, 有了這個概念之后,再配一下代碼。

public abstract class ArrayPool<T>
{
    public static ArrayPool<T> Create()
    {
        return new ConfigurableArrayPool<T>();
    }
}

internal sealed class ConfigurableArrayPool<T> : ArrayPool<T>
{
    private sealed class Bucket
    {
        internal readonly int _bufferLength;
        private readonly T[][] _buffers;
        private int _index;
    }

    private readonly Bucket[] _buckets;     //bucket數(shù)組
}

3. 為什么每一個 bucket 里都有 50 個 buffer[]

這個問題很好回答,初始化時做了 maxArraysPerBucket=50 設(shè)定,當(dāng)然你也可以自定義,具體參考如下代碼:

internal sealed class ConfigurableArrayPool<T> : ArrayPool<T>
{
    internal ConfigurableArrayPool() : this(1048576, 50)
    {
    }

    internal ConfigurableArrayPool(int maxArrayLength, int maxArraysPerBucket)
    {
        int num = Utilities.SelectBucketIndex(maxArrayLength);
        Bucket[] array = new Bucket[num + 1];
        for (int i = 0; i < array.Length; i++)
        {
            array[i] = new Bucket(Utilities.GetMaxSizeForBucket(i), maxArraysPerBucket, id);
        }
        _buckets = array;
    }
}

4.  bucket 中 buffer[].length 為什么依次是 16,32,64 ...

框架做了默認假定,第一個bucket中的 buffer[].length=16, 后續(xù) bucket 中的 buffer[].length 都是 x2 累計,涉及到代碼就是 GetMaxSizeForBucket() 方法,參考如下:

internal ConfigurableArrayPool(int maxArrayLength, int maxArraysPerBucket)
{
    Bucket[] array = new Bucket[num + 1];
    for (int i = 0; i < array.Length; i++)
    {
        array[i] = new Bucket(Utilities.GetMaxSizeForBucket(i), maxArraysPerBucket, id);
    }
}

internal static int GetMaxSizeForBucket(int binIndex)
{
    return 16 << binIndex;
}

5. 初始化時 bucket 到底有多少個?

其實在上圖中我也沒有給出 bucket 到底有多少個,那到底是多少個呢??????? ,當(dāng)我閱讀完源碼之后,這算法還挺有意思的。

先說一下結(jié)果吧,默認 17 個 bucket,你肯定會好奇怎么算的?先說下兩個變量:

  • maxArrayLength=1048576 = 2的20次方
  • buffer.length= 16 = 2的4次方

最后的算法就是取次方的差值:bucket[].length= 20 - 4 + 1 = 17,換句話說最后一個 bucket 下的 buffer[].length=1048576,詳細代碼請參考 SelectBucketIndex() 方法。

internal sealed class ConfigurableArrayPool<T> : ArrayPool<T>
{
    internal ConfigurableArrayPool(): this(1048576, 50)
    { }

    internal ConfigurableArrayPool(int maxArrayLength, int maxArraysPerBucket)
    {
        int num = Utilities.SelectBucketIndex(maxArrayLength);
        Bucket[] array = new Bucket[num + 1];
        for (int i = 0; i < array.Length; i++)
        {
            array[i] = new Bucket(Utilities.GetMaxSizeForBucket(i), maxArraysPerBucket, id);
        }
        _buckets = array;
    }

    internal static int SelectBucketIndex(int bufferSize)
    {
        return BitOperations.Log2((uint)(bufferSize - 1) | 0xFu) - 3;
    }
}

到這里我相信你對 ArrayPool 的池化架構(gòu)思路已經(jīng)搞明白了,接下來看下如何申請和歸還 buffer[]。

三:如何申請和歸還

既然 buffer[] 做了顆粒化,那就應(yīng)該好借好還,反應(yīng)到代碼上就是 Rent() 和 Return() 方法,為了方便理解,上代碼說話:

class Program
    {
        static void Main(string[] args)
        {
            var arrayPool = ArrayPool<int>.Create();

            var bytes = arrayPool.Rent(10);

            for (int i = 0; i < bytes.Length; i++) bytes[i] = 10;

            arrayPool.Return(bytes);

            Console.ReadLine();
        }
    }

圖片圖片

圖片圖片

有了代碼和圖之后,再稍微捋一下流程。

  1. 從 ArrayPool 中借一個 byte[10] 大小的數(shù)組,為了節(jié)省內(nèi)存,先不備貨,臨時生成一個 byte[].size=16 的數(shù)組出來,簡化后的代碼如下,參考 if (flag) 處:
internal T[] Rent()
    {
        T[][] buffers = _buffers;
        T[] array = null;
        bool lockTaken = false;
        bool flag = false;
        try
        {
            if (_index < buffers.Length)
            {
                array = buffers[_index];
                buffers[_index++] = null;
                flag = array == null;
            }
        }
        if (flag)
        {
            array = new T[_bufferLength];
        }
        return array;
    }

這里有一個坑,那就是你以為借了 byte[10],現(xiàn)實給你的是 byte[16],這里稍微注意一下。

  1. 當(dāng)用 ArrayPool.Return 歸還 byte[16] 時, 很明顯看到它落到了第一個bucket的第一個buffer[]上,參考如下簡化后的代碼:
internal void Return(T[] array)
    {
        if (_index != 0)
        {
            _buffers[--_index] = array;
        }
    }

這里也有一個值得注意的坑,那就是還回去的 byte[16] 里面的數(shù)據(jù)默認是不會清掉的,從上面的代碼也是可以看出來的,要想做清理,需要在 Return 方法中指定 clearArray=true,參考如下代碼:

public override void Return(T[] array, bool clearArray = false)
    {
        int num = Utilities.SelectBucketIndex(array.Length);

        if (num < _buckets.Length)
        {
            if (clearArray)
            {
                Array.Clear(array, 0, array.Length);
            }
            _buckets[num].Return(array);
        }
    }

四:總結(jié)

學(xué)習(xí)這其中的 池化架構(gòu) 思想,對平時項目開發(fā)還是能提供一些靈感的,其次對那些一次性使用 byte[] 的場景,用池化是個非常不錯的方法,這也是我對朋友dump分析后提出的一個優(yōu)化思路。

責(zé)任編輯:武曉燕 來源: 一線碼農(nóng)聊技術(shù)
相關(guān)推薦

2021-09-01 07:21:40

ArrayPool源碼Bucket

2022-07-19 13:51:47

數(shù)據(jù)庫Hikari連接池

2019-04-17 18:04:10

網(wǎng)卡虛擬化網(wǎng)絡(luò)設(shè)備

2017-01-12 14:52:03

JVMFinalRefere源碼

2023-01-07 17:41:36

線程池并發(fā)

2020-05-26 18:50:46

JVMAttachJava

2020-07-09 07:00:00

HashMap

2024-12-27 09:32:25

MyBatis代碼

2017-01-11 14:02:32

JVM源碼內(nèi)存

2009-01-18 09:19:00

DHCPVlANIP

2021-11-11 11:31:54

擺動序列數(shù)字

2017-01-11 14:19:26

JVM源碼All

2016-08-29 19:12:52

JavascriptBackbone前端

2010-01-27 10:37:17

Android圖片瀏覽

2015-06-15 10:32:44

Java核心源碼解讀

2024-10-28 08:15:32

2022-02-21 14:32:20

數(shù)字化轉(zhuǎn)型AI算法

2015-10-20 10:57:22

無線充電無線技術(shù)

2011-07-13 10:32:09

開源

2009-07-15 11:00:48

proxool連接池
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 久久精品中文字幕 | 欧美成人一区二区 | 成人天堂 | 欧美成人hd| 色嗨嗨| 日本羞羞影院 | av在线免费看网址 | 国产资源一区二区三区 | 久久三区 | 97精品久久 | 国产伦精品一区二区 | 亚洲视频在线观看 | japan21xxxxhd美女 日本欧美国产在线 | 91久久精品国产91久久性色tv | 日韩成人一区二区 | 日韩电影免费观看中文字幕 | 男女激情网 | 91精品国产综合久久精品 | 人妖一区 | 久久免费高清视频 | av国产在线观看 | 久久成人国产 | 日韩视频在线免费观看 | 国产免费一区二区三区 | 午夜丰满少妇一级毛片 | 亚洲成人一区二区 | 激情欧美一区二区三区中文字幕 | 国产一区二区在线视频 | 龙珠z在线观看 | 国产精品久久精品 | 日韩国产精品一区二区三区 | 激情的网站| 精品久草 | 户外露出一区二区三区 | 婷婷综合久久 | 久久综合99 | 亚洲精选一区 | 精品国产区 | 伊人色综合久久久天天蜜桃 | 久婷婷| www.国产精 |