天天看点

精进不休 .NET 4.0 (2) - asp.net 4.0 新特性之url路由

<a href="http://webabcd.blog.51cto.com/1787395/341053" target="_blank">[索引页]</a>

<a href="http://cid-ebc0a6c186317ea6.office.live.com/self.aspx/Share/VS2010.rar" target="_blank">[源码下载]</a>

精进不休 .NET 4.0 (2) - asp.net 4.0 新特性之url路由, 自定义CacheProvider, 新增的表达式&lt;%: expression %&gt;, QueryExtender控件, 其它新特性

介绍

asp.net 4.0 的新增功能

在 web form 中做 url 路由 

通过实现自定义的 CacheProvider ,来实现自定义的页面缓存逻辑 

新增的表达式 &lt;%: expression %&gt; 相当于 &lt;%= HttpUtility.HtmlEncode(expression) %&gt; 

控件 QueryExtender,对数据源控件获得的数据做再检索 

其它新特性

示例

1、web form 中的 url 路由的 demo

Global.asax.cs

using System; 

using System.Collections.Generic; 

using System.Linq; 

using System.Web; 

using System.Web.Security; 

using System.Web.SessionState; 

using System.Web.Routing; 

namespace AspDotNet 

        public class Global : System.Web.HttpApplication 

        { 

void Application_Start() void Application_Start(object sender, EventArgs e) 

                { 

                        // 关于 Routing 可以参考以前写的 http://webabcd.blog.51cto.com/1787395/341116 

                        // 其中 PageRouteHandler 类的作用是将 URL 路由的功能集成到 Web Form 中 

                        RouteTable.Routes.Add("myRoute", new Route("user/{userName}/{age}", new PageRouteHandler("~/UrlRouting/Default.aspx"))); 

                        /* 对应的配置如下,在 machine.config 中 

                        &lt;system.web&gt; 

                             &lt;httpmodule&gt; 

                                        &lt;add name="RoutingModule" type="System.Web.Routing.UrlRoutingModule"/&gt;    

                             &lt;/httpmodule&gt; 

                         */ 

                } 

        } 

}

UrlRouting/Default.aspx

&lt;%@ Page Title="" Language="C#" MasterPageFile="~/Site.Master" AutoEventWireup="true" 

        CodeBehind="Default.aspx.cs" Inherits="AspDotNet.UrlRouting.Default" %&gt; 

&lt;asp:Content ID="Content1" ContentPlaceHolderID="head" runat="server"&gt; 

&lt;/asp:Content&gt; 

&lt;asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" runat="server"&gt; 

        &lt;!-- 

                在页面上获取 url 路由而来的数据的方法 

                配合以下逻辑的路由规则是:"user/{userName}/{age}" 

        --&gt; 

        &lt;asp:Literal runat="server" Text="&lt;%$ RouteValue:userName%&gt;" /&gt; 

        &lt;br /&gt; 

        &lt;asp:Literal runat="server" Text="&lt;%$ RouteValue:age%&gt;" /&gt; 

                另外,对于数据源控件来说,也多了一种参数类型 RouteParameter    

&lt;%-- 

对应的配置如下,在 machine.config 中 

&lt;system.web&gt; 

     &lt;compilation debug="true" targetFrameworkMoniker=".NETFramework,Version=v4.0"&gt; 

            &lt;expressionBuilders&gt; 

                 &lt;add expressionPrefix="RouteValue" type="System.Web.Compilation.RouteValueExpressionBuilder" /&gt; 

            &lt;/expressionBuilders&gt; 

     &lt;/compilation&gt; 

--%&gt; 

UrlRouting/Default.aspx.cs

using System.Web.UI; 

using System.Web.UI.WebControls; 

namespace AspDotNet.UrlRouting 

        public partial class Default : System.Web.UI.Page 

void Page_Load() void Page_Load(object sender, EventArgs e) 

                        // 获取 url 路由而来的数据 

                        // 配合以下逻辑的路由规则是:"user/{userName}/{age}" 

                        Response.Write("userName: " + RouteData.Values["userName"].ToString()); 

                        Response.Write("&lt;br /&gt;"); 

                        Response.Write("age: " + RouteData.Values["age"].ToString()); 

UrlRouting/RouteUrlExpressionBuilderDemo.aspx

        CodeBehind="RouteUrlExpressionBuilderDemo.aspx.cs" Inherits="AspDotNet.UrlRouting.RouteUrlExpressionBuilderDemo" %&gt; 

                在页面上构造 url 路由的方式 

        &lt;asp:HyperLink ID="lnk1" runat="server" NavigateUrl="&lt;%$ RouteUrl:RouteName=myRoute, userName=webabcd, age=30 %&gt;" 

                Text="goto" /&gt; 

        &lt;br /&gt;&lt;br /&gt; 

        &lt;asp:HyperLink ID="lnk2" runat="server" Text="goto" /&gt; 

                 &lt;add expressionPrefix="RouteUrl" type="System.Web.Compilation.RouteUrlExpressionBuilder"/&gt; 

--%&gt;

UrlRouting/RouteUrlExpressionBuilderDemo.aspx.cs

using System.Web.Compilation; 

        public partial class RouteUrlExpressionBuilderDemo : System.Web.UI.Page 

                        // 在代码中构造 url 路由的方式 

                        string expression = String.Format("RouteName={0}, userName={1}, age={2}", "myRoute", "webabcd", "30"); 

                        lnk2.NavigateUrl = RouteUrlExpressionBuilder.GetRouteUrl(this, expression); 

2、自定义 CacheProvider

CachingEnhancement.aspx

&lt;%-- OutputCache 目前不支持直接设置 providerName 属性 --%&gt; 

&lt;%@ OutputCache Duration="30" VaryByParam="None" %&gt; 

        CodeBehind="CachingEnhancement.aspx.cs" Inherits="AspDotNet.CachingEnhancement" %&gt; 

        &lt;%= DateTime.Now.ToString() %&gt; 

&lt;/asp:Content&gt;

CachingEnhancement.aspx.cs

using System.Web.Caching; 

using System.Security.Cryptography; 

using System.Text; 

using System.IO; 

using System.Runtime.Serialization.Formatters.Binary; 

        public partial class CachingEnhancement : System.Web.UI.Page 

        // 重写 OutputCacheProvider 抽象类,以实现自定义的缓存实现(需要重写的方法是 Add, Get, Remove, Set) 

        // 本 Demo 演示了,如何开发一个自定义的 CacheProvider,来实现将数据缓存到硬盘的功能 

        public class FileCacheProvider : OutputCacheProvider 

                private string _cachePathPrefix = @"c:\"; 

string MD5() string MD5(string s) 

                        var provider = new MD5CryptoServiceProvider(); 

                        var bytes = Encoding.UTF8.GetBytes(s); 

                        var builder = new StringBuilder(); 

                        bytes = provider.ComputeHash(bytes); 

                        foreach (var b in bytes) 

                                builder.Append(b.ToString("x2").ToLower()); 

                        return builder.ToString(); 

                /// &lt;summary&gt; 

                /// 将指定的 key ,做md5 加密后,拼出一个路径,做为保存此 key 对应的对象的文件(此例只做演示用) 

                /// &lt;/summary&gt; 

                /// &lt;param name="key"&gt;缓存 key&lt;/param&gt; 

                /// &lt;returns&gt;&lt;/returns&gt; 

string GetPathFromKey() string GetPathFromKey(string key) 

                        return _cachePathPrefix + MD5(key) + ".txt"; 

                /// 将对象放入自定义的缓存中 

                /// &lt;param name="entry"&gt;缓存对象&lt;/param&gt; 

                /// &lt;param name="utcExpiry"&gt;缓存的过期时间&lt;/param&gt; 

override object Add() override object Add(string key, object entry, DateTime utcExpiry) 

                        var path = GetPathFromKey(key); 

                        // 指定的 key 已被缓存了,则不做任何处理 

                        if (File.Exists(path)) 

                                return entry; 

                        // 将对象缓存到硬盘上的指定文件 

                        using (var file = File.OpenWrite(path)) 

                        { 

                                var item = new CacheItem { Expires = utcExpiry, Item = entry }; 

                                var formatter = new BinaryFormatter(); 

                                formatter.Serialize(file, item); 

                        } 

                        return entry; 

                /// 在缓存中,根据指定的 key 获取缓存对象 

override object Get() override object Get(string key) 

                        // 未找到缓存 

                        if (!File.Exists(path)) 

                                return null; 

                        CacheItem item = null; 

                        // 如果有缓存的话,则取出缓存对象 

                        using (var file = File.OpenRead(path)) 

                                item = (CacheItem)formatter.Deserialize(file); 

                        // 缓存过期或缓存中无数据,则删除此缓存所对应的硬盘上的物理文件 

                        if (item == null || item.Expires &lt;= DateTime.Now.ToUniversalTime()) 

                                Remove(key); 

                        return item.Item; 

                /// 根据指定的 key 删除缓存对象 

override void Remove() override void Remove(string key) 

                                File.Delete(path); 

                /// 更新缓存 

override void Set() override void Set(string key, object entry, DateTime utcExpiry) 

                        var item = new CacheItem { Expires = utcExpiry, Item = entry }; 

                /// 封装了需要被缓存的对象的一个可序列化的对象 

                [Serializable] 

                internal class CacheItem 

                        /// &lt;summary&gt; 

                        /// 缓存对象 

                        /// &lt;/summary&gt; 

                        public object Item; 

                        /// 缓存对象的过期时间 

                        public DateTime Expires; 

Web.config

&lt;!--缓存配置--&gt; 

&lt;caching&gt; 

        &lt;!--默认的缓存实现是 AspNetInternalProvider(即 asp.net 自带的基于内存的缓存实现方式)--&gt; 

        &lt;outputCache defaultProvider="AspNetInternalProvider"&gt; 

                &lt;providers&gt; 

                        &lt;!-- 

                                新增一个缓存的 provider 配置 

                                具体实现见 CachingEnhancement.aspx.cs 

                        --&gt; 

                        &lt;add name="FileCache" type="AspDotNet.FileCacheProvider, AspDotNet"/&gt; 

                &lt;/providers&gt; 

        &lt;/outputCache&gt; 

&lt;/caching&gt;

                // 根据 HttpContext 的不同,可以为其指定不同的 CacheProvider 

override string GetOutputCacheProviderName() override string GetOutputCacheProviderName(HttpContext context) 

                        // 符合此条件的,则缓存的实现使用自定义的 FileCacheProvider 

                        // 自定义缓存实现见 CachingEnhancement.aspx.cs 

                        // CacheProvider 的配置见 web.config 

                        // 页面的缓存时间见 CachingEnhancement.aspx 

                        if (context.Request.Path.ToLower().EndsWith("cachingenhancement.aspx")) 

                                return "FileCache"; 

                        else 

                                return base.GetOutputCacheProviderName(context); 

3、表达式 &lt;%: expression %&gt; 的 demo

HtmlEncodedCodeExpressions.aspx

        CodeBehind="HtmlEncodedCodeExpressions.aspx.cs" Inherits="AspDotNet.HtmlEncodedCodeExpressions" %&gt; 

        &lt;%-- 

                新增的一个表达式 &lt;%: expression %&gt; 相当于 &lt;%= HttpUtility.HtmlEncode(expression) %&gt; 

        --%&gt; 

        &lt;%= "&lt;strong&gt;strong&lt;/strong&gt;" %&gt; 

        &lt;%: "&lt;strong&gt;strong&lt;/strong&gt;" %&gt; 

        &lt;%= HttpUtility.HtmlEncode("&lt;strong&gt;strong&lt;/strong&gt;") %&gt; 

4、QueryExtender 控件的 demo

QueryExtenderDemo.aspx

        CodeBehind="QueryExtenderDemo.aspx.cs" Inherits="AspDotNet.QueryExtenderDemo" %&gt; 

        &lt;asp:GridView ID="GridView1" runat="server" AutoGenerateColumns="False" DataSourceID="LinqDataSource1"&gt; 

                &lt;Columns&gt; 

                        &lt;asp:BoundField DataField="ProductId" HeaderText="ProductId" SortExpression="ProductId" /&gt; 

                        &lt;asp:BoundField DataField="ProductName" HeaderText="ProductName" SortExpression="ProductName" /&gt; 

                        &lt;asp:BoundField DataField="ProductPrice" HeaderText="ProductPrice" SortExpression="ProductPrice" /&gt; 

                &lt;/Columns&gt; 

        &lt;/asp:GridView&gt; 

        &lt;asp:LinqDataSource ID="LinqDataSource1" runat="server" ContextTypeName="AspDotNet.QueryExtenderDemo" 

                EntityTypeName="AspDotNet.Product" TableName="Data"&gt; 

        &lt;/asp:LinqDataSource&gt; 

                QueryExtender - 和数据源控件结合使用,以对数据源控件中检索到的数据做再次检索 

                        SearchExpression - 根据指定的字段查找指定的数据 

                        RangeExpression - 在指定字段中查找指定范围的数据 

                        PropertyExpression - 查找某字段的值为某指定的值的数据 

                        OrderByExpression - 用于排序数据 

                        CustomExpression - 自定义查询表达式 

        &lt;asp:QueryExtender ID="QueryExtender1" runat="server" TargetControlID="LinqDataSource1"&gt; 

                &lt;asp:SearchExpression DataFields="ProductName" SearchType="EndsWith"&gt; 

                        &lt;asp:Parameter Type="String" DefaultValue="0" /&gt; 

                &lt;/asp:SearchExpression&gt; 

        &lt;/asp:QueryExtender&gt; 

QueryExtenderDemo.aspx.cs

        public partial class QueryExtenderDemo : System.Web.UI.Page 

                // 为 GridView 提供数据 

                public List&lt;Product&gt; Data 

                        get 

                                Random random = new Random(); 

                                List&lt;Product&gt; products = new List&lt;Product&gt;(); 

                                for (int i = 0; i &lt; 100; i++) 

                                { 

                                        products.Add(new Product { ProductId = i + 1, ProductName = "名称" + i.ToString().PadLeft(2, '0'), ProductPrice = random.NextDouble() }); 

                                } 

                                return products; 

                // 为 GridView 提供数据的实体类 

                public class Product 

                        public int ProductId { get; set; } 

                        public string ProductName { get; set; } 

                        public double ProductPrice { get; set; } 

5、其他新特性的简单说明

Others.aspx

&lt;%@ Page Title="其它,一笔带过" Language="C#" MasterPageFile="~/Site.Master" AutoEventWireup="true" 

        CodeBehind="Others.aspx.cs" Inherits="AspDotNet.Others" %&gt; 

        &lt;style&gt; 

                body 

                        font-size: 12px; 

                textarea 

                        width: 99%; 

        &lt;/style&gt; 

        &lt;p&gt; 

                1、Permanent Redirect - 可以实现 301 跳转 

                &lt;ul&gt; 

                        &lt;li&gt;Response.RedirectPermanent() - 永久性重定向(http 301)。&lt;/li&gt; 

                        &lt;li&gt;Response.Redirect() - 临时性重定向(http 302)。&lt;/li&gt; 

                &lt;/ul&gt; 

        &lt;/p&gt; 

                2、Session 压缩(设置 sessionState 节点的 compressionEnabled 属性) 

                &lt;br /&gt; 

                对于使用进程外会话状态服务器的会话状态提供程序,或者将会话状态保存在 sqlserver 数据库中的会话状态提供程序,现在为提高其效率新增了压缩 Session 数据的功能(使用System.IO.Compression.GZipStream来压缩数据),像如下这样的配置 

                &lt;textarea rows="6"&gt; 

                        &lt;sessionState 

                            mode="SqlServer" 

                            sqlConnectionString="data source=dbserver;Initial Catalog=aspnetstate" 

                            allowCustomSqlDatabase="true" 

                            compressionEnabled="true" 

                        /&gt; 

                &lt;/textarea&gt; 

                3、httpRuntime 节点的新增配置 

                        &lt;li&gt;maxRequestPathLength - url 路径的最大长度(基于NTFS文件路径的最大长度就是 260)&lt;/li&gt; 

                        &lt;li&gt;maxQueryStringLength - url 的最大长度&lt;/li&gt; 

                        &lt;li&gt;requestPathInvalidChars - 指定 url 路径的无效字符&lt;/li&gt; 

                        &lt;li&gt;requestValidationType - 继承 System.Web.Util.RequestValidator 抽象类,重写其 IsValidRequestString() 方法,以实现自定义的请求验证。在 requestValidationType 可以指定使用这个自定义的类&lt;/li&gt; 

                        &lt;li&gt;encoderType - 重写 System.Web.Util.HttpEncoder,可以实现自定义的 html编码, url编码, http header编码。在 encoderType 指定这个自定义编码的类后,程序中所用到的 System.Web.HttpUtility 或 System.Web.HttpServerUtility 的相关方法将会使用自定义的编码实现&lt;/li&gt; 

                &lt;textarea rows="2"&gt; 

                        &lt;httpRuntime maxRequestPathLength="260" maxQueryStringLength="2048" requestPathInvalidChars="&lt;,&gt;,*,%,&amp;,:,\,?" requestValidationType="Samples.MyValidator, Samples" encoderType="Samples.MyEncoder, Samples" /&gt; 

                4、compilation 节点新增 targetFramework 属性,用于指定程序运行的目标框架 

                &lt;textarea&gt; 

                        &lt;compilation targetFramework="4.0" /&gt; 

                5、asp.net 4.0 结合 iis 7.5 可使 web 应用程序自动启动 

                在 web 程序中实现 System.Web.Hosting.IProcessHostPreloadClient 接口,用于被 iis 启动 

                6、Page 类中新增了两个属性,分别是 MetaDescription 和 MetaKeywords 

                7、以前每个可显示的控件都有 Enabled 属性(如果 Enabled="false" 则对应的 HTML 为 disabled="disabled"),但是 HTML 4.01 的标准是只有 input 才能有 disabled 属性 

                        &lt;li&gt; 

                                在 pages 节点中设置 controlRenderingCompatibilityVersion="3.5",则所有可显示控件都会输出 disabled="disabled" 

                        &lt;/li&gt; 

                                在 pages 节点中设置 controlRenderingCompatibilityVersion="4.0",则只有 input 元素才会输出 disabled="disabled",非 input 元素将会自动标记一个名为 aspnetdisabled 的 css 类 

                8、web form 需要在页面上写入隐藏域(如为了保存 ViewState 的隐藏域),在 asp.net 4.0 中系统将在这类的隐藏域外的 div 上标记一个名为 aspNetHidden 的 css 类,以方便样式控制 

                9、ASP.NET Chart Control - 实例参考 http://code.msdn.microsoft.com/mschart 

OK 

     本文转自webabcd 51CTO博客,原文链接:http://blog.51cto.com/webabcd/341124,如需转载请自行联系原作者