共用方式為


如何:以連續鍵分組結果 (C# 程式設計手冊)

下列範例顯示如何將元素分組成代表連續索引鍵結果的區塊 (Chunk)。 例如,假設指定了下列索引鍵/值組的序列:

機碼

A

We

A

think

A

that

B

Linq

C

A

really

B

cool

B

!

會依下列順序建立下列群組:

  1. We、think、that

  2. Linq

  3. really

  4. cool、!

方案是以擴充方法來實作,它是安全執行緒 (Thread-Safe) 的方法,而且會以資料流 (Streaming) 方式傳回其結果。 換言之,它會隨著在整個來源序列移動而產生其群組。 與 group 或 orderby 運算子不同的是,在讀取完所有序列之前,它就可以開始將群組傳回呼叫端。

如原始程式碼註解所述,在逐一查看來源序列時產生每個群組或區塊的複本,便可以達成安全執行緒的目的。 如果來源序列有大量連續項目序列,Common Language Runtime 可能會擲回 OutOfMemoryException

範例

下列範例會顯示擴充方法,也一併顯示使用擴充方法的用戶端程式碼。

using System;
using System.Collections.Generic;
using System.Linq;


namespace ChunkIt
{
    // Static class to contain the extension methods. 
    public static class MyExtensions
    {
        public static IEnumerable<IGrouping<TKey, TSource>> ChunkBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
        {
            return source.ChunkBy(keySelector, EqualityComparer<TKey>.Default);
        }

        public static IEnumerable<IGrouping<TKey, TSource>> ChunkBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
        {
            // Flag to signal end of source sequence. 
            const bool noMoreSourceElements = true;

            // Auto-generated iterator for the source array.        
            var enumerator = source.GetEnumerator();

            // Move to the first element in the source sequence. 
            if (!enumerator.MoveNext()) yield break;

            // Iterate through source sequence and create a copy of each Chunk. 
            // On each pass, the iterator advances to the first element of the next "Chunk" 
            // in the source sequence. This loop corresponds to the outer foreach loop that 
            // executes the query.
            Chunk<TKey, TSource> current = null;
            while (true)
            {
                // Get the key for the current Chunk. The source iterator will churn through 
                // the source sequence until it finds an element with a key that doesn't match. 
                var key = keySelector(enumerator.Current);

                // Make a new Chunk (group) object that initially has one GroupItem, which is a copy of the current source element.
                current = new Chunk<TKey, TSource>(key, enumerator, value => comparer.Equals(key, keySelector(value)));

                // Return the Chunk. A Chunk is an IGrouping<TKey,TSource>, which is the return value of the ChunkBy method. 
                // At this point the Chunk only has the first element in its source sequence. The remaining elements will be 
                // returned only when the client code foreach's over this chunk. See Chunk.GetEnumerator for more info. 
                yield return current;

                // Check to see whether (a) the chunk has made a copy of all its source elements or  
                // (b) the iterator has reached the end of the source sequence. If the caller uses an inner 
                // foreach loop to iterate the chunk items, and that loop ran to completion, 
                // then the Chunk.GetEnumerator method will already have made 
                // copies of all chunk items before we get here. If the Chunk.GetEnumerator loop did not 
                // enumerate all elements in the chunk, we need to do it here to avoid corrupting the iterator 
                // for clients that may be calling us on a separate thread. 
                if (current.CopyAllChunkElements() == noMoreSourceElements)
                {
                    yield break;
                }
            }
        }

        // A Chunk is a contiguous group of one or more source elements that have the same key. A Chunk  
        // has a key and a list of ChunkItem objects, which are copies of the elements in the source sequence. 
        class Chunk<TKey, TSource> : IGrouping<TKey, TSource>
        {
            // INVARIANT: DoneCopyingChunk == true ||  
            //   (predicate != null && predicate(enumerator.Current) && current.Value == enumerator.Current) 

            // A Chunk has a linked list of ChunkItems, which represent the elements in the current chunk. Each ChunkItem 
            // has a reference to the next ChunkItem in the list. 
            class ChunkItem
            {
                public ChunkItem(TSource value)
                {
                    Value = value;
                }
                public readonly TSource Value;
                public ChunkItem Next = null;
            }
            // The value that is used to determine matching elements 
            private readonly TKey key;

            // Stores a reference to the enumerator for the source sequence 
            private IEnumerator<TSource> enumerator;

            // A reference to the predicate that is used to compare keys. 
            private Func<TSource, bool> predicate;

            // Stores the contents of the first source element that 
            // belongs with this chunk. 
            private readonly ChunkItem head;

            // End of the list. It is repositioned each time a new 
            // ChunkItem is added. 
            private ChunkItem tail;

            // Flag to indicate the source iterator has reached the end of the source sequence. 
            internal bool isLastSourceElement = false;

            // Private object for thread syncronization 
            private object m_Lock;

            // REQUIRES: enumerator != null && predicate != null 
            public Chunk(TKey key, IEnumerator<TSource> enumerator, Func<TSource, bool> predicate)
            {
                this.key = key;
                this.enumerator = enumerator;
                this.predicate = predicate;

                // A Chunk always contains at least one element.
                head = new ChunkItem(enumerator.Current);

                // The end and beginning are the same until the list contains > 1 elements.
                tail = head;

                m_Lock = new object();
            }

            // Indicates that all chunk elements have been copied to the list of ChunkItems,  
            // and the source enumerator is either at the end, or else on an element with a new key. 
            // the tail of the linked list is set to null in the CopyNextChunkElement method if the 
            // key of the next element does not match the current chunk's key, or there are no more elements in the source. 
            private bool DoneCopyingChunk { get { return tail == null; } }

            // Adds one ChunkItem to the current group 
            // REQUIRES: !DoneCopyingChunk && lock(this) 
            private void CopyNextChunkElement()
            {
                // Try to advance the iterator on the source sequence. 
                // If MoveNext returns false we are at the end, and isLastSourceElement is set to true
                isLastSourceElement = !enumerator.MoveNext();

                // If we are (a) at the end of the source, or (b) at the end of the current chunk 
                // then null out the enumerator and predicate for reuse with the next chunk. 
                if (isLastSourceElement || !predicate(enumerator.Current))
                {
                    enumerator = null;
                    predicate = null;
                }
                else
                {
                    tail.Next = new ChunkItem(enumerator.Current);
                }

                // tail will be null if we are at the end of the chunk elements 
                // This check is made in DoneCopyingChunk.
                tail = tail.Next;
            }

            // Called after the end of the last chunk was reached. It first checks whether 
            // there are more elements in the source sequence. If there are, it  
            // Returns true if enumerator for this chunk was exhausted. 
            internal bool CopyAllChunkElements()
            {
                while (true)
                {
                    lock (m_Lock)
                    {
                        if (DoneCopyingChunk)
                        {
                            // If isLastSourceElement is false, 
                            // it signals to the outer iterator 
                            // to continue iterating. 
                            return isLastSourceElement;
                        }
                        else
                        {
                            CopyNextChunkElement();
                        }
                    }
                }
            }

            public TKey Key { get { return key; } }

            // Invoked by the inner foreach loop. This method stays just one step ahead 
            // of the client requests. It adds the next element of the chunk only after 
            // the clients requests the last element in the list so far. 
            public IEnumerator<TSource> GetEnumerator()
            {
                //Specify the initial element to enumerate.
                ChunkItem current = head;

                // There should always be at least one ChunkItem in a Chunk. 
                while (current != null)
                {
                    // Yield the current item in the list. 
                    yield return current.Value;

                    // Copy the next item from the source sequence,  
                    // if we are at the end of our local list. 
                    lock (m_Lock)
                    {
                        if (current == tail)
                        {
                            CopyNextChunkElement();
                        }
                    }

                    // Move to the next ChunkItem in the list.
                    current = current.Next;
                }
            }

            System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
            {
                return GetEnumerator();
            }
        }
    }

    // A simple named type is used for easier viewing in the debugger. Anonymous types 
    // work just as well with the ChunkBy operator. 
    public class KeyValPair
    {
        public string Key { get; set; }
        public string Value { get; set; }
    }

    class Program
    {
        // The source sequence. 
        public static IEnumerable<KeyValPair> list;

        // Query variable declared as class member to be available 
        // on different threads. 
        static IEnumerable<IGrouping<string, KeyValPair>> query;


        static void Main(string[] args)
        {
            // Initialize the source sequence with an array initializer.
            list = new[]
        {
            new KeyValPair{ Key = "A", Value = "We" },
            new KeyValPair{ Key = "A", Value = "Think" },
            new KeyValPair{ Key = "A", Value = "That" },
            new KeyValPair{ Key = "B", Value = "Linq" },
            new KeyValPair{ Key = "C", Value = "Is" },
            new KeyValPair{ Key = "A", Value = "Really" },
            new KeyValPair{ Key = "B", Value = "Cool" },
            new KeyValPair{ Key = "B", Value = "!" }
        };

            // Create the query by using our user-defined query operator.
            query = list.ChunkBy(p => p.Key);


            // ChunkBy returns IGrouping objects, therefore a nested 
            // foreach loop is required to access the elements in each "chunk".
            foreach (var item in query)
            {
                Console.WriteLine("Group key = {0}", item.Key);
                foreach (var inner in item)
                {
                    Console.WriteLine("\t{0}", inner.Value);
                }
            }

            Console.WriteLine("Press any key to exit");
            Console.ReadKey();
        }       

     }
}

若要在專案中使用擴充方法,請將 MyExtensions 靜態類別複製到新的或現有的原始程式碼檔,並視需要在其所在位置處加入命名空間 (Namespace) 的 using 指示詞。

請參閱

概念

LINQ 查詢運算式 (C# 程式設計手冊)

依據執行方式將標準查詢運算式分類