ConcurrentStack.txt 20191106 ConcurrentStack Represents a thread-safe last in-first out (LIFO) collection. 參考 Concurrent.txt The following example shows how to use a ConcurrentStack to push and pop individual items: using System; using System.Collections.Concurrent; using System.Threading.Tasks; class Example { // Demonstrates: // ConcurrentStack.Push(); // ConcurrentStack.TryPeek(); // ConcurrentStack.TryPop(); // ConcurrentStack.Clear(); // ConcurrentStack.IsEmpty; static async Task Main() { int items = 10000; ConcurrentStack stack = new ConcurrentStack(); // Create an action to push items onto the stack Action pusher = () => { for (int i = 0; i < items; i++) { stack.Push(i); } }; // Run the action once pusher(); if (stack.TryPeek(out int result)) { Console.WriteLine($"TryPeek() saw {result} on top of the stack."); } else { Console.WriteLine("Could not peek most recently added number."); } // Empty the stack stack.Clear(); if (stack.IsEmpty) { Console.WriteLine("Cleared the stack."); } // Create an action to push and pop items Action pushAndPop = () => { Console.WriteLine($"Task started on {Task.CurrentId}"); int item; for (int i = 0; i < items; i++) stack.Push(i); for (int i = 0; i < items; i++) stack.TryPop(out item); Console.WriteLine($"Task ended on {Task.CurrentId}"); }; // Spin up five concurrent tasks of the action var tasks = new Task[5]; for (int i = 0; i < tasks.Length; i++) tasks[i] = Task.Factory.StartNew(pushAndPop); // Wait for all the tasks to finish up await Task.WhenAll(tasks); if (!stack.IsEmpty) { Console.WriteLine("Did not take all the items off the stack"); } } } The following example shows how to use a ConcurrentStack to push and pop ranges of items: using System; using System.Collections.Concurrent; using System.Linq; using System.Threading; using System.Threading.Tasks; class Example { // Demonstrates: // ConcurrentStack.PushRange(); // ConcurrentStack.TryPopRange(); static async Task Main() { int numParallelTasks = 4; int numItems = 1000; var stack = new ConcurrentStack(); // Push a range of values onto the stack concurrently await Task.WhenAll(Enumerable.Range(0, numParallelTasks).Select(i => Task.Factory.StartNew((state) => { // state = i * numItems int index = (int)state; int[] array = new int[numItems]; for (int j = 0; j < numItems; j++) { array[j] = index + j; } Console.WriteLine($"Pushing an array of ints from {array[0]} to {array[numItems - 1]}"); stack.PushRange(array); }, i * numItems, CancellationToken.None, TaskCreationOptions.DenyChildAttach, TaskScheduler.Default)).ToArray()); int numTotalElements = 4 * numItems; int[] resultBuffer = new int[numTotalElements]; await Task.WhenAll(Enumerable.Range(0, numParallelTasks).Select(i => Task.Factory.StartNew(obj => { int index = (int)obj; int result = stack.TryPopRange(resultBuffer, index, numItems); Console.WriteLine($"TryPopRange expected {numItems}, got {result}."); }, i * numItems, CancellationToken.None, TaskCreationOptions.LongRunning, TaskScheduler.Default)).ToArray()); for (int i = 0; i < numParallelTasks; i++) { // Create a sequence we expect to see from the stack taking the last number of the range we inserted var expected = Enumerable.Range(resultBuffer[i*numItems + numItems - 1], numItems); // Take the range we inserted, reverse it, and compare to the expected sequence var areEqual = expected.SequenceEqual(resultBuffer.Skip(i * numItems).Take(numItems).Reverse()); if (areEqual) { Console.WriteLine($"Expected a range of {expected.First()} to {expected.Last()}. Got {resultBuffer[i * numItems + numItems - 1]} to {resultBuffer[i * numItems]}"); } else { Console.WriteLine($"Unexpected consecutive ranges."); } } } } Remarks Note ConcurrentStack implements the IReadOnlyCollection interface starting with the .NET Framework 4.6; in previous versions of the .NET Framework, the ConcurrentStack class did not implement this interface. ConcurrentStack provides a few main operations: .Push inserts an element at the top of the ConcurrentStack. .TryPop removes an element from the top of the ConcurrentStack, or returns false if the item cannot be removed. .TryPeek returns an element that is at the top of the ConcurrentStack but does not remove it from the ConcurrentStack. .The TryPopRange and PushRange methods provide efficient pushing and popping of multiple elements in a single operation. ConcurrentStack 主要提供幾個功能: .Push 插入元素到頂端. .TryPop 取得並移除最頂端元素. 若無法完成, 則回傳 false. .TryPeek 取得最頂端元素, 但不刪除. .TryPopRange 及 PushRange 功能同上, 但可一次處理多個元素.