namespace WpfCopy.Controls
{
public class CacheFileEventArgs : EventArgs
{
public bool IsFaulted { get; private set; }
public CacheFileModel CacheFile { get; private set; } public CacheFileEventArgs(CacheFileModel cacheFile)
{
CacheFile = cacheFile;
IsFaulted = false;
} public CacheFileEventArgs()
{
IsFaulted = true;
} } public class CacheFileModel
{
public string RemoteFile { get; set; }
public string LocalFile { get; set; } public DateTime CreateTime { get; set; }
public DateTime LastUseTime { get; set; }
} class FileCacheMgr
{
private const string CacheDir = "CacheFile"; private const string CacheDataFile = "file.cache"; /// <summary>
/// 缓存数据文件的读写锁
/// </summary>
readonly object _cacheDataFileLock = new object(); /// <summary>
/// 管理缓存数据的锁
/// </summary>
readonly object _cacheLock = new object(); /// <summary>
/// 缓存数据任务的锁
/// </summary>
readonly object _cacheTaskLock = new object(); /// <summary>
/// 缓存数据字典
/// </summary>
Dictionary<string, CacheFileModel> _cacheDict = new Dictionary<string, CacheFileModel>(); /// <summary>
/// 下载任务字典
/// </summary>
readonly Dictionary<string, WeakDelegateCollection<CacheFileEventArgs>> _cacheTaskDict = new Dictionary<string, WeakDelegateCollection<CacheFileEventArgs>>(); private static readonly FileCacheMgr instance = new FileCacheMgr();
public static FileCacheMgr Instance { get { return instance; } } public FileCacheMgr()
{ } /// <summary>
/// 读取缓存
/// </summary>
void LoadCacheData()
{
lock (_cacheDataFileLock)
{
if (!File.Exists(CacheDataFile) && Directory.Exists(CacheDir))
Directory.Delete(CacheDir, true);
var xs = new XmlSerializer(typeof(List<CacheFileModel>));
using (Stream stream = new FileStream(CacheDataFile, FileMode.Open, FileAccess.Read))
{
var list = xs.Deserialize(stream) as List<CacheFileModel> ?? new List<CacheFileModel>(); _cacheDict = list.ToDictionary(m => m.RemoteFile);
}
}
}
/// <summary>
/// 保存缓存
/// </summary>
void SaveCacheData()
{
lock (_cacheDataFileLock)
{
try
{
var xs = new XmlSerializer(typeof(List<CacheFileModel>));
using (Stream stream = new FileStream(CacheDataFile, FileMode.Create, FileAccess.Write))
{
xs.Serialize(stream, _cacheDict.Values.ToList<CacheFileModel>());
}
}
catch (Exception)
{
File.Delete(CacheDataFile);
}
}
}
/// <summary>
/// 清除过期缓存
/// </summary>
public void ClearExpireCache()
{
try
{
List<string> clearList = new List<string>(); foreach (var item in _cacheDict)
{
if (DateTime.Now - item.Value.LastUseTime > TimeSpan.FromDays(7))
clearList.Add(item.Key);
} foreach (var item in clearList)
{
File.Delete(_cacheDict[item].LocalFile);
_cacheDict.Remove(item);
} SaveCacheData();
}
catch (Exception ex)
{
throw ex;
}
}
/// <summary>
/// 添加缓存数据
/// </summary>
/// <param name="model"></param>
public void AddCacheData(CacheFileModel model)
{
if (model == null)
throw new ArgumentException("model"); lock (_cacheLock)
{
if (_cacheDict.ContainsKey(model.RemoteFile) == false)
{
_cacheDict.Add(model.RemoteFile, model);
SaveCacheData();
}
}
}
/// <summary>
/// 删除缓存文件--
/// </summary>
/// <param name="model"></param>
public void RemoveCacheData(CacheFileModel model)
{
if (model == null)
throw new ArgumentException("model"); if (File.Exists(model.LocalFile))
File.Delete(model.LocalFile); if (_cacheDict.ContainsKey(model.RemoteFile))
{
_cacheDict.Remove(model.RemoteFile);
SaveCacheData();
}
} /// <summary>
/// 获取缓存数据,如果不存在,则创建下载任务
/// </summary>
/// <param name="remoteFile"></param>
/// <param name="callback"></param>
/// <param name="getFtpFunc"></param>
void GetCacheFile(string remoteFile, EventHandler<CacheFileEventArgs> callback, Func<MyFtp> getFtpFunc)
{ if (_cacheDict.ContainsKey(remoteFile))
{
CacheFileModel cache = _cacheDict[remoteFile];
if (File.Exists(cache.LocalFile))
{
cache.LastUseTime = DateTime.Now;
SaveCacheData(); if (callback != null)
{
callback(this, new CacheFileEventArgs(cache));
}
return;
}
else
{
_cacheDict.Remove(remoteFile);
}
} CreateDownLoadTask(remoteFile, getFtpFunc(), callback);
} void CreateDownLoadTask(string remoteFile, MyFtp myFtp, EventHandler<CacheFileEventArgs> callBack)
{
lock (_cacheTaskLock)
{
bool exist = _cacheTaskDict.ContainsKey(remoteFile);
AddCallBackToDictNoLock(remoteFile, callBack);
if (exist == false)
{
Task.Factory.StartNew(() =>
{
DownloadFileWork(remoteFile, myFtp, callBack);
}, TaskCreationOptions.PreferFairness);
}
}
} void DownloadFileWork(string remoteFile, MyFtp myFtp, EventHandler<CacheFileEventArgs> callback)
{
string localFile = Path.Combine(CacheDir, Guid.NewGuid().ToString() + Path.GetExtension(remoteFile)); string path = Path.GetDirectoryName(localFile); if (Directory.Exists(path) == false)
{
Directory.CreateDirectory(path);
}
var eventArgs = new CacheFileEventArgs();
try
{
bool dlRet = myFtp.DownLoad(remoteFile, localFile);
if (dlRet && File.Exists(localFile))
{
var cacheModel = new CacheFileModel()
{
RemoteFile = remoteFile,
LocalFile = localFile
};
eventArgs = new CacheFileEventArgs(cacheModel);
AddCacheData(cacheModel);
}
}
finally
{
try
{
InvokeCallBack(remoteFile, eventArgs);
}
finally
{
RemoveCallBack(remoteFile);
}
}
} void AddCallBackToDictNoLock(string remoteFile, EventHandler<CacheFileEventArgs> callback)
{
if (_cacheTaskDict.ContainsKey(remoteFile) == false)
_cacheTaskDict.Add(remoteFile, new WeakDelegateCollection<CacheFileEventArgs>()); var weakEvent = _cacheTaskDict[remoteFile];
weakEvent.WeakEvent += callback;
} void RemoveCallBack(string remoteFile)
{
lock (_cacheTaskLock)
{
if (_cacheTaskDict.ContainsKey(remoteFile))
_cacheTaskDict.Remove(remoteFile);
}
} void InvokeCallBack(string remoteFile, CacheFileEventArgs args)
{
lock (_cacheTaskLock)
{
if (_cacheTaskDict.ContainsKey(remoteFile) == false)
{
return;
} _cacheTaskDict[remoteFile].Invoke(this, args);
}
} }
}

【C#】缓存数据的更多相关文章

  1. plain framework 1 网络流 缓存数据详解

    网络流是什么?为什么网络流中需要存在缓存数据?为什么PF中要采用缓存网络数据的机制?带着这几个疑问,让我们好好详细的了解一下在网络数据交互中我们容易忽视以及薄弱的一块.该部分为PF现有的网络流模型,但 ...

  2. thinkphp 缓存数据

    thinkphp 中内置了缓存操作 3.1版本的数据缓存方法是cache 基本用法: S(array('type'=>'xcache','expire'=>60)); 缓存初始化 缓存初始 ...

  3. HTML5本地缓存数据

    //HTML5本地缓存数据 function putObj(key, data) { if (!!window.localStorage) { var obj = { "key": ...

  4. 刷新本地的DNS缓存数据

    ipconfig /flushdns”执行,刷新本地的DNS缓存数据. ipconfig /displaydns      查看本地DNS缓存记录的命令为:ipconfig /displaydns.你 ...

  5. 网站缓存数据到tomcat服务器

    通过缓存使相同的数据不用重复加载,降低数据库的访问 public class CacheFilter implements Filter { //实例变量[每线程共享] private Map< ...

  6. iOS五种本地缓存数据方式

    iOS五种本地缓存数据方式   iOS本地缓存数据方式有五种:前言 1.直接写文件方式:可以存储的对象有NSString.NSArray.NSDictionary.NSData.NSNumber,数据 ...

  7. IE浏览器中ajax使用缓存数据的问题

    今天做了一个小功能:点击鼠标实时更新系统时间,采用ajax,过程很顺利,没遇到啥差错,谷歌,火狐,欧鹏一律通过,怀着忐忑的心情点开了IE8,果然,IE要对得起前端杀手的称号:更新不了时间. 查了一下这 ...

  8. Java通过SpyMemcached来缓存数据

    配置好Magent+memcached后,很明显数据之间的输入与输出都是通过代理服务器的,magent是做代理服务器的很明显java在memecached的调用驱动在magent同样适用. 这里选择S ...

  9. ThinkPHP使用Memcached缓存数据

    ThinkPHP默认使用文件缓存数据,支持Memcache等其他缓存方式,有两个PHP扩展:Memcache和Memcached,Memcahe官方有说明,主要说一下Memcached. 相对于PHP ...

  10. NCache实现Oracle数据与分布式缓存数据同步的3个步骤

    多层次结构的应用程序是目前发展的趋势,这种程序都需要庞大的数据库支持.而数据传输的能力直接影响程序性能,成为程序可扩展性的瓶颈.因此很多开发者开始在程序中使用内存分布式缓存来提高程序性能. 同时,内存 ...

随机推荐

  1. socket编程 ------ sockaddr_in 和 sockaddr 的区别

    struct sockaddr 和 struct sockaddr_in 这两个结构体用来处理网络通信的地址. // 以下是 IPv4 的定义 struct sockaddr { unsigned s ...

  2. (转) 使用vivado创建工程 3

    Create a Hello World application In this experiment we will use Xilinx SDK to create a simple Hello ...

  3. time_t与GMT格式互转

    time_t Time::timeFromGMT(string gmt) { char week[4]; memset(week,0,4); char month[4]; memset(month,0 ...

  4. centos7下配置mysql5.7.24主从复制

    前置条件 准备两台服务器(可以是虚拟机),系统为centos7 此处演示的两台服务器:192.168.8.134.192.168.8.135 第一步:安装mysql5.7.24 先在两台服务器上安装m ...

  5. 全国排名的问题(linq 的连表查询 等同于sql的left join)

    前言:要获得全国排名,(因为权限问题,显示的数据不是全国的数据,而是某个分区的数据,因此,不能获得数据后排序得到排名) 显示本部的员工积分并且获得在全国的排名. 我的思路:获得显示的员工信息集合1,获 ...

  6. Win7 32位安装Oracle11g R2 图解示例

    Win7 32位操作系统安装Oracle11g R2 图解示例.废话不说了,直接上图. 1.下载的两个oracle 11gR2压缩包解压到单独的文件夹中. 2.找到解压的database文件夹中的Se ...

  7. rabbitmq常见运维命令和问题总结

    常见运维命令作用: yum安装erlang的环境配置: ERLANG_HOME=/usr/lib64/erlang export PATH=$PATH:$ERLANG_HOME/bin 常见rabbi ...

  8. dfs序+主席树 BZOJ 2588 当然树链剖分+主席树也可以?

    2588: Spoj 10628. Count on a tree Time Limit: 12 Sec  Memory Limit: 128 MBSubmit: 5822  Solved: 1389 ...

  9. c# windows server安装启动与卸载

    使用installutil.exe安装卸载服务时,由于需要指向服务的全路径,由于生成目录往往不是服务发布的最终目录,很不便利,下面介绍两种方式方便操作: 方式一: 项目中加入install.bat与u ...

  10. 修复 Plugin execution not covered by lifecycle configuration: org.codehaus.mojo:build-helper-maven-plugin:1.8:add-source (execution: add-source, phase: generate-sources)

    在maven项目中使用add-source时,pom.xml报如下错误: Plugin execution not covered by lifecycle configuration: org.co ...