7

.NET 4.0ConditionalWeakTable<T>は事実上、ディクショナリのキーが弱く参照され、収集できるディクショナリです。これはまさに私が必要としているものです。問題は、この辞書からすべてのライブ キーを取得できるようにする必要があることですが、MSDN は次のように述べています。

ディクショナリに通常含まれるすべてのメソッド (GetEnumerator や Contains など) が含まれているわけではありません。

からライブ キーまたはキーと値のペアを取得する可能性はありますConditionalWeakTable<T>か?

4

3 に答える 3

9

私は自分のラッパーを作成することになりました:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.CompilerServices;

public sealed class ConditionalHashSet<T> where T : class
{
    private readonly object locker = new object();
    private readonly List<WeakReference> weakList = new List<WeakReference>();
    private readonly ConditionalWeakTable<T, WeakReference> weakDictionary =
        new ConditionalWeakTable<T, WeakReference>();

    public void Add(T item)
    {
        lock (this.locker)
        {
            var reference = new WeakReference(item);
            this.weakDictionary.Add(item, reference);
            this.weakList.Add(reference);
            this.Shrink();
        }
    }

    public void Remove(T item)
    {
        lock (this.locker)
        {
            WeakReference reference;

            if (this.weakDictionary.TryGetValue(item, out reference))
            {
                reference.Target = null;
                this.weakDictionary.Remove(item);
            }
        }
    }

    public T[] ToArray()
    {
        lock (this.locker)
        {
            return (
                from weakReference in this.weakList
                let item = (T)weakReference.Target
                where item != null
                select item)
                .ToArray();
        }
    }

    private void Shrink()
    {
        // This method prevents the List<T> from growing indefinitely, but 
        // might also cause  a performance problem in some cases.
        if (this.weakList.Capacity == this.weakList.Count)
        {
            this.weakList.RemoveAll(weak => !weak.IsAlive);
        }
    }
}
于 2013-09-07T14:48:19.437 に答える