`
jiagou
  • 浏览: 2608111 次
文章分类
社区版块
存档分类
最新评论

使用ASP.NET 2.0 Profile存储用户信息[翻译] Level 200

 
阅读更多

作者: Stephen Walther
原文地址:http://msdn.microsoft.com/asp.net/default.aspx?pull=/library/en-us/dnvs05/html/UserProfiles.asp
译者:Tony Qu

概要:许多ASP.NET应用程序需要跨访问的用户属性跟踪功能,在ASP.NET1.1中,我们只能人工实现这一功能。但如今,使用 ASP.NET 2.0的Profile对象,这个过程变得异常简单。Stephen Walther将验证该对象,并向你展示如何使用Profile来跟踪用户属性、创建一个购物篮,及其他一些例子。

总目录
介绍
User Profile总揽
定义User Profile
使用Profile组
使用复杂的Profile属性
继承一个Profile
迁移匿名Profile设置
配置Profile Provider
管理Profiles并生成Profile报告
总结
相关书籍

Microsoft ASP.NET 2.0支持被称为Profile的新对象,它可以自动在多个Web应用程序的访问之间存储用户信息。一个User Profile中可以存储各种类型的信息,这些信息既可以是简单的string和integer类型,也可以是复杂的自定义类型。例如,你可以存储用户的姓、购物篮、用户属性或网站使用情况统计。
本文中,你将学习如何在一个应用中定义user profile。我们也会向你演示如何配置使用不同provider的profile。最后,你将学习如何管理和生成user profile的报告。

User Profiles总揽
Profile 对象与Session对象十分相似,但是更好用一些。与Session相似的地方在于,Profile是相对于一个特定的用户的,也就是说,每个Web应用程序的用户都有他们自己的profile对象。与Session不同的是,Profile对象是持久对象。如果你向Session中添加一个项,在你离开网站时,该项就会消失。而Profile则完全不同,当你修改Profile的状态时,修改在多个访问之间均有效。

profile使用provider模式来存储信息,默认情况下,user profile的内容会保存在SQL Server Express数据库中,该数据库位于网站的App_Data目录。然而,在本文的后半部分,你将了解如何使用其他数据提供者(data provider)来存储信息,如完整版的SQL Server中的一个数据库或者一个Oracle数据库。

与Session不同,Profile是强类型的,Session对象仅仅是一个项集合而已,而profile对象则有强类型属性。
使用强类型是有它的道理的。例如,使用强类型,你就可以在Microsoft Visual Web Developer中使用智能感知技术,当你键入Profile和一个点的时候,智能感知会弹出你已经定义过的profile属性列表。

定义user profile
你既可以在machine.config中,也可以在web.config中定义一个user profile,由于你不能在应用程序的二级目录中创建一个包含文件profile节的web.config文件,这意味着你将无法在一个应用程序中定义两个以上的profile。
在列表1的web.config中,列举了一个简单的profile定义的实例,该profile有三个属性,FirstName, LastName和PageVisits。

列表1
<configuration>
<system.web>
<authenticationmode="Forms"/>

<anonymousIdentificationenabled="true"/>

<profile>
<properties>
<add
name="FirstName"
defaultValue
="??"
allowAnonymous
="true"/>
<add
name="LastName"
defaultValue
="??"
allowAnonymous
="true"/>
<add
name="PageVisits"
type
="Int32"
allowAnonymous
="true"/>
</properties>
</profile>
</system.web>
</configuration>

由于该profile需要同时被匿名用户和已认证用户使用,因此我们在web.config文件中增加包含一个< anonymousIdentification>元素,有了这个元素,系统就会自动为匿名用户生成唯一的ID。仔细看的话我们会发现,每一个 profile属性都有一个allowAnonymous特性,该特性表明这个profile属性是否允许被匿名用户使用。

默认的profile属性类型是System.String类型。列表1中,由于没有为FirstName和LastName这两个profile属性增加type特性,那么系统默认它们是string类型,而PageVisits属性则指定了type特性为Int32,因此该profile属性可用于表示一个整型值。

最后,注意FirstName和LastName属性都有defaultValue特性。你可以为简单的数据类型设置defaultValue特性,但你不能为复杂类型设置defaultValue特性。

当你定义好一个profile之后,系统会自动在下一次页面被调用时,生成一个与该profile相对应的类。这个类会被保存在"Temporary ASP.NET Files Directory"目录(该目录也用于存放用于动态生成页面的类)。你可以使用HttpContext的Profile属性(Property)调用该类。
当你定义好一个profile后,你可以使用如下方法为profile属性赋值。

[VisualBasic.NET]
Profile.FirstName
="Bill"

[C#]
Profile.FirstName
="Bill";

任何在web.config中定义的profile属性都会在Profile对象中呈现。
列表2演示了你该如何使用profile来持久化保存用户信息。这个页显示了FirstName,LastName, PageVisits三个属性的值,同时它包含了一个能够用于修改这三个属性的表单(form)。在Page_Load中更新PageVisits的值,这意味着每一次刷新页面,PageVisits的值都会改变。

图1 使用简单的profile

列表 2. Simple.aspx (Visual Basic .NET)
<%@PageLanguage="VB"%>
<scriptrunat="server">

SubPage_Load()
Profile.PageVisits
+=1
EndSub

SubUpdateProfile(ByValsAsObject,ByValeAsEventArgs)
Profile.FirstName
=txtFirstName.Text
Profile.LastName
=txtLastName.Text
EndSub

</script>

<html>
<head>
<title>Simple</title>
</head>
<body>
<formid="form1"runat="server">
<b>Name:</b><%=Profile.FirstName%><%=Profile.LastName%>
<br/>
<b>PageVisits:</b><%=Profile.PageVisits%>

<hr/>

<b>FirstName:</b>
<asp:TextBoxID="txtFirstName"Runat="Server"/>
<br/>
<b>LastName:</b>
<asp:TextBoxID="txtLastName"Runat="Server"/>
<br/>
<asp:Button
Text
="UpdateProfile"
OnClick
="UpdateProfile"
Runat
="server"/>

</form>
</body>
</html>

列表 2. Simple.aspx (C#)
<%@PageLanguage="C#"%>
<scriptrunat="server">

voidPage_Load(){
Profile.PageVisits
++;
}

voidUpdateProfile(Objects,EventArgse){
Profile.FirstName
=txtFirstName.Text;
Profile.LastName
=txtLastName.Text;
}

</script>

<html>
<head>
<title>Simple</title>
</head>
<body>
<formid="form1"runat="server">
<b>Name:</b><%=Profile.FirstName%><%=Profile.LastName%>
<br/>
<b>PageVisits:</b><%=Profile.PageVisits%>

<hr/>

<b>FirstName:</b>
<asp:TextBoxID="txtFirstName"Runat="Server"/>
<br/>
<b>LastName:</b>
<asp:TextBoxID="txtLastName"Runat="Server"/>
<br/>
<asp:ButtonID="Button1"
Text
="UpdateProfile"
OnClick
="UpdateProfile"
Runat
="server"/>

</form>
</body>
</html>

如果你多次访问列表2中的页面,你会注意到PageVisits在不断增大。如果你关闭的浏览器,并在一周之后调用该页面,PageVisits属性仍然会保留原值。从这一点可以看出Profile为每个用户自动保存一个副本。

使用Profile组

尽管你仅可以为一个应用程序定义一个profile,但如果你需要让几个profile属性一起工作,把它们放在组中,会让你觉得它们更易管理。

例如,在列表3中,有一个带有两个组的profile,这两个组分别是Address和Preferences

列表3. Web.Config
<configuration>
<system.web>

<anonymousIdentificationenabled="true"/>

<profile>
<properties>
<groupname="Address">
<add
name="Street"
allowAnonymous
="true"/>
<add
name="City"
allowAnonymous
="true"/>
</group>
<groupname="Preferences">
<add
name="ReceiveNewsletter"
type
="Boolean"
defaultValue
="false"
allowAnonymous
="true"/>
</group>
</properties>
</profile>
</system.web>
</configuration>

当你用组来定义profile时,你应该使用组名来设置或读取profile属性。例如,在列表3中,你可以使用以下一些句子来完成三个profile属性的赋值。

[VisualBasic.NET]

Profile.Address.City
="Modesto"
Profile.Address.Street
="111KingArthurLn"
Profile.Preferences.ReceiveNewsletter
=False

[C#]

Profile.Address.City
="Modesto";
Profile.Address.Street
="111KingArthurLn";
Profile.Preferences.ReceiveNewsletter
=false;

一个profile的定义只能包含一层组,换句话说,你不能把其他的组放在一个profile组的下面一层。

使用复杂的profile属性

到目前为止,我们已经介绍了声明包含简单类型(如string或整型)属性的profile,其实你也可以在profile中声明复杂属性。
举个例子,假设你现在需要在profile中存储一个购物篮,如果这样做的话,你就可以在每次访问网站时获得自己的购物篮。
列表4 声明了一个包含profile,这个profile包含一个名为ShoppingCart的属性,而该属性的type特性是一个叫ShoppingCart的类(我们接下来会创建该类),该类名是有效的。
我们还会注意到,该声明中包含一个serializeAs特性,该特性可以帮助ShoppingCart使用二进制序列化器(binary serializer)进行持久化,而不是使用xml序列化器。

列表4 Web.config

<
configuration>
<system.web>

<anonymousIdentificationenabled="true"/>

<profile>
<properties>
<add
name="ShoppingCart"
type
="ShoppingCart"
serializeAs
="Binary"
allowAnonymous
="true"/>
</properties>
</profile>
</system.web>
</configuration>

列表5 中有一个简单购物篮的实现代码,该购物篮拥有添加和删除项(item)的方法(method),同时它还拥有两个属性(property),一个是用于获得该购物篮中的所有项的,一个是用于表示所有商品的总价的。

列表5 ShoppingCart (Visual Basic.NET)

Imports
Microsoft.VisualBasic

<Serializable()>_
PublicClassShoppingCart
Public_CartItemsAsNewHashtable()

'ReturnalltheitemsfromtheShoppingCart
PublicReadOnlyPropertyCartItems()AsICollection
Get
Return_CartItems.Values
EndGet
EndProperty

'Thesumtotaloftheprices
PublicReadOnlyPropertyTotal()AsDecimal
Get
DimsumAsDecimal
ForEachitemAsCartItemIn_CartItems.Values
sum
+=item.Price*item.Quantity
Next
Returnsum
EndGet
EndProperty

'Addanewitemtotheshoppingcart
PublicSubAddItem(ByValIDAsInteger,_
ByValNameAsString,ByValPriceAsDecimal)
DimitemAsCartItem=CType(_CartItems(ID),CartItem)
IfitemIsNothingThen
_CartItems.Add(ID,
NewCartItem(ID,Name,Price))
Else
item.Quantity
+=1
_CartItems(ID)
=item
EndIf
EndSub

'Removeanitemfromtheshoppingcart
PublicSubRemoveItem(ByValIDAsInteger)
DimitemAsCartItem=CType(_CartItems(ID),CartItem)
IfitemIsNothingThen
Return
EndIf
item.Quantity
-=1
Ifitem.Quantity=0Then
_CartItems.Remove(ID)
Else
_CartItems(ID)
=item
EndIf
EndSub

EndClass

<Serializable()>_
PublicClassCartItem

Private_IDAsInteger
Private_NameAsString
Private_PriceAsDecimal
Private_QuantityAsInteger=1

PublicReadOnlyPropertyID()AsInteger
Get
Return_ID
EndGet
EndProperty

PublicReadOnlyPropertyName()AsString
Get
Return_Name
EndGet
EndProperty

PublicReadOnlyPropertyPrice()AsDecimal
Get
Return_Price
EndGet
EndProperty

PublicPropertyQuantity()AsInteger
Get
Return_Quantity
EndGet
Set(ByValvalueAsInteger)
_Quantity
=value
EndSet
EndProperty

PublicSubNew(ByValIDAsInteger,_
ByValNameAsString,ByValPriceAsDecimal)
_ID
=ID
_Name
=Name
_Price
=Price
EndSub
EndClass

列表5 ShoppingCart (c#)

using
System;
usingSystem.Collections;

[Serializable]
publicclassShoppingCart
{
publicHashtable_CartItems=newHashtable();

//ReturnalltheitemsfromtheShoppingCart
publicICollectionCartItems
{
get{return_CartItems.Values;}
}

//Thesumtotaloftheprices
publicdecimalTotal
{
get
{
decimalsum=0;
foreach(CartItemitemin_CartItems.Values)
sum
+=item.Price*item.Quantity;
returnsum;
}
}

//Addanewitemtotheshoppingcart
publicvoidAddItem(intID,stringName,decimalPrice)
{
CartItemitem
=(CartItem)_CartItems[ID];
if(item==null)
_CartItems.Add(ID,
newCartItem(ID,Name,Price));
else
{
item.Quantity
++;
_CartItems[ID]
=item;
}
}

//Removeanitemfromtheshoppingcart
publicvoidRemoveItem(intID)
{
CartItemitem
=(CartItem)_CartItems[ID];
if(item==null)
return;
item.Quantity
--;
if(item.Quantity==0)
_CartItems.Remove(ID);
else
_CartItems[ID]
=item;
}

}

[Serializable]
publicclassCartItem
{
privateint_ID;
privatestring_Name;
privatedecimal_Price;
privateint_Quantity=1;

publicintID
{
get{return_ID;}
}

publicstringName
{
get{return_Name;}
}

publicdecimalPrice
{
get{return_Price;}
}

publicintQuantity
{
get{return_Quantity;}
set{_Quantity=value;}
}

publicCartItem(intID,stringName,decimalPrice)
{
_ID
=ID;
_Name
=Name;
_Price
=Price;
}
}

如果你把列表5中的代码添加到应用程序的App_Code目录中,购物篮会自动被编译。

在列表5中有一点值得注意,那就是ShoppingCart和CartItem类都加上了可序列化的特性,这一点对于他们能否被序列化十分重要,只有这样才能保存在Profile对象中。

最后,列表6的页面显示了可以被添加到购物篮中的产品。购物篮是通过BindShoppingCart方法从Profile对象中载入,该方法把购物篮中的对象绑定到一个GridView对象上,这些对象可以通过ShoppingCart类的CartItems属性获得。


图2 在profile中存储购物篮

AddCartItem方法用于在购物篮中添加一个产品,该方法中包含了检测Profile是否存在ShoppingCart的代码。对于Profile中存储的对象,你必须自己实例化这些对象,他们不会自动实例化。

RemoveCartItem方法用于从购物篮中移除一个产品,该方法只是简单地通过调用Profile中的ShoppingCart对象的RemoveItem方法。

列表 6 - Products.aspx (Visual Basic .NET)

<
%@PageLanguage="VB"%>

<scriptrunat="server">

SubPage_Load()
IfNotIsPostBackThen
BindShoppingCart()
EndIf
EndSub

SubBindShoppingCart()
IfNotProfile.ShoppingCartIsNothingThen
CartGrid.DataSource
=Profile.ShoppingCart.CartItems
CartGrid.DataBind()
lblTotal.Text
=Profile.ShoppingCart.Total.ToString("c")
EndIf
EndSub

SubAddCartItem(ByValsAsObject,ByValeAsEventArgs)
DimrowAsGridViewRow=ProductGrid.SelectedRow

DimIDAsInteger=CInt(ProductGrid.SelectedDataKey.Value)
DimNameAsString=row.Cells(1).Text
DimPriceAsDecimal=CDec(row.Cells(2).Text)

IfProfile.ShoppingCartIsNothingThen
Profile.ShoppingCart
=NewShoppingCart
EndIf
Profile.ShoppingCart.AddItem(ID,Name,Price)
BindShoppingCart()
EndSub

SubRemoveCartItem(ByValsAsObject,ByValeAsEventArgs)
DimIDAsInteger=CInt(CartGrid.SelectedDataKey.Value)
Profile.ShoppingCart.RemoveItem(ID)
BindShoppingCart()
EndSub
</script>

<html>
<head>
<title>Products</title>
</head>
<body>
<formid="form1"runat="server">

<tablewidth="100%">
<tr>
<tdvalign="top">
<h2>Products</h2>
<asp:GridView
ID
="ProductGrid"
DataSourceID
="ProductSource"
DataKeyNames
="ProductID"
AutoGenerateColumns
="false"
OnSelectedIndexChanged
="AddCartItem"
ShowHeader
="false"
CellPadding
="5"
Runat
="Server">
<Columns>
<asp:ButtonField
CommandName
="select"
Text
="Buy"/>
<asp:BoundField
DataField
="ProductName"/>
<asp:BoundField
DataField
="UnitPrice"
DataFormatString
="{0:c}"/>
</Columns>
</asp:GridView>




<asp:SqlDataSource
ID
="ProductSource"
ConnectionString
=
"Server=localhost;Database=Northwind;Trusted_Connection=true;"
SelectCommand
=
"SELECTProductID,ProductName,UnitPriceFROMProducts"
Runat
="Server"/>
</td>
<tdvalign="top">
<h2>ShoppingCart</h2>
<asp:GridView
ID
="CartGrid"
AutoGenerateColumns
="false"
DataKeyNames
="ID"
OnSelectedIndexChanged
="RemoveCartItem"
CellPadding
="5"
Width
="300"
Runat
="Server">
<Columns>
<asp:ButtonField
CommandName
="select"
Text
="Remove"/>
<asp:BoundField
DataField
="Name"
HeaderText
="Name"/>
<asp:BoundField
DataField
="Price"
HeaderText
="Price"
DataFormatString
="{0:c}"/>
<asp:BoundField
DataField
="Quantity"
HeaderText
="Quantity"/>
</Columns>
</asp:GridView>
<b>Total:</b>
<asp:LabelID="lblTotal"Runat="Server"/>
</td>
</tr>
</table>
</form>
</body>
</html>

列表 6. Products.aspx (C#)

<%
@PageLanguage="C#"%>
<%@ImportNamespace="System.Globalization"%>
<scriptrunat="server">

voidPage_Load(){
if(!IsPostBack)
BindShoppingCart();
}

voidBindShoppingCart()
{
if(Profile.ShoppingCart!=null)
{
CartGrid.DataSource
=Profile.ShoppingCart.CartItems;
CartGrid.DataBind();
lblTotal.Text
=Profile.ShoppingCart.Total.ToString("c");
}
}

voidAddCartItem(Objects,EventArgse)
{
GridViewRowrow
=ProductGrid.SelectedRow;

intID=(int)ProductGrid.SelectedDataKey.Value;
StringName
=row.Cells[1].Text;
decimalPrice=Decimal.Parse(row.Cells[2].Text,
NumberStyles.Currency);

if(Profile.ShoppingCart==null)
Profile.ShoppingCart
=newShoppingCart();

Profile.ShoppingCart.AddItem(ID,Name,Price);
BindShoppingCart();
}

voidRemoveCartItem(Objects,EventArgse)
{
intID=(int)CartGrid.SelectedDataKey.Value;
Profile.ShoppingCart.RemoveItem(ID);
BindShoppingCart();
}
</script>

<html>
<head>
<title>Products</title>
</head>
<body>
<formid="form1"runat="server">

<tablewidth="100%">
<tr>
<tdvalign="top">
<h2>Products</h2>
<asp:GridView
ID
="ProductGrid"
DataSourceID
="ProductSource"
DataKeyNames
="ProductID"
AutoGenerateColumns
="false"
OnSelectedIndexChanged
="AddCartItem"
ShowHeader
="false"
CellPadding
="5"
Runat
="Server">
<Columns>
<asp:ButtonField
CommandName
="select"
Text
="Buy"/>
<asp:BoundField
DataField
="ProductName"/>
<asp:BoundField
DataField
="UnitPrice"
DataFormatString
="{0:c}"/>
</Columns>
</asp:GridView>




<asp:SqlDataSource
ID
="ProductSource"
ConnectionString
=
"Server=localhost;Database=Northwind;Trusted_Connection=true;"
SelectCommand
=
"SELECTProductID,ProductName,UnitPriceFROMProducts"
Runat
="Server"/>
</td>
<tdvalign="top">
<h2>ShoppingCart</h2>
<asp:GridView
ID
="CartGrid"
AutoGenerateColumns
="false"
DataKeyNames
="ID"
OnSelectedIndexChanged
="RemoveCartItem"
CellPadding
="5"
Width
="300"
Runat
="Server">
<Columns>
<asp:ButtonField
CommandName
="select"
Text
="Remove"/>
<asp:BoundField
DataField
="Name"
HeaderText
="Name"/>
<asp:BoundField
DataField
="Price"
HeaderText
="Price"
DataFormatString
="{0:c}"/>
<asp:BoundField
DataField
="Quantity"
HeaderText
="Quantity"/>
</Columns>
</asp:GridView>
<b>Total:</b>
<asp:LabelID="lblTotal"Runat="Server"/>
</td>
</tr>
</table>
</form>
</body>
</html>

继承一个profile
你也可以通过从一个已经存在的profile类中继承一个profile来完成对profile的定义,这种特性能够帮助你在多个应用程序中使用相同的profile。
例如,列表7中列出了一个拥有多个用户属性的类,该类是从ProfileBase类继承而来的(你可以在System.Web.Profile中找到)

在列表8中的Web.config包含一个从UserInfo类继承而来的profile,通过该声明,新的profile可以获得UserInfo类的所有属性。

列表 7. UserInfo (Visual Basic .NET)

Imports
Microsoft.VisualBasic
ImportsSystem.Web.Profile

PublicClassUserInfo
InheritsProfileBase

Private_FirstNameAsString
Private_LastNameAsString

PublicPropertyFirstName()AsString
Get
Return_FirstName
EndGet
Set(ByValvalueAsString)
_FirstName
=value
EndSet
EndProperty

PublicPropertyLastName()AsString
Get
Return_LastName
EndGet
Set(ByValvalueAsString)
_LastName
=value
EndSet
EndProperty

EndClass

列表 7. UserInfo (C#)
using System;
using System.Web.Profile;

public class UserInfo : ProfileBase
{
    private string _FirstName;
    private string _LastName;

    public string FirstName 
    {
        get { return _FirstName; }
        set { _FirstName = value; }
    }
    public string LastName
    {
        get { return _LastName; }
        set { _LastName = value; }
    }
}

using
System;
usingSystem.Web.Profile;

publicclassUserInfo:ProfileBase
{
privatestring_FirstName;
privatestring_LastName;

publicstringFirstName
{
get{return_FirstName;}
set{_FirstName=value;}
}
publicstringLastName
{
get{return_LastName;}
set{_LastName=value;}
}
}

列表 8. Web.Config

<
configuration>
<system.web>
<anonymousIdentificationenabled="true"/>
<profileinherits="UserInfo"/>
</system.web>
</configuration>


迁移匿名Profile设置
Profile对象既可用于匿名用户也可以用于已认证用户。然而,当用户从匿名用户状态转换为已认证用户状态时,Profile对象能够以一种令人难以理解的方式完成任务。
当匿名用户使用Profile对象时,用户profile是与一个随机生成的号码相关联的,该号码是根据每个用户唯一生成的,它保存在浏览器的cookie中,无论何时该用户返回应用程序,该用户的Profile设置会被自动加载。
如果匿名用户通过认证的话,所有与该用户相关的profile就会丢失,同时系统会生成一个新的profile。这时该Profile信息将与用户名相关联,而非唯一识别号。
要想理解所有这些工作,最好的方法就是看看下面的例子。列表9中的web.config定义了一个profile,该profile只有一个FavoriteColor属性。

列表 9 Web.config

<
configuration>
<system.web>

<authenticationmode="Forms"/>

<anonymousIdentificationenabled="true"/>

<profile>
<properties>
<add
name="FavoriteColor"
allowAnonymous
="true"
defaultValue
="Red"/>
</properties>
</profile>
</system.web>
</configuration>

列表10中有一个包含两个按钮的页面,分别是login和logout按钮,其中还有一个用于更新FavoriteColor属性的表单。
列表10. Anonymous.aspx (Visual Basic .NET)

<
%@PageLanguage="VB"%>

<scriptrunat="server">

SubLogin(ByValsAsObject,ByValeAsEventArgs)
FormsAuthentication.SetAuthCookie(
"Bill",False)
Response.Redirect(Request.Path)
EndSub

SubLogout(ByValsAsObject,ByValeAsEventArgs)
FormsAuthentication.SignOut()
Response.Redirect(Request.Path)
EndSub

SubUpdateProfile(ByValsAsObject,ByValeAsEventArgs)
Profile.FavoriteColor
=txtFavoriteColor.Text
EndSub

SubPage_PreRender()
lblUsername.Text
=Profile.UserName
lblFavoriteColor.Text
=Profile.FavoriteColor
EndSub

</script>

<html>
<head>
<title>Anonymous</title>
</head>
<body>
<formid="form1"runat="server">

<asp:ButtonID="Button1"
Text
="Login"
OnClick
="Login"
Runat
="Server"/>
<asp:ButtonID="Button2"
Text
="Logout"
OnClick
="Logout"
Runat
="Server"/>
<hr/>
<asp:TextBox
id
="txtFavoriteColor"
Runat
="Server"/>
<asp:ButtonID="Button3"
Text
="UpdateProfile"
OnClick
="UpdateProfile"
Runat
="Server"/>
<hr/>
<b>Username:</b>
<asp:Label
id
="lblUsername"
Runat
="Server"/>
<br/>
<b>FavoriteColor:</b>
<asp:Label
id
="lblFavoriteColor"
Runat
="Server"/>

</form>
</body>
</html>

列表10. Anonymous.aspx (C#)

<%
@PageLanguage="C#"%>

<scriptrunat="server">

voidLogin(Objects,EventArgse)
{
FormsAuthentication.SetAuthCookie(
"Bill",false);
Response.Redirect(Request.Path);
}

voidLogout(Objects,EventArgse)
{
FormsAuthentication.SignOut();
Response.Redirect(Request.Path);
}

voidUpdateProfile(Objects,EventArgse)
{
Profile.FavoriteColor
=txtFavoriteColor.Text;
}

voidPage_PreRender()
{
lblUsername.Text
=Profile.UserName;
lblFavoriteColor.Text
=Profile.FavoriteColor;
}

</script>

<html>
<head>
<title>Anonymous</title>
</head>
<body>
<formid="form1"runat="server">

<asp:Button
Text
="Login"
OnClick
="Login"
Runat
="Server"/>
<asp:ButtonID="Button1"
Text
="Logout"
OnClick
="Logout"
Runat
="Server"/>
<hr/>
<asp:TextBox
id
="txtFavoriteColor"
Runat
="Server"/>
<asp:Button
Text
="UpdateProfile"
OnClick
="UpdateProfile"
Runat
="Server"/>
<hr/>
<b>Username:</b>
<asp:Label
id
="lblUsername"
Runat
="Server"/>
<br/>
<b>FavoriteColor:</b>
<asp:Label
id
="lblFavoriteColor"
Runat
="Server"/>

</form>
</body>
</html>

当你打开第一个页面时,UserName的值是一个随机生成的唯一识别号(见图3)。当你按下Login按钮后,你就完成了身份认证,它是通过用户票据(User Bill)完成的。


图3 使用匿名和认证profile

列表10的页面中包含一个用于更新FavoriteColor的表单,要注意的是,在你登录登出的时候,会分别生成两个不同的profile。例如当你先登录,后登出的话,那么系统会生成一个随机的唯一识别号。

在很多情况下,你需要把匿名profile迁移到认证profile状态,如果你需要迁移profile属性值的话,你可以利用 ProfileModule类的MigrateAnonymous事件完成该任务,该事件只能在Global.asax文件中进行处理。列表11中的 Global.asax演示了你如何才能实现FavoriteColor属性的迁移。

列表11.Global.asax(VisualBasic.NET)

<%@ApplicationLanguage="VB"%>
<scriptrunat="server">

SubProfile_MigrateAnonymous(ByValsAsObject,_
ByValeAsProfileMigrateEventArgs)
DimanonProfileAsProfileCommon=_
Profile.GetProfile(e.AnonymousId)
Profile.FavoriteColor
=anonProfile.FavoriteColor
EndSub
</script>

列表11.Global.asax(C#)

<%@ApplicationLanguage="C#"%>
<scriptrunat="server">

voidProfile_MigrateAnonymous(Objects,
ProfileMigrateEventArgse)
{
ProfileCommonanonProfile
=
Profile.GetProfile(e.AnonymousId);
Profile.FavoriteColor
=anonProfile.FavoriteColor;
}
</script>

通过Profile类的GetProfile()方法你可以获得匿名profile,该方法接收一个唯一识别号,并且返回与唯一识别号对应的profile。ProfileMigrateEventArgs对象包含一个匿名识别号。

配置Profile Provider
默认情况下,profile被保存在sqlserver 2005 express数据库,它位于App_Data目录中,这或许在你开发一些普通的asp.net应用程序时是没有问题的,但很有可能,你需要把你的应用程序的profile保存在另一个数据库中,比如一个完整版的SqlServer 2005的实例中,而该数据库又位于你局域网的某个位置。

Profile使用Provider模式,通过修改web.config或machine.config的设置来告诉系统把信息存储在哪里。
ASP.NET本身配了一个profile provider,叫SqlProfileProvider。如果你感到困惑,你可以通过继承ProfileProvider基类来创建一个自己的 provider。例如,你可以创建一个基于Oracle数据库或MySql数据库的Provider。在这里,我们将只讨论最简单的方法,即通过SqlServer数据库来保存profile信息。

要使用Microsoft SQL Server存储profile信息,必须完成两个步骤。首先,你必须安装SQL Server数据库,然后你必须重新设置配置文件。

ASP.NET 2.0框架提供了一个用于配置SQL Server来存储Profile信息的工具,该工具叫做aspnet_regsql,它位于Windows\Microsoft.NET\ Framework\[.NET版本号]。执行该工具后,你会看到图4中的ASP.NET SQL Server安装向导。

图4 使用ASP.NET SQL Server安装程序

SQL Server安装向导会指导你完成必要的步骤,完成这些步骤后,向导会自动创建用于存储profile信息的存储过程和表结构。

在你完成SQL Server数据库的配置后,你需要修改web.config或machine.config中的数据库连接设置来指向服务器上的SQL Server数据库,本例中该数据库的实例名为MyServer,列表12列出了该配置文件。

列表 12.Web.Config

<configuration>
<connectionStrings>
<add
name="myConnectionString"
connectionString
=
"Server=MyServer;Trusted_Connection=true;database=MyDatabase"
/>
</connectionStrings>
<system.web>
<anonymousIdentificationenabled="true"/>
<profiledefaultProvider="MyProfileProvider">
<providers>
<add
name="MyProfileProvider"
type
="System.Web.Profile.SqlProfileProvider"
connectionStringName
="myConnectionString"/>
</providers>
<properties>
<add
name="FirstName"
allowAnonymous
="true"/>
<add
name="LastName"
allowAnonymous
="true"/>
</properties>
</profile>
</system.web>
</configuration>

在列表12中的profile配置中,包含了一个defaultProvider特性,这个特性指向一个叫MyProfileProvider的 profile provider,而这个provider定义是在profile标记的<providers>节中完成的。 MyProfileProvider则使用一个叫MyConnectionString的连接字符串完成数据库连接,并保存profile信息到数据库中。MyConnectionString可以在位于web.config开头的<connectionStrings>节中找到。

管理profile并生成profile报告
Profile会对象自动保存用户profile信息,这既是好事业是坏事。说它是好事,是因为你不需要写存储信息的所有逻辑代码,说它是坏事,是因为这样可能造成一大堆无用的信息被保存在数据库中。

幸运的是,ASP.NET 2.0框架包含一个叫做ProfileManager的类,你可以使用它来管理profile信息。它包含了相当多的方法使你能够有效地管理profile并且生成profile报表,下面列出了一些该类的重要方法:
  • DeleteInactiveProfiles. 删除一个特定日期之前的所有profile
  • DeleteProfile. 删除特定用户的profile
  • DeleteProfiles. 删除一个profile集合
  • FindInactiveProfilesByUserName. 返回一个ProfileInfo对象的集合,该集合表示的profile是匹配一个某个名字,并且是从某个特定日期开始一直未被使用
  • FindProfilesByUserName. 返回一个ProfileInfo对象集合,该集合与某个用户名相关联
  • GetAllInactiveProfiles. 返回一个ProfileInfo对象集合,该集合表示的profile是从某个特定日期开始一直未被使用的profile
  • GetAllProfiles. 返回一个ProfileInfo对象集合,该集合表示所有的profile
  • GetNumberOfInactiveProfiles. 返回从某个特定日期开始一直未被使用的profile的数量
  • GetNumberOfProfiles. 返回profile总数
这些方法中,虽然所有的方法都返回一个ProfileInfo对象集合,但没有一个返回一个真正的profile。ProfileInfo对象包含以下profile属性
  • IsAnonymous. 表示该profile是否为匿名profile
  • LastActivityDate. 最后一次profile被访问的时间和日期
  • LastUpdatedDate. 最后一次profile被升级的时间和日期
  • Size. 表示profile的大小,这是在profile provider存储profile信息时记录的
  • UserName. 与profile关联的用户名
ProfileManager有几个方法提供了额外的参数用于支持分页。例如,GetAllProfiles方法的一个重载版本就提供了专门用于设置页面索引、页面大小、总共的记录数的参数,这些参数在需要分页的页面中十分有用。

ProfileManager既可以在asp.net页面下使用,也可以在其它程序中使用。例如,你可能需要做一个控制台程序用于每天清除长时间未使用的 profile。列表14的控制台程序会删除七天未使用的profile,你可以使用Windows计划任务(Windows Scheduled Tasks)来安排该程序的执行时间。

列表14.DeleteInactiveProfiles(VisualBasic.NET)

ImportsSystem.Web.Profile

PublicClassDeleteInactiveProfiles

PublicSharedSubMain()
DimdeletedAsInteger
deleted
=
ProfileManager.DeleteInactiveProfiles(
ProfileAuthenticationOption.All,
DateTime.Now.AddDays(
-7))
Console.WriteLine(
"Deleted"&deleted&"profiles")
EndSub

EndClass

列表14.DeleteInactiveProfiles(C#)

usingSystem;
usingSystem.Web.Profile;

publicclassDeleteInactiveProfiles
{
publicstaticvoidMain()
{
intdeleted=0;
deleted
=
ProfileManager.DeleteInactiveProfiles(
ProfileAuthenticationOption.All,
DateTime.Now.AddDays(
-7));
Console.WriteLine(
"Deleted"+
deleted.ToString()
+"profiles");
}
}

你可以通过一下的命令行指令对列表14进行编译

[VisualBasic.NET]
C:\WINDOWS\Microsoft.NET\Framework\v2.0.40607\vbc
/r:System.Web.dllDeleteInactiveProfiles.vb

[C#]
C:\WINDOWS\Microsoft.NET\Framework\v2.0.40607\csc
DeleteInactiveProfiles.cs

你还可以使用ProfileManager类生成profile信息报表。例如,如果你打算生成一个用户调查的报表,你可以把用户调查保存在profile中,这样就可以轻易的使用ProfileManager生成你需要的报表。

列表15中的web.config中有三个属性:SurveyCompleted、FavoriteLanguageFavoriteEnvironment

Listing15.Web.Config

<configurationxmlns="http://schemas.microsoft.com/.NetConfiguration/v2.0">

<system.web>


<anonymousIdentificationenabled="true"/>

<profile>
<properties>
<add
name="SurveyCompleted"
type
="Boolean"
allowAnonymous
="true"/>
<add
name="FavoriteLanguage"
allowAnonymous
="true"/>
<add
name="FavoriteEnvironment"
allowAnonymous
="true"/>
</properties>
</profile>
</system.web>
</configuration>

列表16中的页面演示了一个简单的用户调查。该页面包含两个Panel控件,第一个控件中有两个调查问题,当用户完成调查后,第一个控件会自动隐藏,而第二个会显示出来,第二个Panel有一段表示感谢的文字。

列表16.Survey.aspx(VisualBasic.NET)

<%@PageLanguage="VB"%>
<scriptrunat="server">

SubSaveSurvey(ByValsAsObject,ByValeAsEventArgs)
Profile.FavoriteLanguage
=rdlLanguage.SelectedItem.Text
Profile.FavoriteEnvironment
=rdlEnvironment.SelectedItem.Text
Profile.SurveyCompleted
=True
EndSub

SubPage_PreRender()
IfProfile.SurveyCompletedThen
pnlSurvey.Visible
=False
pnlSurveyCompleted.Visible
=True
Else
pnlSurvey.Visible
=True
pnlSurveyCompleted.Visible
=False
EndIf
EndSub

</script>

<html>
<head>
<title>Survey</title>
</head>
<body>
<formid="form1"runat="server">

<asp:PanelID="pnlSurvey"Runat="Server">
Whatisyourfavoriteprogramminglanguage?
<br/>
<asp:RadioButtonList
id="rdlLanguage"
runat
="Server">
<asp:ListItemText="VB.NET"Selected="True"/>
<asp:ListItemText="C#"/>
<asp:ListItemText="J#"/>
</asp:RadioButtonList>
<p>&nbsp;</p>
Whatisyourfavoritedevelopmentenvironment?
<br/>
<asp:RadioButtonList
id="rdlEnvironment"
runat
="Server">
<asp:ListItemText="VS.NET"Selected="True"/>
<asp:ListItemText="WebMatrix"/>
<asp:ListItemText="Notepad"/>
</asp:RadioButtonList>
<p>&nbsp;</p>
<asp:Button
Text="SubmitSurvey"
Onclick
="SaveSurvey"
Runat
="Server"/>
</asp:Panel>
<asp:PanelID="pnlSurveyCompleted"Runat="Server">
Thankyouforcompletingthesurvey!
</asp:Panel>
</form>
</body>
</html>

列表 16. Survey.aspx(C#)

<%@PageLanguage="C#"%>
<scriptrunat="server">

voidSaveSurvey(Objects,EventArgse)
{
Profile.FavoriteLanguage
=rdlLanguage.SelectedItem.Text;
Profile.FavoriteEnvironment
=rdlEnvironment.SelectedItem.Text;
Profile.SurveyCompleted
=true;
}

voidPage_PreRender()
{
if(Profile.SurveyCompleted)
{
pnlSurvey.Visible
=false;
pnlSurveyCompleted.Visible
=true;
}
else
{
pnlSurvey.Visible
=true;
pnlSurveyCompleted.Visible
=false;
}
}

</script>

<html>
<head>
<title>Survey</title>
</head>
<body>
<formid="form1"runat="server">

<asp:PanelID="pnlSurvey"Runat="Server">
Whatisyourfavoriteprogramminglanguage?
<br/>
<asp:RadioButtonList
id="rdlLanguage"
runat
="Server">
<asp:ListItemText="VB.NET"Selected="True"/>
<asp:ListItemText="C#"/>
<asp:ListItemText="J#"/>
</asp:RadioButtonList>
<p>&nbsp;</p>
Whatisyourfavoritedevelopmentenvironment?
<br/>
<asp:RadioButtonList
id="rdlEnvironment"
runat
="Server">
<asp:ListItemText="VS.NET"Selected="True"/>
<asp:ListItemText="WebMatrix"/>
<asp:ListItemText="Notepad"/>
</asp:RadioButtonList>
<p>&nbsp;</p>
<asp:ButtonID="Button1"
Text
="SubmitSurvey"
Onclick
="SaveSurvey"
Runat
="Server"/>
</asp:Panel>
<asp:PanelID="pnlSurveyCompleted"Runat="Server">
Thankyouforcompletingthesurvey!
</asp:Panel>
</form>
</body>
</html>

列表17中显示调查的结果,该页面中有一个显示ProfileInfo对象集合的GridView控件,该ProfileInfo对象集合是由 ProfileManager的GetAllProfiles方法获得的。当你点击GridView中的任意一行的Select链接时,你将会看到对这个问题的调查结果,该调查结果是由Profile类的GetProfile方法获得的。


图5 显示调查结果

列表 17. SurveyResults.aspx(VisualBasic.NET)

<%@PageLanguage="VB"%>
<scriptrunat="server">

SubPage_Load()
ResultsGrid.DataSource
=_
ProfileManager.GetAllProfiles(ProfileAuthenticationOption.All)
ResultsGrid.DataBind()
EndSub

SubDisplayProfileDetails(ByValsAsObject,ByValeAsEventArgs)
DimSelectedProfileAsProfileCommon
SelectedProfile
=Profile.GetProfile(ResultsGrid.SelectedValue)
lblLanguage.Text
=SelectedProfile.FavoriteLanguage
lblEnvironment.Text
=SelectedProfile.FavoriteEnvironment
EndSub

</script>

<html>
<head>
<title>SurveyResults</title>
</head>
<body>
<formid="form1"runat="server">
<h2>SurveyResults</h2>
<asp:GridView
id="ResultsGrid"
DataKeyNames
="UserName"
AutoGenerateSelectButton
="true"
OnSelectedIndexChanged
="DisplayProfileDetails"
SelectedRowStyle-BackColor
="LightYellow"
Runat
="Server"/>
<p>&nbsp;</p>
<h2>SurveyDetails</h2>
<b>FavoriteLanguage:</b>
<asp:Label
id="lblLanguage"
Runat
="Server"/>
<br/>
<b>FavoriteEnvironment:</b>
<asp:Label
id="lblEnvironment"
Runat
="Server"/>

</form>
</body>
</html>

列表 17.SurveyResults.aspx(C#)

<%@PageLanguage="C#"%>
<scriptrunat="server">

voidPage_Load()
{
ResultsGrid.DataSource
=
ProfileManager.GetAllProfiles(ProfileAuthenticationOption.All);
ResultsGrid.DataBind();
}

voidDisplayProfileDetails(Objects,EventArgse)
{
ProfileCommonSelectedProfile
=
Profile.GetProfile(ResultsGrid.SelectedValue.ToString());
lblLanguage.Text
=SelectedProfile.FavoriteLanguage;
lblEnvironment.Text
=SelectedProfile.FavoriteEnvironment;
}

</script>

<html>
<head>
<title>SurveyResults</title>
</head>
<body>
<formid="form1"runat="server">
<h2>SurveyResults</h2>
<asp:GridView
id="ResultsGrid"
DataKeyNames
="UserName"
AutoGenerateSelectButton
="true"
OnSelectedIndexChanged
="DisplayProfileDetails"
SelectedRowStyle-BackColor
="LightYellow"
Runat
="Server"/>
<p>&nbsp;</p>
<h2>SurveyDetails</h2>
<b>FavoriteLanguage:</b>
<asp:Label
id="lblLanguage"
Runat
="Server"/>
<br/>
<b>FavoriteEnvironment:</b>
<asp:Label
id="lblEnvironment"
Runat
="Server"/>
</form>
</body>
</html>


总结
当建立Web应用程序时,我依旧花费了大量的时间和精力用于做一些大伤脑筋的事情。其中的一个任务就是写一些用于从数据库存储和获得用户信息的代码。虽然 Profile对象引入的都是asp.net 1.0中可以实现的功能,但是这个新特性帮助我们从乏味的编码工作中解脱出来,这样也能让我们在写Web应用程序的过程中,把更多的精力放在我们更感兴趣的事情上。
分享到:
评论

相关推荐

    ASP.NET 2.0入门经典:C#编程篇.pdf

    本书《ASP.NET 2.0入门经典:C#编程篇》显然是针对初学者设计的,旨在帮助读者掌握使用C#语言进行ASP.NET 2.0开发的基础知识。 1. **.NET Framework**:.NET Framework是ASP.NET的基础,它提供了一组类库、运行时...

    使用ASP.NET 2.0 Profile存储用户信息

    许多ASP.NET应用程序需要跨访问的用户...但如今,使用 ASP.NET 2.0的Profile对象,这个过程变得异常简单。Stephen Walther将验证该对象,并向你展示如何使用Profile来跟踪用户属性、创建一个购物篮,及其他一些例子。

    Programming Microsoft ASP.NET 2.0 Applications - Advanced Topics

    《Programming Microsoft ASP.NET 2.0 Applications - Advanced Topics》是一本深入探讨ASP.NET 2.0高级应用开发的专业书籍。本书旨在帮助开发者充分利用ASP.NET 2.0的强大功能,实现高效、可扩展和高度定制化的Web...

    ASP.net 2.0网络编程从入门到精通-修订.rar

    - ASP.NET使用页面生命周期(Page Life Cycle)处理用户请求,包括初始化、加载、验证、呈现和卸载等阶段。 - 控件生命周期(Control Life Cycle)与页面生命周期紧密关联,允许开发者在特定时刻对控件进行操作。 3. ...

    ASP.NET 2.0程序设计案例教程(课件)

    ASP.NET 2.0是微软开发的一个用于构建Web应用程序的框架,它是在.NET Framework 2.0版本上运行的。这个“ASP.NET 2.0程序设计案例教程”旨在为学习者提供深入理解和实践ASP.NET技术的机会。通过课件的形式,我们可以...

    asp.net 2.0动态网站开发教程

    虽然ASP.NET 2.0本身并不直接支持AJAX,但可以通过使用UpdatePanel控件和ScriptManager组件实现部分页面更新,这在一定程度上提供了类似AJAX的功能。 九、Web服务和WCF ASP.NET 2.0还提供了创建和消费Web服务的能力...

    天轰穿ASP.NET2.0视频教程代码(共两部份)

    此"天轰穿ASP.NET2.0视频教程"针对初学者和进阶开发者,通过详细的教学指导,帮助用户掌握ASP.NET 2.0的核心概念和技术。 1. **ASP.NET 2.0架构** ASP.NET 2.0基于.NET Framework 2.0,引入了统一的页面生命周期...

    ASP.NET 2.0入门经典(第4版)

    《ASP.NET 2.0 入门经典》将逐步引导您使用 ASP.NET 2.0 创建动态的、数据驱动的、复杂的Web站点。在本章结束时,我们将解释一些基本的想法并介绍一个完整的示例站点。然后将学习怎样使用 Visual Web Developer ...

    ASP.NET2.0 经典案例教程

    熟悉ASP.NET 1.x的程序员亟需快速掌握ASP.NET 2.0新增的功能和控件,以便立刻使用ASP.NET 2.0来编写程序。  本书通过深入剖析12个使用ASP.NET 2.0开发的项目,全面阐述了ASP.NET 2.0应用程序的架构及ASP.NET 2.0...

    Wrox Asp.Net 2.0 Instant Results(asp.net2.0经典案例教程)

    案例会展示如何设置用户认证和授权策略,以及如何使用SQL Server存储用户信息。 7. **配置和部署**:了解如何正确配置应用程序设置和进行部署是开发过程中的重要环节。书中会涵盖Web.config文件的使用,以及发布和...

    ASP.NET 2.0高级编程(特别版)》[英文CHM电子书+源代码].

    在全面介绍ASP.NET各种编程技能的同时重点介绍了ASP.NET 2.0版本中的巨大变化,详细阐述了ASP.NET 2.0中所包含的每个新特性。书中提供了大量的实例,可帮助读者快速掌握如何在.NET Framework下构建功能强大的ASP.NET...

    ASP.NET2.0+sql server 网络应用开发详解

    《ASP.NET 2.0 + SQL Server 网络应用开发详解》这本书是针对使用ASP.NET 2.0和SQL Server进行Web应用开发的详细指南。ASP.NET是Microsoft .NET框架的一部分,它提供了一种高效、面向对象的方式来构建动态网站、web...

    ASP.NET2.0实用教程(C#版)书中例子.rar

    这个“ASP.NET2.0实用教程(C#版)书中例子”PPT很可能会涵盖以上这些主题,并通过实例演示如何在实践中应用这些技术。PPT中的例子可能包括创建简单的Web表单、使用控件进行数据绑定、实现用户身份验证以及利用AJAX...

    Asp.net 2.0高级编程(pdf)

    内容代码,使用VB.net和C# 两种解释 &lt;br&gt;第1章ASP.NET2.0概述 1 1.1简史 1 1.2ASP.NET2.0的目标 2 1.2.1开发人员的效率 3 1.2.2管理 5 1.2.3性能和可伸缩性 8 1.3ASP.NET2.0的其他新特性 8 ...

    ASP.NET 2.0 入门经典(第4版).rar

    《ASP.NET 2.0 入门经典》将逐步引导您使用 ASP.NET 2.0 创建动态的、数据驱动的、复杂的Web站点。在本章结束时,我们将解释一些基本的想法并介绍一个完整的示例站点。然后将学习怎样使用 Visual Web Developer ...

    ASP.NET 2.0入门经典(第4版)源码

    ASP.NET 2.0是微软推出的Web应用程序开发框架,它基于.NET Framework,为开发者提供了构建动态网站、Web应用和Web服务的强大工具。本资源“ASP.NET 2.0入门经典(第4版)源码”提供了该书配套的源代码,帮助初学者深入...

    asp.net 2.0 宝典

    《ASP.NET 2.0 宝典》是一本深度探讨ASP.NET 2.0开发技术的专业书籍,其随书光盘包含了一系列的示例代码和教程,帮助读者深入理解和掌握这个强大的Web应用程序开发框架。虽然上传的文件只是部分内容,但它们涵盖了多...

    asp.net2.0 调用websevice 实现天气预报

    在本项目中,“asp.net2.0 调用websevice 实现天气预报”,我们将探讨如何利用ASP.NET 2.0集成Web服务(Web Service)来获取并展示实时天气信息,从而实现一个无刷新的天气预报应用。 首先,Web服务是一种基于XML的...

    ASP.NET AJAX程序设计——第I卷:服务器端ASP.NET 2.0 AJAX Extensions与ASP.NET AJAX Control Toolkit 源代码

    本卷从最易于理解和使用的那部分入手,介绍ASP.NET AJAX框架中能够与传统ASP.NET无缝对接的服务器端部分,包括服务器端ASP.NET AJAX Extensions与ASP.NET AJAX Control Toolkit。这部分内容不需要读者有任何的客户端...

Global site tag (gtag.js) - Google Analytics