最佳做法是在内存中存储数组
本文关键字:存储 数组 内存 最佳 | 更新日期: 2023-09-27 18:36:30
所以我们有一个WCF服务,它向客户端提供一些对象。我们的 wcf 服务从数据访问层 (DAL) 获取数据。DAL 从 API 获取数据,并将其格式化为我们可以使用的对象。对于大多数对象,这效果很好。
但我们还需要来自 API 的对象列表,并且对象不会更改。针对 API 的查询需要 15 到 20 秒。对同一数据多次运行此查询的方式。因此,我们希望将列表存储在内存中。因为DBA不是我们最好的朋友,SQLite或SQL CE不是一种选择。
现在我们有这样的东西,第一次加载组时,我们将它们存储在私有静态变量中。这是做这样的事情的正确方法,还是有更好的方法?
public static class GroupStore
{
private static DTO.Group[] _groups;
public static DTO.Group[] GetAll()
{
if (_groups == null)
{
var dal = PluginHandler.Instance.GetPlugin();
_groups = dal.GetAll();
}
return _groups;
}
}
如果您使用的是 .NET4 您可以使用Lazy<T>
类。它是线程安全的,并为您包装延迟加载。
代码变得更加干净:
public static class GroupStore
{
private static Lazy<DTO.Group[]> _groups = new Lazy<DTO.Group[]>(GetAll);
public static DTO.Group[] Groups { get { return _groups.Value; } }
private static DTO.Group[] GetAll()
{
var dal = PluginHandler.Instance.GetPlugin();
return dal.GetAll();
}
}
更新
您在评论中链接的答案是可以的。但是IsValueCreated
无缘无故地使用锁定(因为列表可能在检查和下一次访问之间发生了变化)。
如果调用"ToString",我也不会创建值。
更干净的解决方案:
public sealed class Lazy<T> where T : class
{
private readonly object _syncRoot = new object();
private readonly Func<T> _factory;
private T _value;
public Lazy(Func<T> factory)
{
if (factory == null) throw new ArgumentNullException("factory");
_factory = factory;
}
public T Value
{
get
{
if (_value == null)
{
lock (_syncRoot)
{
if (_value == null)
{
_value = _factory();
}
}
}
return _value;
}
}
public override string ToString()
{
return _value == null ? "Not created" : _value.ToString();
}
}