我正在使用Memcached来存储数据以便快速访问.我已经读过创建MemcachedClient的成本很高,并且将MemcachedClient用作静态(参见:
link)
所以我为我的客户使用Singleton模式:
public class CommonObjectsCache
{
private static CommonObjectsCache _cache;
private static MemcachedClient _client;
public static MemcachedClient Client
{
get
{
if (_client == null)
_client = new MemcachedClient();
return _client;
}
private set
{
_client = value;
}
}
private CommonObjectsCache()
{
_client = new MemcachedClient();
}
public static CommonObjectsCache Cache
{
get
{
if (_cache == null)
_cache = new CommonObjectsCache();
return _cache;
}
}
}
在我的DAL中,我使用它们如下:
public static List<Item1> AllItem1s
{
get
{
if (CommonObjectsCache.Client.Get<List<Item1>>("AllItem1s") == null)
RefreshItem1Cache();
return CommonObjectsCache.Client.Get<List<Item1>>("AllItem1s");
}
private set
{
CommonObjectsCache.Client.Store(StoreMode.Set, "AllItem1s", value);
}
}
public static List<Item2> AllItem2s
{
get { // Same as above }
private set { // Same as above }
}
public static List<Item3> AllItem3s
{
get { // Same as above }
private set { // Same as above }
}
public static List<Item4> AllItem4s
{
get { // Same as above }
private set { // Same as above }
}
并填写为:
public static void RefreshItem1Cache()
{
List<Item1> items = (from i ctx.Item1
select i).ToList();
AllItem1s = items;
}
在我的DAL代码中,我有一个方法,如:
public static MyModel GetMyModel(int? id)
{
// I use AllItem1s here.
}
当我运行代码时,它有时会说AllItem1s.Count == 0,但是当我在AllItem1s中放置断点并诊断该值时,我发现它已被填充.所以,我更新了代码如下,以检查我是否做错了:
public static MyModel GetMyModel(int? id)
{
if (AllItem1s == null || AllItem1s.Count == 0 || AllItem2s == null || AllItem2s.Count == 0 || AllItem3s == null || AllItem3s.Count == 0 || AllItem4s == null || AllItem4s.Count == 0)
{
string msg = "Error!!!!!";
}
// I use AllItem1s here.
}
令人惊讶的是,代码落到了字符串msg =“Error !!!!!”;块!!!
但是当我在if块中放置一个断点并观察每个集合的Count属性时,我发现它有数字.
所以我得出结论,在获取AllItemXs属性时存在竞争条件.当它检查条件时,其中至少有一个没有被正确设置(这没有意义,因为它们在同一个线程上,并且属性的getter不能返回空集合).
任何人都可以解释为什么会发生这种情况以及如何克服这个问题?
最佳答案 您的单例实现不是线程安全的.
想象一下,两个(或更多)线程同时命中空检查:两个(所有)线程将初始化它们自己的CommonObjectsCache实例.
您可以使用lock语句对实例进行空检查和初始化,也可以使用双重检查锁定模式.
只需谷歌在C#中实现线程安全的单例实现.