ASP.NET底层架构探索之再谈.NET运行时(一)

本文将重点讲解如何为特定的硬件设备自定义ASP.NET Web移动程序,以及使用模板自定义Form和Panel等控件。通过属性重写这个特性,使ASP.NET Web移动程序可以根据移动设备的硬件功能,为特定的硬件指定其控件的属性值。例如,应用程序可能需要一个Label控件在一些设备上显示较长的文本,而在另一些设备上显示较简短的文本。这就需要我们为每个控件都提供一组属性,应用程序可以针对各个设备重写这些属性。所有的ASP.NET移动控件都具有默认的外观和布局。对于ASP.NET移动控件,你可通过设置属性或使用样式来改变移动控件的默认外观。你还可以使用模板自定义某些移动控件的外观。本章还将重点介绍模板和模板集,以及它们的使用方法。

  自定义的方式

  在前面的几个专题中,我们提过如何使用设备筛选器为特定设备进行自定义。除此之外,我们还可以使用模板化技术和属性重写功能使你可以自定义你的应用程序,以便对特定的设备类型以不同的方式呈现控件。如果你在.aspx页面中为一个控件使用上述的任何一种自定义方法,都可以使用一个名为DeviceSpecific/Choice构造的移动控件语法。DeviceSpecific/Choice构造通常是为特定于设备的选项模板集以及控件内在多个备选内容之间指定一个选项。例如,下面的代码是一个Label控件的声明语法,在这个声明中包含了一个用来鉴别当前浏览器是否支持HTML的DeviceSpecific/Choice构造。

<mobile:Label id="Label1" runat="server" Text="Default text">
<DeviceSpecific>
<Choice Filter="isHTML32" Text="Text for selected devices" ></Choice>
</DeviceSpecific>
</mobile:Label>

  其中Filter属性指定了一个设备筛选器,该设备筛选器其名称为isHTML32,它用来鉴别一个移动设备是否内置了HTML浏览器。如果内置了HTML浏览器的话,将会在Label控件上应用"Text for selected device"这段文本。相反的,就不作任何的显示。正如上述的代码所示,在<DeviceSpecific></DeviceSpecific>内通常包含一个或多个<Choice>元素,每个元素都包含指定如何根据目标设备功能计算选项的属性。运行时依次计算每个选项,并使用成功计算出的第一个选项(这和C#里的switch case语句非常相似)。

  在开始学习如何定义一个设备筛选器前,先让我们先了解一些基本的概念,例如属性重写和模板化的具体含义。

  属性重写

  一般情况下,我们开发的ASP.NET Web移动程序并不是针对某个具体的设备,而是可以根据移动设备硬件特性的不同,通过适应性的调整从而可以在几乎所有的硬件设备上进行良好的呈现。

  但是因为品牌和型号的不同,这些移动设备间就不可避免地存在一些差异。例如支持的颜色、屏幕大小、输入功能以及浏览器使用的标记语言等。前面提到过,ASP.NET控件是可以被几乎所有的硬件设备支持的,因为它们本身就具有适应性调整的功能,而且控件设置的属性值也会伴随ASP.NET控件应用到具体的应用程序中,并不会因为硬件设备的不同使属性值不同。但是,在某些情况下你可能舍弃默认的呈现,采用自定义的呈现方式。一个典型的例子就是字符的显示,例如我们使用Label控件进行字符串的显示,在一些屏幕较小的设备上,每一行显示的字符数也是较少的,因此我们可以为该控件的Text属性设置为"我们使用ASP.NET",而针对一些大屏幕的设备,我们可以将Label控件的Text属性设置为"我们使用ASP.NET构建一个功能完善的移动应用程序",正是因为属性重写功能,我们才可以使用上述的方法来构建一个针对不同移动设备的硬件特性做出最好呈现的ASP.NET Web移动应用程序。

  模板

  Form、Panel、List和ObjectList等移动控件都是模板化的控件。开发人员通常使用模板来改变或丰富应用程序的外观或内容。例如,如果为Form控件指定并选择了页眉或页脚模板,则该模板中包含的标记将被添加到窗体内容中,分别作为页眉或页脚进行呈现。请注意模板不同于样式,模板定义要显示的内容和控件。当应用程序呈现模板附加到的控件时,即会呈现模板。而样式指定内容和控件的外观,您的应用程序可在不使用模板的情况下使用样式,它也可在模板内将样式应用于您定义的移动控件。 除了模板外,ASP.NET移动控件在此基础上还扩展了一个新的模型,并引入了模板集的概念。模板集是由模板组成的集合。但是,单个模板化控件可能引用多个模板集,而每个模板集都具有不同的特定于设备的条件。模板集的具体实现将在后续章节具体介绍。

  设备筛选器详解

  使用设备筛选器,应用程序可为特定硬件设备或设备类别自定义控件的外观。该自定义基于用来浏览应用程序的硬件设备的功能。 例如,假设开发人员正在开发主要用于支持位图 (.bmp) 图像的特定品牌手持式设备的应用程序。在此设备上,开发人员决定同时用来自.bmp文件的文本和图标显示List控件中的所有项。通过使用设备筛选器,应用程序可检测到它是否正在目标手持式设备上被用户浏览。这会导致应用程序使用同时用文本和图标显示列表项的设备特定的模板。此技术可为特定硬件设备自定义应用程序。进一步扩充此示例,假设您还希望可在显示.gif图像的任意类型设备上浏览该应用程序。您的应用程序可应用检测程序何时在此类设备上使用的设备筛选器。在该情况下,该程序指定使用文本和列表项图标的 .gif 图像(而不是.bmp图像)显示列表项的模板。这就为一类设备自定义了应用程序。

  设备筛选器可完成的一些其他任务包括:

  ■ 根据设备类型选择样式。

  ■ 在支持使用更为丰富的表现形式呈现控件的设备上,我们可以尽量利用该硬件设备的 性能,使用一些更加细致的、具体的呈现方式。

  ■ 在显示功能受限制的设备上,使用更为简单的表现形式呈现控件。

  在DeviceSpecific/Choice构造中的<Choice>元素依赖于移动设备的功能。当你使用某移动设备请求一个ASP.NET移动页面时,首先就会将包含该移动设备型号和内置浏览器等信息通过HTTP文件头传送到服务器端,这时服务器就会根据HTTP文件头传送的这些信息构建一个System.Web.Mobile.MobileCapabilities对象,以此来响应客户端请求。而设备筛选器就是利用MobileCapabilities对象的一些只读属性进行工作的。一个典型的例子就是设备筛选器如何鉴别那些支持HTML 3.2标记语言的浏览器,实现这种鉴别就是检测和客户请求相关的MobileCapabilities对象的PreferredRenderingType属性值是否为"html32",是的话就是支持HTML 3.2标记语言的浏览器。

  注意,MobileCapabilities对象的各个属性是和浏览页面的移动设备戚戚相关的。例如,当使用Pocket Internet Explorer浏览器请求页面时,MobileCapabilities对象的Browser属性将被设置为"Pocket IE", PreferredRenderingType属性值设置为"html32",而ScreenPixelsWidth 和ScreenPixelsHeight属性取决于移动设备的具体型号,因为Pocket PC、Smartphone和Windows CE .NET都可以使用Pocket Internet Explorer浏览器,但是这些设备屏幕的分辨率是不同的。

  因为不同移动设备使用的浏览器是不同的,因此相应的MobileCapabilities对象各个属性也存在差异。如果你要查看特定浏览器的MobileCapabilities对象,你可以在C:\WINDOWS\Microsoft.NET\Framework\[版本号]\CONFIG\Browsers目录下,查看openwave、Pocket Internet Explorer、palm和nokia等众多浏览器的信息。下面是openwave浏览器对应文件的一段代码片断,在使用该浏览器请求页面时,这些信息就会通过HTTP文件头传送到服务器端,并构建具有类似属性值的MobileCapabilities对象。

<capabilities>
<capability name="browser" value="Phone.com" />
<capability name="canInitiateVoiceCall" value="true" />
<capability name="canSendMail" value="false" />
<capability name="deviceID" value="${deviceID}" />
<capability name="deviceVersion" value="${deviceVersion}" />
<capability name="inputType" value="telephoneKeypad" />
<capability name="isMobileDevice" value="true" />
<capability name="majorVersion" value="${browserMajorVersion}" />
<capability name="maximumRenderedPageSize" value="1492" />
<capability name="minorVersion" value="${browserMinorVersion}" />
<capability name="numberOfSoftkeys" value="2" />
<capability name="optimumPageWeight" value="700" />
<capability name="preferredImageMime" value="image/vnd.wap.wbmp" />
<capability name="preferredRenderingMime" value="text/vnd.wap.wml" />
<capability name="preferredRenderingType" value="wml11" />
<capability name="requiresAdaptiveErrorReporting" value="true" />
<capability name="rendersBreakBeforeWmlSelectAndInput" value="true" />
<capability name="rendersWmlDoAcceptsInline" value="false" />
<capability name="rendersWmlSelectsAsMenuCards" value="true" />
<capability name="requiresFullyQualifiedRedirectUrl" value="true" />
<capability name="requiresNoescapedPostUrl" value="true" />
<capability name="requiresPostRedirectionHandling" value="true" />
<capability name="supportsRedirectWithCookie" value="false" />
<capability name="type"
value="Phone.com${browserMajorVersion}" />
<capability name="version"
value="${browserMajorVersion}${browserMinorVersion}" />
</capabilities>

 

<%@ Page Inherits="System.Web.UI.MobileControls.MobilePage"
Language="c#" %>
<script language="c#" runat="server">
public void Page_Load(Object sender, EventArgs e)
{
 System.Web.Mobile.MobileCapabilities currentCapabilities;
 MobileCapabilities currentCapabilities = (MobileCapabilities)Request.Browser;
 // 创建一个MobileCapabilities对象来获取浏览器的信息

 if(currentCapabilities.PreferredRenderingMIME=="text/html")
 {
  Label2.Text = "你使用的是支持HTML的移动设备";
  //如果是支持HTML的移动设备的话,显示上面的文本信息
 }
 else if(currentCapabilities.PreferredRenderingMIME == "text/vnd.wap.wml")
 {
  Label2.Text = "你使用的是支持WML的移动设备";
  //如果是支持WML的移动设备的话,显示上面的文本信息
 }
 Label1.Text = "屏幕的宽度(字符数): " + currentCapabilities.ScreenCharactersWidth;
 //显示当前移动设备每一行可以显示的字符数
}
</script>
 <Mobile:Form runat="server" id=frmTemplate >
  <mobile:label ID="Label1" runat="server" />
  <mobile:label ID="Label2" runat="server" />
 </Mobile:Form>

 

  提示:在某些情况下,某特定移动设备的MobileCapabilities对象获得的属性值并不是你期望的值。例如,我们经常在开发过程中直接使用桌面的IE来测试程序,你也许会认为PreferredRenderingType属性值为"html40",而实际上这个值为"html32",这是因为ASP.NET移动控件只支持HTML 3.2,所以发送到桌面IE的标记语言依然是HTML 3.2。   当然,你还可以通过代码的方式将请求页面的浏览器信息通过MobileCapabilities对象显示页面上,代码如下:
共2页: 上一页 1 [2]
下一页

时间: 2025-01-27 22:56:33

ASP.NET底层架构探索之再谈.NET运行时(一)的相关文章

ASP.NET底层架构探索之再谈.NET运行时(二)

在这里我们有一个在ISAPI扩展中活动的,可调用的ISAPIRuntime对象的实例.每次运行时是启动的并运行着的时候(译注:相对的,如果运行时并没有启动,就需要象上一章所说的那样载入运行时),ISAPI的代码调用ISAPIRuntime.ProcessRequest()方法,这个方法是真正的进入ASP.NET管道的入口,这个流程在图4中显示. 记住ISAPI是多线程的,所以请求也会通过AppDomainFactory.Create()(译注:原文为ApplicationDomainFactor

ASP.NET底层架构探索之再谈.NET运行时

在这里我们有一个在ISAPI扩展中活动的,可调用的ISAPIRuntime对象的实例.每次运行时是启动的并运行着的时候(译注:相对的,如果运行时并没有启动,就需要象上一章所说的那样载入运行时),ISAPI的代码调用ISAPIRuntime.ProcessRequest()方法,这个方法是真正的进入ASP.NET管道的入口,这个流程在图4中显示. 记住ISAPI是多线程的,所以请求也会通过AppDomainFactory.Create()(译注:原文为ApplicationDomainFactor

ASP.NET底层架构探索之进入ASP.NET

asp.net|架构 ASP.NET是一个非常强大的构建Web应用的平台,它提供了极大的灵活性和能力以致于可以用它来构建所有类型的Web应用.绝大多数的人只熟悉高层的框架如WebForms和WebServices-这些都在ASP.NET层次结构在最高层.在这篇文章中我将会讨论ASP.NET的底层机制并解释请求(request)是怎么从Web服务器传送到ASP.NET运行时然后如何通过ASP.NET管道来处理请求. 对我而言了解平台的内幕通常会带来满足感和舒适感,深入了解也能帮助我写出更好的应用.

ASP.NET底层架构探索之进入.NET运行时

进入.NET运行时的真正的入口发生在一些没有被文档记载的类和接口中(译著:当然,你可以用Reflector来查看J).除了微软,很少人知道这些接口,微软的家伙们也并不热衷于谈论这些细节,他们认为这些实现细节对于使用ASP.NET开发应用的开发人员并没有什么用处. 工作进程(IIS5中是ASPNET_WP.EXE,IIS6中是W3WP.EXE)寄宿.NET运行时和ISAPI DLL,它(工作进程)通过调用COM对象的一个小的非托管接口最终将调用发送到ISAPIRuntime类的一个实例上(译注:原

ASP.NET底层架构探索之IE6的革新

asp.net|架构 IIS 5 和6以不同的方式工作 当一个请求来到时,IIS检查脚本映射(扩展名映射)然后把请求路由到aspnet_isapi.dll.这个DLL的操作和请求如何进入ASP.NET运行时在IIS5和6中是不同的.图2显示了这个流程的一个粗略概览. 在IIS5中,aspnet_isapi.dll直接寄宿在inetinfo.exe进程中,如果你设置了Web站点或虚拟目录的隔离度为中或高,则会寄宿在IIS单独的(被隔离的)工作进程中.当第一个ASP.NET请求来到,DLL(aspn

ASP.NET底层架构探索之HttpHandlers

asp.net|架构 模块是相当底层的,而且对每个来到ASP.NET应用程序的请求都会被触发.Http处理器更加的专注并处理映射到这个处理器上的请求. Http处理器需要实现的东西非常简单,但是通过访问HttpContext对象它可以变得非常强大.Http处理器通过实现一个非常简单的IHttpHandler接口(或是它的异步版本,IHttpAsyncHandler),这个接口甚至只含有一个方法-ProcessRequest()-和一个属性IsReusable.关键部分是ProcessReques

ASP.NET底层架构探索之处理请求

asp.net|架构|请求 当一个请求到来时,它被路由到ISAPIRuntime.ProcessRequest()方法.这个方法调用HttpRuntime.ProcessRequest方法,它作一些重要的事情(用Reflector查看System.Web.HttpRuntime.ProcessRequestInternal方法): ·为请求创建一个新的HttpContext实例 ·获取一个HttpApplication实例 ·调用HttpApplication.Init()方法来设置管道的事件

ASP.NET底层架构探索之ASP.NET管道

asp.net|架构 HttpApplication触发事件来通知你的程序有事发生,以此来负责请求流转.这作为HttpApplication.Init()函数的一部分发生(用Reflector查看System.Web.HttpApplication.InitInternal()方法和HttpApplication.ResumeSteps()方法来了解更多详情),连续设置并启动一系列事件,包括执行所有的处理器(handler).这些事件处理器映射到global.asax中自动生成的哪些事件中,同时

了解ASP.NET底层架构

asp.net|架构 了解ASP.NET底层架构 进入底层 这篇文章以非常底层的视角讲述了Web请求(request)在ASP.NET框架中是如何流转的,从Web服务器,通过ISAPI直到请求处理器(handler)和你的代码.看看在幕后都发生了些什么,不要再把ASP.NET看成一个黑盒了.   ASP.NET是一个非常强大的构建Web应用的平台,它提供了极大的灵活性和能力以致于可以用它来构建所有类型的Web应用.绝大多数的人只熟悉高层的框架如WebForms和WebServices-这些都在A