13

我想在 Lazy 对象上实现过期时间。过期冷却必须从第一次检索值开始。如果我们得到值,并且过期时间过去了,那么我们重新执行函数并重置过期时间。

我不熟悉扩展、部分关键字,也不知道最好的方法。

谢谢

编辑 :

到目前为止的代码:

新编辑:

新代码:

public class LazyWithExpiration<T>
{
    private volatile bool expired;
    private TimeSpan expirationTime;
    private Func<T> func;
    private Lazy<T> lazyObject;

    public LazyWithExpiration( Func<T> func, TimeSpan expirationTime )
    {
        this.expirationTime = expirationTime;
        this.func = func;

        Reset();
    }

    public void Reset()
    {
        lazyObject = new Lazy<T>( func );
        expired = false;
    }

    public T Value
    {
        get
        {
            if ( expired )
                Reset();

            if ( !lazyObject.IsValueCreated )
            {
                Task.Factory.StartNew( () =>
                {
                    Thread.Sleep( expirationTime );
                    expired = true;
                } );
            }

            return lazyObject.Value;
        }
    }

}
4

3 回答 3

8

2021年编辑:

您可能不应该使用此代码。也许对于一个非常简单的应用程序来说已经足够了,但它有一些问题。它不支持缓存失效,由于它使用 DateTime 的方式,它可能存在与 DST 或其他时区更改相关的问题,并且它以一种安全但可能相当慢的方式使用锁。考虑改用MemoryCache之类的东西。

原答案:

我同意其他评论者的观点,即您可能根本不应该碰 Lazy。如果您忽略多个线程安全选项,那么 Lazy 并不是很复杂,因此只需从头开始实现它。

顺便说一句,我非常喜欢这个想法,尽管我不知道我是否愿意将它用作通用缓存策略。对于一些更简单的场景可能就足够了。

这是我的尝试。如果您不需要它是线程安全的,您可以删除锁定的东西。我认为这里不可能使用双重检查锁模式,因为缓存值可能会在锁内失效。

public class Temporary<T>
{
    private readonly Func<T> factory;
    private readonly TimeSpan lifetime;
    private readonly object valueLock = new object();

    private T value;
    private bool hasValue;
    private DateTime creationTime;
    
    public Temporary(Func<T> factory, TimeSpan lifetime)
    {
        this.factory = factory;
        this.lifetime = lifetime;
    }
    
    public T Value
    {
        get
        {
            DateTime now = DateTime.Now;
            lock (this.valueLock)
            {
                if (this.hasValue)
                {
                    if (this.creationTime.Add(this.lifetime) < now)
                    {
                        this.hasValue = false;
                    }
                }
                
                if (!this.hasValue)
                {
                    this.value = this.factory();
                    this.hasValue = true;

                    // You can also use the existing "now" variable here.
                    // It depends on when you want the cache time to start
                    // counting from.
                    this.creationTime = Datetime.Now;
                }
                
                return this.value;
            }
        }
    }
}
于 2013-09-08T15:37:47.050 回答
3

我认为Lazy<T>这里不会有任何影响,它更像是一种通用方法,本质上类似于单例模式。

您需要一个简单的包装类,它要么返回真实对象,要么将所有调用传递给它。

我会尝试这样的事情(内存不足,因此可能包含错误):

public class Timed<T> where T : new() {
    DateTime init;
    T obj;

    public Timed() {
        init = new DateTime(0);
    }

    public T get() {
        if (DateTime.Now - init > max_lifetime) {
            obj = new T();
            init = DateTime.Now;
        }
        return obj;
    }
}

要使用,您只需使用Timed<MyClass> obj = new Timed<MyClass>();而不是MyClass obj = new MyClass();. 实际调用将obj.get().doSomething()代替obj.doSomething().

编辑:

请注意,您不必将类似于我上面的方法结合起来,Lazy<T>因为您实际上已经在强制延迟初始化。例如,您当然可以在构造函数中定义最大生命周期。

于 2013-09-08T15:10:47.830 回答
2

我需要同样的东西。但是我更喜欢在没有写入时没有锁定读取的实现。

public class ExpiringLazy<T>
{
    private readonly Func<T> factory;
    private readonly TimeSpan lifetime;
    private readonly ReaderWriterLockSlim locking = new ReaderWriterLockSlim(LockRecursionPolicy.NoRecursion);

    private T value;
    private DateTime expiresOn = DateTime.MinValue;

    public ExpiringLazy(Func<T> factory, TimeSpan lifetime)
    {
        this.factory = factory;
        this.lifetime = lifetime;
    }

    public T Value
    {
        get
        {
            DateTime now = DateTime.UtcNow;
            locking.EnterUpgradeableReadLock();
            try
            {
                if (expiresOn < now)
                {
                    locking.EnterWriteLock();
                    try
                    {
                        if (expiresOn < now)
                        {
                            value = factory();
                            expiresOn = DateTime.UtcNow.Add(lifetime);
                        }
                    }
                    finally
                    {
                        locking.ExitWriteLock();
                    }
                }

                return value;
            }
            finally
            {
                locking.ExitUpgradeableReadLock();
            }
        }
    }
}
于 2019-10-18T09:44:39.803 回答