ASP.NET缓存 Cache

缓存介绍

如果每次进入页面的时候都查询数据库生成页面内容的话,如果访问量非常大,则网站性能会非常差,
而如果只有第一次访问的时候才查询数据库生成页面内容,以后都直接输出内容,则能提高系统性能,
这样无论多少人访问都只访问一次数据库,数据库压力不变
缓存是一种用空间换取时间的技术,存在于计算机中很多地方,用来将一些慢速设备中的常用数据保存
在快速设备中,取数据的时候直接从快速设备中取,比如CPU二级缓存,WINDOWS文件读取缓存
缓存存在失效的问题:为了保证从缓存中读取数据和慢速数据中数据一致,则需要在慢速数据中对应
的数据发生变化的时候,清除缓存中相应的数据
缓存是改进网站性能的第一个手段,就像索引是改进数据库性能的第一个手段一样


ASP.NET缓存主要分为:

页面缓存,

数据源缓存,

数据缓存

1.页面缓存
给页面添加 <%@OutputCache Duration="15" VaryByParam="none"%> 标签就可以启用页面缓存,
这样整个页面的内容都会被缓存,页面中的ASP.NET代码,数据源在缓存期间都不会运行,而是直接
输出缓存的页面内容,Duration 表示缓存时候,以秒为单位,超过这个时间则缓存失效,再次生成以
后会再缓存15秒,以些类推,在在Page_Load处设置断点,修改数据库可测试
缓存是存在服务器,不是客户端,因为用HttpWatch还是能看到向服务器提交请求

缓存是针对所有这个页面的访问者,这样1个访问者和 1万个访问者,一次访问和100万次访问对数据
的压力是一样的
对于看新闻页面来讲,如果如上设置的话,则会缓存在第一个看到的新闻,因为?id=2,?id=3只是页面
的不两只参数而已,为了能让不同的新闻各自缓存,因此可以设置VaryByParam="id",表示对于不同的
id参数进行单独缓存,如果有多个确定缓存的参数,则将参数名用 分号; 隔开即可,比如VaryByParam="id;num"
如果想让任何不同的查询字符串都创建不同的缓存,则设置VaryByParam="*",一般情况下设置*就足够
在WebUserControl中也可以像页面缓存一样设置控件的缓存

2.数据源缓存

设定ObjectDataSource的CacheDuration (缓存时间:秒),EnableCaching=true
这样每隔CacheDuration指定的时间段才调用SelectMethod指定的方法来执行数据库查询,
其他时候都是直接返回缓存的数据。
缓存固定的时间适用于首页、文章列表等访问频繁的页面,对于看贴页面则不适合,假设有100万个
帖子,如果每个帖子都是固定缓存1小时的话,假设一小时之内有10万个帖子被看了,那么就要缓存
十万个帖子,非常占用内存,因为“百年一看”的“坟帖”偶然被访问一次也缓存一个小时,占用内存
这时候可以采用“滑动窗口(sliding)”策略,比如帖子缓存10分钟,如果10分钟之内又访问了,则
缓存的失效时间修改为从被访问这一刻起的10分钟之后,以此类推。这样经常访问的帖子就可以“长
期缓存”,而不经常访问的帖子也不会因为偶然访问而长期占用缓存。设置方法,
数据源:CacheExpirationPolicy="Sliding"
貌似滑动有问题。不是问题,Sliding只是策略,服务器会参考

3.缓存其他 (自定义缓存)

页面缓存、数据源缓存等内部都是使用HttpRuntime.Cache来实现缓存的,在一些页面缓存、数据源
缓存完成不了的特殊的缓存要求中,可以直接调用HttpRuntime.Cache进行缓存。在如鹏网项目中会讲到
(*)ASP.Net缓存默认是保存在内存中的,还可以配置保存到数据库中,大型网站还会配合使用Memcached等技术
清除缓存。在缓存还未失效的时候可能需要立即清空缓存,让数据库的修改立即反映到界面中
ASP.Net没有提供现成的方法,可以使用Hack级别的代码

// 保存缓存  null 表示缓存依赖
// 键,值,缓存依赖对象,绝对过期时间,区间
Cache.Insert("user","xgao",null,DateTime.Now.AddSeconds(10),TimeSpan.Zero);

// 简单保存缓存
Cache["test"] = "这是一个测试!";

//读取缓存
lbl.text = Cache["user"];

 

缓存依赖(依赖于文件)

原理:它是依赖指定的文件,一但文件被删除,修改,缓存将也会被删除

依赖于文件内容 CacheDependency cDep = new CacheDependency(filePath);

实例如下: 实现当文件内容不变的时候,就读缓存的,内容一变就更新缓存

protected void Page_Load(object sender, EventArgs e)
{
    
    if (Cache["fileText"] == null)
    {
    Response.Write("文件被修改,缓存被删除,但又创建了新的缓存,下次访问可得到!");
    // 得到指定文件的物理路径
    string filePath = Server.MapPath("~/CacheDep.txt");
    // 读取指定文件
    string Text = File.ReadAllText(filePath);

    // 创建缓存的文件依赖对象(依赖的文件的服务物理路径)
    CacheDependency cDep = new CacheDependency(filePath);
    // 键,值,缓存依赖对象,无绝对过期时间,无滑动过期时间,优先级,更新回调函数
    Cache.Insert("fileText", Text, cDep,
        System.Web.Caching.Cache.NoAbsoluteExpiration, System.Web.Caching.Cache.NoSlidingExpiration, 
        System.Web.Caching.CacheItemPriority.Normal, callback);
    }
    else
    {
    Response.Write(Cache["fileText"].ToString());
    }
}

// 缓存依赖的回调函数(当缓存被清除时调用些方法)
public void callback(string key, object value, CacheItemRemovedReason reason)
{
    string filePath = Server.MapPath("~/CacheLog.txt");
    string msg = "Cache["+key+"]="+value+"缓存被移除,因为:"+reason+"\r\n";
    File.AppendAllText(filePath, msg);
}

 

缓存依赖(依赖于数据库)

 

依赖于数据库内容(轮询机制/通知机制)

轮询机制: 是FW不定期的去检查数据库

1.在数据库新建版本表(ID、Ver 字段 用来保存某张表的版本)
2.在数据库新建触发器(比如在新闻表上新建)
3.使用C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727中的aspnet_regsql.exe:
注册:aspnet_regsql -S . -E -ed -d 数据库名 -et -t 版本表名
删除:aspnet_regsql -S . -E -d 数据库名 -dt -t 版本表名
取消数据库缓存依赖: aspnet_regsql -S . -E -dd 数据库名 版本表名
列出已注册表:aspnet_regsql -S . -E -d 数据库名 –lt
4.配置web.config
5.数据库依赖对象
SqlCacheDependency cDep = new SqlCacheDependency("DepName", "BankVer");

aspnet_regsql -S(服务器) . -E(集成登陆)/-U sa -P 123 -ed(启动/-dd关闭) -d(数据库名) GSSMS -et(指定缓存依赖的表名/-dt禁用表名) -t(表名) Aticle


数据库依赖 实例:

1>. 表bank如下:
------------------------
ID Name Maney
------------------------
1 xgao 1000
2 zsan 1
3 lshi 2000
------------------------

表BankVer如下:
--------------
ID VerNum
--------------
1 0
--------------

2>. 在bank表里创建触发器:

create Trigger [bankTri]
on [bank]
for insert,update,delete
as
  update BankVer set VerNum=VerNum+1

3>. 开启数据库的缓存依赖(CMD下运行):

aspnet_regsql -S . -E -ed -d TestData -et -t BankVer

4>. 在网站的 web.config配置在 <system.web> 下:

<caching>
  <sqlCacheDependency enabled="true">
    <databases>
        <!-- pollTime为轮询时间间隔 15 秒  -->
    <add name="DepName" connectionStringName="conStr" pollTime="15000"/>
    </databases>
  </sqlCacheDependency>
</caching>

5>. 前台代码:

 <form id="form1" runat="server">
    <div>
        <asp:GridView ID="gvBankList" runat="server"></asp:GridView>
    </div>
  </form>

6>. 后台代码:

 protected void Page_Load(object sender, EventArgs e)
        {
            if (Cache["bankList"] == null)
            {
                Response.Write("数据以更改!以下从数据库读取的!</br>");

                DataTable dt = GetBankList();
                gvBankList.DataSource = dt;
                gvBankList.DataBind();

                SqlCacheDependency sqlDep = new SqlCacheDependency("DepName", "BankVer");
                Cache.Insert("bankList", dt, sqlDep, System.Web.Caching.Cache.NoAbsoluteExpiration, System.Web.Caching.Cache.NoSlidingExpiration, CacheItemPriority.Normal,callback);

            }
            else
            {
                Response.Write("以下是从缓存里读取的</br>");
                gvBankList.DataSource = Cache["bankList"];
                gvBankList.DataBind();
            }
        }
        public void callback(string key,object value,CacheItemRemovedReason reason)
        {
            string filePath = Server.MapPath("~/CacheLog.txt");
            string msg = "数据库依赖 Cache[" + key + "]=" + value + "缓存被移除,因为:" + reason + "\r\n";
            File.AppendAllText(filePath, msg);
        }

        public DataTable GetBankList()
        {
            string connStr = ConfigurationManager.ConnectionStrings["ConnStr"].ConnectionString;
            SqlConnection conn = new SqlConnection(connStr);
            SqlCommand cmd = conn.CreateCommand();
            cmd.CommandText = "select * from bank";

            DataSet ds = new DataSet();
            SqlDataAdapter sda = new SqlDataAdapter(cmd);
            sda.Fill(ds);
            return ds.Tables[0];
        }

所使用的通知机制:是数据通信FW

 

posted @ 2014-12-19 16:39  悠悠清风~  阅读(433)  评论(0编辑  收藏  举报