開發(fā)BtoC電子商務(wù)系統(tǒng)(ASP.NET)
發(fā)表時(shí)間:2024-02-22 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]原文還有兩個(gè)圖在對(duì)ASP.NET Web表單的編程模型有了基本的認(rèn)識(shí)后,通過應(yīng)用于現(xiàn)實(shí)的開發(fā)案例來提高對(duì)ASP.NET Web表單內(nèi)在運(yùn)作機(jī)制的了解,以及由此帶來的對(duì)系統(tǒng)架構(gòu)的掌控是很有必要的。我們沒有為編程而編程的高貴姿態(tài),我們深深懂得能夠開發(fā)出高效,健壯,強(qiáng)大的應(yīng)用程序始終是編程的終極。我們下...
原文還有兩個(gè)圖
在對(duì)ASP.NET Web表單的編程模型有了基本的認(rèn)識(shí)后,通過應(yīng)用于現(xiàn)實(shí)的開發(fā)案例來提高對(duì)ASP.NET Web表單內(nèi)在運(yùn)作機(jī)制的了解,以及由此帶來的對(duì)系統(tǒng)架構(gòu)的掌控是很有必要的。我們沒有為編程而編程的高貴姿態(tài),我們深深懂得能夠開發(fā)出高效,健壯,強(qiáng)大的應(yīng)用程序始終是編程的終極。我們下面通過一個(gè)完整的BToC電子商務(wù)系統(tǒng)的開發(fā)流程來展示ASP.NET Web表單是怎樣具體搭建面向下一代網(wǎng)絡(luò)平臺(tái)的。
這是一個(gè)典型的基于B/S(瀏覽器/服務(wù)器) 三層架構(gòu)的食品,飲料電子商務(wù)零售系統(tǒng)——“玉米地零食店”。前端為產(chǎn)品瀏覽器,為消費(fèi)者提供瀏覽/選購(gòu)商品,下訂單購(gòu)物等各個(gè)環(huán)節(jié)的功能;中間層為銷售商的稅率,優(yōu)惠等商務(wù)邏輯;后端為與整個(gè)零售系統(tǒng)相關(guān)的產(chǎn)品,顧客,訂單等數(shù)據(jù)庫(kù)。我們采用ASP.NET+IIS 5來構(gòu)建前端和中間層,SQL Server 2000來管理后端數(shù)據(jù)庫(kù),整個(gè)系統(tǒng)運(yùn)行于Windows XP。相關(guān)硬件配置只要滿足上述軟件的基本配置,系統(tǒng)性能便可保證。下面為該網(wǎng)上零售系統(tǒng)的前端界面圖示:
在編制Web 表單商業(yè)前端和中間層之前,我們有必要對(duì)后端數(shù)據(jù)庫(kù)做一個(gè)簡(jiǎn)單的介紹。后端數(shù)據(jù)庫(kù) CornfieldGrocer 由4個(gè)表組成:產(chǎn)品類別表Categories ,產(chǎn)品細(xì)節(jié)表 Details ,產(chǎn)品表 Products ,客戶信息表Customers?紤]到演示系統(tǒng)的的簡(jiǎn)潔性,我們沒有添加相關(guān)的存儲(chǔ)過程,視圖,規(guī)則等,這些在實(shí)際的系統(tǒng)的開發(fā)中對(duì)提高系統(tǒng)的性能是很有必要,尤其是在大數(shù)據(jù)量的情況下。下面為4個(gè)表的字段的圖示介紹:
各個(gè)表的字段的表義已經(jīng)相當(dāng)清楚,我們不在這里贅述。我們下面向大家展示一下整個(gè)電子商務(wù)零售系統(tǒng)——“玉米地零食店”的物理文件組成及其結(jié)構(gòu),下圖為示意圖:
所有的文件位于ASP.NET站點(diǎn)目錄CornfieldGrocer下,其中還有Web表單頁(yè)面用到的圖片子目錄Images下的文件就不再在這里列出了。
下面我們開始編寫前端和中間層代碼,為了更清楚地展示W(wǎng)eb Form ASP.NET的底層代碼構(gòu)造,我們采用記事本來完成整個(gè)代碼的編寫過程。需要說明的是在真正的工程項(xiàng)目開發(fā)實(shí)踐中,如能借助Visual Studio.Net等可視集成開發(fā)工具,開發(fā)效率會(huì)大大提高。但在ASP.NET代碼的底層機(jī)制沒有諳熟的情況下,筆者強(qiáng)烈建議初期的開發(fā)學(xué)習(xí)不妨放在Windows系統(tǒng)自帶的“記事本”這一簡(jiǎn)單卻能夠把代碼暴露得相當(dāng)清晰的工具里。
由于篇幅有限,我們不可能將所有的代碼都在這里展示給大家。如前所述,web.config為每個(gè)站點(diǎn)級(jí)的基于XML的配置文件,負(fù)責(zé)一些ASP.NET的安全認(rèn)證,編碼選擇,診斷測(cè)試等ASP.NET的配置工作,為瀏覽器請(qǐng)求ASP.NET Web表單時(shí)通過 IIS處理后的第一站。下面為其內(nèi)容:
<configuration>
<system.web>
<globalization requestEncoding="UTF-8" responseEncoding="UTF-8" />
</system.web>
</configuration>
容易看到這里的配置內(nèi)容相當(dāng)簡(jiǎn)單,僅指定請(qǐng)求/發(fā)送的編碼為“UTF-8”。我們對(duì)此不再贅述。
global.asax文件及其由后端代碼文件global.asax.cs編譯成的Bin\CornfieldGrocer.dll共同組成該網(wǎng)上零售系統(tǒng)的ASP.NET應(yīng)用程序定義。我們先來看文件global.asax:
<%@ Application Inherits="CornfieldGrocer.Global" %>
該文件只有一行指示符,它表示ASP.NET應(yīng)用程序的定義繼承自Global類,而Global類正是在global.asax.cs文件中定義:
using System;
using System.Collections;
using System.ComponentModel;
using System.Web;
using System.Web.SessionState;
namespace CornfieldGrocer
{
public class Global : System.Web.HttpApplication
{
protected void Session_Start(Object sender, EventArgs e)
{
if (Session["ShoppingCart"] == null)
{
Session["ShoppingCart"] = new CornfieldGrocer.OrderList();
}
}
}
}
在Global類里,我們定義了區(qū)段(Session)意義下的購(gòu)物卡(ShoppingCart)——這里采用了C#中的索引器。購(gòu)物卡的類型為CornfieldGrocer命名空間中的OrderList類,在CornfieldGrocer.cs文件中有定義。我們當(dāng)然也可以在global.asax文件中用腳本語言的形式將上面兩個(gè)文件的內(nèi)容合并起來,但那不是ASP.NET推薦的做法,因?yàn)槟_本語言的第一次執(zhí)行還要進(jìn)行動(dòng)態(tài)編譯,這回?fù)p失一部分性能,而將CS文件提前編譯成dll文件則會(huì)降低這種代價(jià)——當(dāng)然這里的編譯的意思還是指將CS的源代碼文件編譯成微軟中間語言的過程。其次,頁(yè)面與后端代碼分離的原則易于項(xiàng)目管理,是Visual Studio.NET推薦的工程性的做法。
文件Default.aspx為整個(gè)網(wǎng)上零售系統(tǒng)的前端頁(yè)面HTML代碼,Default.aspx.cs為其后端控制Web表單行為的CS代碼。由于篇幅關(guān)系我們這里不再贅述其HTML代碼,實(shí)際上從前面給出的前端界面圖示,我們可以基本了解Default.aspx的HTML代碼結(jié)構(gòu)。Style.css文件為Default.aspx文件的頁(yè)面樣式定義文件,定義一些頁(yè)面元素的顏色,格式,間距等修飾性的東西,我們也不再多言。下面只向大家展示Default.aspx的頁(yè)面指示符:
<%@ AutoEventWireup="false" Inherits="CornfieldGrocer.MainForm" %>
我們用“Inherits="CornfieldGrocer.MainForm"”來表示我們的頁(yè)面繼承自MainForm類,這樣我們就實(shí)現(xiàn)了對(duì)ASP.NET Web 表單行為的控制代碼與頁(yè)面顯示的HTML的分離。其中“AutoEventWireup="false"”表示頁(yè)面事件非自動(dòng)使能——頁(yè)面事件非自動(dòng)使能的意思是所有頁(yè)面事件必須經(jīng)過用戶明確的操作才能觸發(fā),由于該屬性缺省為“true”表示自動(dòng)使能,但我們的商業(yè)邏輯要求非自動(dòng)使能,故這里的語句很有必要,否則會(huì)引起系統(tǒng)處理的混亂。下面我們來看MainForm類:
using System;
using System.Collections;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Web;
using System.Web.SessionState;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.HtmlControls;
namespace CornfieldGrocer
{
public class MainForm: System.Web.UI.Page
{
protected System.Web.UI.WebControls.Label CurrentCategory;
protected System.Web.UI.WebControls.Label Name;
protected System.Web.UI.WebControls.Label SubTotal;
protected System.Web.UI.WebControls.ImageButton Imagebutton1;
protected System.Web.UI.WebControls.Label Description;
protected System.Web.UI.WebControls.Label Company;
protected System.Web.UI.WebControls.Repeater DetailsListing;
protected System.Web.UI.WebControls.DataList ProductListing;
protected System.Web.UI.WebControls.DataList ShoppingCartList;
protected System.Web.UI.HtmlControls.HtmlSelect CategoryList;
protected System.Web.UI.WebControls.Button btnSelect;
protected System.Web.UI.WebControls.Label Tax;
protected System.Web.UI.WebControls.Label Total;
protected System.Web.UI.WebControls.ImageButton Imagebutton4;
protected System.Web.UI.WebControls.ImageButton Imagebutton5;
protected System.Web.UI.WebControls.ImageButton Imagebutton6;
protected System.Web.UI.HtmlControls.HtmlInputText Qty;
protected System.Web.UI.HtmlControls.HtmlGenericControl CheckoutPanel;
protected System.Web.UI.HtmlControls.HtmlImage SelectedProdPicture;
public MainForm()
{
Page.Init += new System.EventHandler(Page_Init);
}
private void Page_Load(object sender, System.EventArgs e)
{
if (!IsPostBack)
{
ProductListing.SelectedIndex = 0;
UpdateProducts();
UpdateShoppingCart();
}
}
private void Page_Init(object sender, EventArgs e)
{
InitializeComponent();
}
private void InitializeComponent()
{
this.btnSelect.Click +=
new System.EventHandler(this.CategoryList_Select);
this.ProductListing.SelectedIndexChanged+=
new System.EventHandler(this.ProductListing_Select);
this.Imagebutton1.Click+=
new ImageClickEventHandler(this.AddBtn_Click);
this.Imagebutton4.Click+=
new ImageClickEventHandler(this.Recalculate_Click);
this.Imagebutton6.Click+=
new ImageClickEventHandler(this.ClearCart_Click);
this.Load +=
new System.EventHandler(this.Page_Load);
}
private void CategoryList_Select(Object sender, EventArgs e)
{
CurrentCategory.Text =
CategoryList.Items[CategoryList.SelectedIndex].Text;
UpdateProducts();
}
private void ProductListing_Select(Object sender, EventArgs e)
{
UpdateProducts();
}
private void AddBtn_Click(Object sender, ImageClickEventArgs e)
{
int productID = Int32.Parse
(ProductListing.DataKeys[ProductListing.SelectedIndex].ToString());
InventoryDB market = new InventoryDB();
DataRow product = market.GetProduct(productID);
CornfieldGrocer.OrderList shoppingCart =
((CornfieldGrocer.OrderList) Session["ShoppingCart"]);
shoppingCart.Add(new CornfieldGrocer.OrderItem(productID,
(String) product["ProductName"],
Double.Parse(product["UnitPrice"].ToString()), 1));
UpdateShoppingCart();
}
private void Recalculate_Click(Object sender, ImageClickEventArgs e)
{
CornfieldGrocer.OrderList shoppingCart =
((CornfieldGrocer.OrderList) Session["ShoppingCart"]);
for (int i=0; i<ShoppingCartList.Items.Count; i++) >
{
HtmlInputText qty =
(HtmlInputText) ShoppingCartList.Items[i].FindControl("Qty");
try
{
shoppingCart[(String) ShoppingCartList.DataKeys][i]].Quantity
= Int32.Parse(qty.Value);
}
catch (Exception)
{
}
}
UpdateShoppingCart();
}
private void ClearCart_Click(Object sender, ImageClickEventArgs e)
{
CornfieldGrocer.OrderList shoppingCart =
((CornfieldGrocer.OrderList) Session["ShoppingCart"]);
shoppingCart.ClearCart();
UpdateShoppingCart();
}
void UpdateProducts()
{
InventoryDB market = new InventoryDB();
int categoryID = Int32.Parse
(CategoryList.Items[CategoryList.SelectedIndex].Value);
ProductListing.DataSource =
market.GetProducts(categoryID).DefaultView;
ProductListing.DataBind();
int productID = Int32.Parse
(ProductListing.DataKeys[ProductListing.SelectedIndex].ToString());
DataRow product = market.GetProduct(productID);
Name.Text = product["ProductName"].ToString();
SelectedProdPicture.Src = product["ImagePath"].ToString();
Description.Text = product["ProductDescription"].ToString();
Company.Text = product["Manufacturer"].ToString();
DetailsListing.DataSource =
market.GetProductCalories(productID).DefaultView;
DetailsListing.DataBind();
}
void UpdateShoppingCart()
{
CornfieldGrocer.OrderList shoppingCart =
((CornfieldGrocer.OrderList) Session["ShoppingCart"]);
SubTotal.Text = String.Format("{0:C}", shoppingCart.SubTotal);
Tax.Text = String.Format("{0:C}", shoppingCart.Tax);
Total.Text = String.Format("{0:C}", shoppingCart.Total);
ShoppingCartList.DataSource=shoppingCart.Values;
ShoppingCartList.DataBind();
}
}
}
MainForm類中共有11個(gè)方法,19個(gè)保護(hù)域。其中的19個(gè)保護(hù)域和前面給出的前端界面圖示的頁(yè)面元素相對(duì)應(yīng),這里不再贅述。11個(gè)方法中MainForm()為構(gòu)建器,其添加了頁(yè)面初始化事件Page_Init(),這是ASP.NET Web表單最先處理的事件,一般進(jìn)行一些基礎(chǔ)的初始化操作。我們可以看到在Page_Init()中進(jìn)行了初始化組件InitializeComponent()的操作。Page_Load()事件出現(xiàn)在用戶發(fā)出請(qǐng)求后,頁(yè)面裝載的時(shí)候,在這里一般可做一些商業(yè)邏輯初始化方面的操作,比如數(shù)據(jù)庫(kù)的連接,購(gòu)物卡的初始化等。我們這里進(jìn)行了產(chǎn)品展示UpdateProducts()和購(gòu)物卡的初始化UpdateShoppingCart()的操作。
其他四個(gè)方法分別為產(chǎn)品類別的選擇ProductListing_Select(),購(gòu)買產(chǎn)品的添加AddBtn_Click(),購(gòu)物卡的重新計(jì)算Recalculate_Click(),購(gòu)物卡的清除ClearCart_Click()都是通過對(duì)ASP.NET控件的操作來觸發(fā)相應(yīng)的事件完成商業(yè)邏輯。上面的代碼已經(jīng)展示的相當(dāng)清楚,我們不再贅述。
最后我們要向大家說明的是中間層商務(wù)邏輯的組件。它由三個(gè)類構(gòu)成:庫(kù)存數(shù)據(jù)類InventoryDB,訂單項(xiàng)目類OrderItem和訂單列表類OrderList。它們共同在文件CornfieldGrocer.cs文件中定義。自解釋的編程方式已經(jīng)它們的結(jié)構(gòu)展示的相當(dāng)清除,我們下面只給出該文件的CS源代碼:
using System;
using System.Data;
using System.Data.SqlClient;
using System.Collections;
namespace CornfieldGrocer
{
public class InventoryDB
{
public DataTable GetProducts(int categoryID)
{
SqlConnection sqlConnect= new SqlConnection
("server=(local);database=CornfieldGrocer;Trusted_Connection=yes");
SqlDataAdapter sqlAdapter1 = new SqlDataAdapter
("Select * from Products where categoryid="+categoryID,sqlConnect);
DataSet products = new DataSet();
sqlAdapter1.Fill(products, "products");
return products.Tables[0];
}
public DataRow GetProduct(int productID)
{
SqlConnection sqlConnect= new SqlConnection
("server=(local);database=CornfieldGrocer;Trusted_Connection=yes");
SqlDataAdapter sqlAdapter1 = new SqlDataAdapter
("Select * from Products where productID=" + productID, sqlConnect);
DataSet product = new DataSet();
sqlAdapter1.Fill(product, "product");
return product.Tables[0].Rows[0];
}
public DataTable GetProductCalories(int productID)
{
SqlConnection sqlConnect = new SqlConnection
("server=(local);database=CornfieldGrocer;Trusted_Connection=yes");
SqlDataAdapter sqlAdapter1 = new SqlDataAdapter
("Select * from Details where productID="+productID,sqlConnect);
DataSet details = new DataSet();
sqlAdapter1.Fill(details, "details");
return details.Tables[0];
}
}
public class OrderItem
{
public int productID;
public int quantity;
public String name;
public double price;
public OrderItem(int productID, String name, double price, int quantity)
{
this.productID = productID;
this.quantity = quantity;
this.name = name;
this.price = price;
}
public int ProductID
{
get { return ProductID; }
}
public int Quantity
{
get { return quantity; }
set { quantity=value; }
}
public String Name
{
get { return name; }
}
public double Price
{
get { return price; }
}
public double Total
{
get { return quantity * price; }
}
}
public class OrderList
{
private Hashtable orders = new Hashtable();
private double taxRate = 0.08;
public double SubTotal
{
get
{
if (orders.Count == 0)
return 0.0;
double subTotal = 0;
IEnumerator items = orders.Values.GetEnumerator();
while(items.MoveNext())
{
subTotal += ((OrderItem) items.Current).Price *
((OrderItem) items.Current).Quantity;
}
return subTotal;
}
}
public double TaxRate
{
get { return taxRate; }
set { taxRate = value; }
}
public double Tax
{
get { return SubTotal * taxRate; }
}
public double Total
{
get { return SubTotal * (1 + taxRate); }
}
public ICollection Values {
get {
return orders.Values;
}
}
public OrderItem this[String name] {
get {
return (OrderItem) orders[name];
}
}
public void Add(OrderItem value)
{
if (orders[value.Name] == null) {
orders.Add(value.Name, value);
}
else
{
OrderItem oI = (OrderItem)orders[value.Name];
oI.Quantity = oI.Quantity + 1;
}
}
public void ClearCart() {
orders.Clear();
}
}
}
需要說明的是我們將三個(gè)文件CornfieldGrocer.cs,Default.aspx.cs,Global.asax.cs用編譯命令“csc /t:library /out:CornfieldGrocer.dll cornfieldgrocer.cs default.aspx.cs global.asax.cs”將它們?nèi)糠庋b在CornfieldGrocer命名空間里,雖然這并不是必須的。上面的編譯器輸出CornfieldGrocer.dll文件,我們配置該網(wǎng)上零售站點(diǎn)時(shí)只需將該文件拷貝到站點(diǎn)根目錄中的Bin目錄下即可。
到此為止,我們已經(jīng)完整的向大家展示了利用ASP.NET Web表單建立一個(gè)小型的網(wǎng)上交易系統(tǒng)的編碼,配置等工作。當(dāng)然作為演示案例,它還沒有真正系統(tǒng)的完善的性能,安全,界面等各個(gè)方面的優(yōu)化考慮和設(shè)計(jì)。但它向我們展示的ASP.NET Web表單模型卻非常典型且底層,大家不防在此基礎(chǔ)上通過不斷的修改和擴(kuò)充來開發(fā)適合自己的交易系統(tǒng)。比如對(duì)于Default.aspx文件中AutoEventWireup="false"如果設(shè)置為“true”或去掉這個(gè)語句,在運(yùn)行頁(yè)面時(shí)會(huì)出現(xiàn)什么情況?通過這些練習(xí)便會(huì)不斷的加深我們對(duì)ASP.NET底層的認(rèn)識(shí),最后達(dá)到游刃有余的把握。實(shí)際上技術(shù)的學(xué)習(xí),尤其是編程,除了一定的興趣和悟性外,大量代碼實(shí)例的鍛煉也是很有必要的,這本身就是筆者成長(zhǎng)的一個(gè)過程,也是本文中筆者竭力要給大家展示的。