首頁 > 軟體

C#多執行緒之執行緒池ThreadPool用法

2022-03-09 13:00:19

一、ThreadPool

ThreadPool是.Net Framework 2.0版本中出現的。

ThreadPool出現的背景:Thread功能繁多,而且對執行緒數量沒有管控,對於執行緒的開闢和銷燬要消耗大量的資源。每次new一個THread都要重新開闢記憶體。

如果某個執行緒的建立和銷燬的代價比較高,同時這個物件還可以反覆使用的,就需要一個池子(容器),儲存多個這樣的物件,需要用的時候從池子裡面獲取,用完之後不用銷燬,在放到池子裡面。這樣不但能節省記憶體資源,提高效能,而且還能管控執行緒的總數量,防止濫用。這時就需要使用ThreadPool了。

我們來看看ThreadPool中常用的一些方法。

1、QueueUserWorkItem()

QueueUserWorkItem()方法用來啟動一個多執行緒。我們先看看方法的定義:

QueueUserWorkItem()方法有一個WaitCallback型別的引數,在看看WaitCallback的定義:

可以看到WaitCallback就是有一個object型別引數的委託,所以ThreadPool啟動多執行緒使用下面的程式碼:

using System;
using System.Threading;

namespace ThreadPoolDemo
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine($"start ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            // ThreadPoll啟動多執行緒
            ThreadPool.QueueUserWorkItem(p => DoSomethingLong("啟動多執行緒"));

            Console.WriteLine($"end ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            Console.ReadKey();
        }

        static void DoSomethingLong(string para)
        {
            Console.WriteLine($"{para}  ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
        }
    }
}

執行結果:

2、GetMaxThreads()

GetMaxThreads()用來獲取執行緒池中最多可以有多少個輔助執行緒和最多有多少個非同步執行緒。

ThreadPool.GetMaxThreads(out int workerThreads, out int completionPortThreads);
Console.WriteLine($"GetMaxThreads workerThreads={workerThreads} completionPortThreads={completionPortThreads}");

程式執行結果:

3、GetMinThreads()

GetMinThreads()用來獲取執行緒池中最少可以有多少個輔助執行緒和最少有多少個非同步執行緒。

ThreadPool.GetMinThreads(out int minworkerThreads, out int mincompletionPortThreads);
Console.WriteLine($"GetMinThreads workerThreads={minworkerThreads} completionPortThreads={mincompletionPortThreads}");

程式執行結果:

4、SetMaxThreads()和SetMinThreads()

SetMaxThreads()和SetMinThreads()分別用來設定執行緒池中最多執行緒數和最少執行緒數。

using System;
using System.Threading;

namespace ThreadPoolDemo
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine($"start ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            // ThreadPoll啟動多執行緒
            ThreadPool.QueueUserWorkItem(p => DoSomethingLong("啟動多執行緒"));

            // 獲取最大執行緒
            ThreadPool.GetMaxThreads(out int workerThreads, out int completionPortThreads);
            Console.WriteLine($"GetMaxThreads workerThreads={workerThreads} completionPortThreads={completionPortThreads}");

            // 獲取最小執行緒
            ThreadPool.GetMinThreads(out int minworkerThreads, out int mincompletionPortThreads);
            Console.WriteLine($"GetMinThreads workerThreads={minworkerThreads} completionPortThreads={mincompletionPortThreads}");

            // 設定執行緒池執行緒
            SetThreadPool();
            // 輸出設定後的執行緒池執行緒個數
            Console.WriteLine("輸出修改後的最多執行緒數和最少執行緒數");
            ThreadPool.GetMaxThreads(out int maxworkerThreads, out int maxcompletionPortThreads);
            Console.WriteLine($"GetMaxThreads workerThreads={maxworkerThreads} completionPortThreads={maxcompletionPortThreads}");
            ThreadPool.GetMinThreads(out int workerEditThreads, out int completionPortEditThreads);
            Console.WriteLine($"GetMinThreads workerThreads={workerEditThreads} completionPortThreads={completionPortEditThreads}");
            Console.WriteLine($"end ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            Console.ReadKey();
        }

        static void DoSomethingLong(string para)
        {
            Console.WriteLine($"{para}  ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
        }

        /// <summary>
        /// 設定執行緒池執行緒個數
        /// </summary>
        static void SetThreadPool()
        {

            Console.WriteLine("************設定最多執行緒數和最少執行緒數****************");
            // 設定最大執行緒
            ThreadPool.SetMaxThreads(16, 16);
            // 設定最小執行緒
            ThreadPool.SetMinThreads(8, 8);

        }
    }
}

程式執行結果:

二、執行緒等待

先來看下面一個小例子:

ThreadPool.QueueUserWorkItem(p => DoSomethingLong("啟動多執行緒"));
Console.WriteLine("等著QueueUserWorkItem完成後才執行");

我們想讓非同步多執行緒執行完以後再輸出“等著QueueUserWorkItem完成後才執行” 這句話,上面的程式碼執行效果如下:

從截圖中可以看出,效果並不是我們想要的,Thread中提供了暫停、恢復等API,但是ThreadPool中沒有這些API,在ThreadPool中要實現執行緒等待,需要使用到ManualResetEvent類。

ManualResetEvent類的定義如下:

ManualResetEvent需要一個bool型別的引數來表示暫停和停止。上面的程式碼修改如下:

// 引數設定為false
ManualResetEvent manualResetEvent = new ManualResetEvent(false);
ThreadPool.QueueUserWorkItem(p => 
{
      DoSomethingLong("啟動多執行緒");
      // 設定為true
      manualResetEvent.Set();
});
// 
manualResetEvent.WaitOne();
Console.WriteLine("等著QueueUserWorkItem完成後才執行");

結果:

ManualResetEvent類的引數值執行順序如下:

(1)、false--WaitOne等待--Set--true--WaitOne直接過去
(2)、true--WaitOne直接過去--ReSet--false--WaitOne等待

注意:一般情況下,不要阻塞執行緒池中的執行緒,因為這樣會導致一些無法預見的錯誤。來看下面的一個例子:

static void SetWait()
{
            // 設定最大執行緒
            ThreadPool.SetMaxThreads(16, 16);
            // 設定最小執行緒
            ThreadPool.SetMinThreads(8, 8);
            ManualResetEvent manualResetEvent = new ManualResetEvent(false);
            for (int i = 0; i < 20; i++)
            {
                int k = i;
                ThreadPool.QueueUserWorkItem(p =>
                {
                    Console.WriteLine(k);
                    if (k < 18)
                    {
                        manualResetEvent.WaitOne();
                    }
                    else
                    {
                        // 設為true
                        manualResetEvent.Set();
                    }
                });
            }
            if (manualResetEvent.WaitOne())
            {
                Console.WriteLine("沒有死鎖、、、");
            }
            else
            {
                Console.WriteLine("發生死鎖、、、");
            }
}

啟動20個執行緒,如果k小於18就阻塞當前的執行緒,結果:

從截圖中看出,只執行了16個執行緒,後面的執行緒沒有執行,這是為什麼呢?因為我們在上面設定了執行緒池中最多可以有16個執行緒,當16個執行緒都阻塞的時候,會造成死鎖,所以後面的執行緒不會再執行了。

三、執行緒重用

ThreadPool可以很好的實現執行緒的重用,這樣就可以減少記憶體的消耗,看下面的程式碼:

/// <summary>
/// 測試ThreadPool執行緒重用
/// </summary>
static void ThreadPoolTest()
{
            // 執行緒重用
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            Thread.Sleep(10 * 1000);
            Console.WriteLine("前面的計算都完成了。。。。。。。。");
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t =>DoSomethingLong("ThreadPool"));
}

然後在Main方法裡面呼叫該方法,輸出結果如下圖所示:

我們在程式碼裡面總共建立了10個執行緒,而結果裡面只有4個執行緒ID,這就說明ThreadPool可以實現執行緒的重用。下面我們在看看Thread是否可以實現執行緒的重用,程式碼如下:

/// <summary>
/// 測試Thread執行緒重用
/// </summary>
static void ThreadTest()
{
            for (int i = 0; i < 5; i++)
            {
                new Thread(() => DoSomethingLong("Threads")).Start();
            }
            Thread.Sleep(10 * 1000);
            Console.WriteLine("前面的計算都完成了。。。。。。。。");
            for (int i = 0; i < 5; i++)
            {
                new Thread(() => DoSomethingLong("btnThreads")).Start();
            }
}

然後在Main方法裡面呼叫,輸入結果如下圖所示:

我們同樣在程式碼裡面建立了10個執行緒,結果輸出了10個執行緒的ID,這就說明Thread不能實現執行緒的重用。同樣也說明THread的效率沒有ThreadPool高。

程式完整程式碼:

using System;
using System.Threading;

namespace ThreadPoolDemo
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine($"start ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            //// ThreadPoll啟動多執行緒
            //ThreadPool.QueueUserWorkItem(p => DoSomethingLong("啟動多執行緒"));

            //// 獲取最大執行緒
            //ThreadPool.GetMaxThreads(out int workerThreads, out int completionPortThreads);
            //Console.WriteLine($"GetMaxThreads workerThreads={workerThreads} completionPortThreads={completionPortThreads}");

            //// 獲取最小執行緒
            //ThreadPool.GetMinThreads(out int minworkerThreads, out int mincompletionPortThreads);
            //Console.WriteLine($"GetMinThreads workerThreads={minworkerThreads} completionPortThreads={mincompletionPortThreads}");

            //// 設定執行緒池執行緒
            //SetThreadPool();
            //// 輸出設定後的執行緒池執行緒個數
            //Console.WriteLine("輸出修改後的最多執行緒數和最少執行緒數");
            //ThreadPool.GetMaxThreads(out int maxworkerThreads, out int maxcompletionPortThreads);
            //Console.WriteLine($"GetMaxThreads workerThreads={maxworkerThreads} completionPortThreads={maxcompletionPortThreads}");
            //ThreadPool.GetMinThreads(out int workerEditThreads, out int completionPortEditThreads);
            //Console.WriteLine($"GetMinThreads workerThreads={workerEditThreads} completionPortThreads={completionPortEditThreads}");
            //Console.WriteLine($"end ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");

            //// 引數設定為false
            //ManualResetEvent manualResetEvent = new ManualResetEvent(false);
            //ThreadPool.QueueUserWorkItem(p => 
            //{
            //    DoSomethingLong("啟動多執行緒");
            //    // 設定為true
            //    manualResetEvent.Set();
            //});
            //// 
            //manualResetEvent.WaitOne();
            //Console.WriteLine("等著QueueUserWorkItem完成後才執行");

            // SetWait();

            // ThreadPool實現執行緒的重用
            // ThreadPoolTest();

            // Thread
            ThreadTest();
            Console.WriteLine($"end ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
            Console.ReadKey();
        }

        static void DoSomethingLong(string para)
        {
            Console.WriteLine($"{para}  ThreadId: {Thread.CurrentThread.ManagedThreadId.ToString("00")}");
        }

        /// <summary>
        /// 設定執行緒池執行緒個數
        /// </summary>
        static void SetThreadPool()
        {

            Console.WriteLine("************設定最多執行緒數和最少執行緒數****************");
            // 設定最大執行緒
            ThreadPool.SetMaxThreads(16, 16);
            // 設定最小執行緒
            ThreadPool.SetMinThreads(8, 8);

        }

        static void SetWait()
        {
            // 設定最大執行緒
            ThreadPool.SetMaxThreads(16, 16);
            // 設定最小執行緒
            ThreadPool.SetMinThreads(8, 8);
            ManualResetEvent manualResetEvent = new ManualResetEvent(false);
            for (int i = 0; i < 20; i++)
            {
                int k = i;
                ThreadPool.QueueUserWorkItem(p =>
                {
                    Console.WriteLine(k);
                    if (k < 18)
                    {
                        manualResetEvent.WaitOne();
                    }
                    else
                    {
                        // 設為true
                        manualResetEvent.Set();
                    }
                });
            }
            if (manualResetEvent.WaitOne())
            {
                Console.WriteLine("沒有死鎖、、、");
            }
            else
            {
                Console.WriteLine("發生死鎖、、、");
            }
        }

        /// <summary>
        /// 測試ThreadPool執行緒重用
        /// </summary>
        static void ThreadPoolTest()
        {
            // 執行緒重用
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            Thread.Sleep(10 * 1000);
            Console.WriteLine("前面的計算都完成了。。。。。。。。");
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
            ThreadPool.QueueUserWorkItem(t => DoSomethingLong("ThreadPool"));
        }

        /// <summary>
        /// 測試Thread執行緒重用
        /// </summary>
        static void ThreadTest()
        {
            for (int i = 0; i < 5; i++)
            {
                new Thread(() => DoSomethingLong("Threads")).Start();
            }
            Thread.Sleep(10 * 1000);
            Console.WriteLine("前面的計算都完成了。。。。。。。。");
            for (int i = 0; i < 5; i++)
            {
                new Thread(() => DoSomethingLong("btnThreads")).Start();
            }
        }
    }
}

到此這篇關於C#多執行緒之執行緒池ThreadPool用法的文章就介紹到這了。希望對大家的學習有所幫助,也希望大家多多支援it145.com。


IT145.com E-mail:sddin#qq.com