分享
 
 
 

提高DB2 Web应用程序性能的五条规则

王朝other·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

优秀的代码行行相似,糟糕的代码各个不同。

规则其实并不重要,重要的是养成良好习惯。

——译者题记

提高DB2 Web应用程序性能的5条规则

作者:Glenn Stephens

2003年8月

内容

简介

规则1:按需检索

规则2:优化数据库

规则3:使用DB2 UDB的OLAP函数改善分页

规则4:使用存储过程

规则5:尽可能使用缓存

总结

简介

速度和可升级性是网络开发重要的性能要求,而且它们也不难达到。应用一些简单的规则会提高网络应用的性能。在本文中,您将了解到使用Borland® C#Builder和IBM® DB2®通用数据库(UDB)8.1开发快速Microsoft® ASP.NET Web应用程序的5条规则。

分析Web应用程序的性能需要利用一些手段来检测每个操作的性能。为此,我创建了一个TimeDiff类(见程序清单1),它可以计算数据库操作的用时。你可以利用TimeDiff类的检测结果作为衡量数据库操作性能的基准,来观察哪些操作最为有效。我还创建了一个配合TimeDiff 类使用的LOTSOFRECORDS表(见程序清单2),它包含10,000条记录,你可以通过对它的操作来观察不同技术之间的性能差异。DB2具有内部缓冲池,一旦运行一个查询,内部缓冲池启用,因此二次查询的速度较快。在检测查询速度时,请忽略缓冲池启用前的结果而采用启用后的结果。

程序清单1. TimeDiff类

using System;

namespace Effeciency

{

///

/// 这个类用于时间计算。在例子中,

/// 我们将利用它检测数据库操作的速度,

/// 以便作性能上的比较。

///

public class TimeDiff

{

DateTime StartTime;

DateTime EndTime;

public TimeDiff() {}

public void Start()

{

StartTime = DateTime.Now;

}

public void Stop()

{

EndTime = DateTime.Now;

}

public string TimeDifferenceText

{

get

{

TimeSpan TimeDifference = EndTime - StartTime;

return TimeDifference.ToString();

}

}

}

}

程序清单2. LOTSOFRECORDS的表定义

CREATE TABLE "GLENN "."LOTSOFRECORDS" (

"KEYCOL" INTEGER NOT NULL ,

"COL1" CHAR(50) ,

"COL2" CHAR(50) ,

"COL3" CHAR(50) ,

"COL4" CHAR(50) ,

"COL5" CHAR(50) ,

"COL6" CHAR(50) ,

"COL7" CHAR(50) ,

"COL8" CHAR(50) ,

"COL9" CHAR(50) ,

"COL10" CHAR(50) )

IN "USERSPACE1" ;

COMMENT ON TABLE "GLENN "."LOTSOFRECORDS" IS 'Table designed to Contain Lots of Records';

-- DDL Statements for primary key on Table "GLENN "."LOTSOFRECORDS"

ALTER TABLE "GLENN "."LOTSOFRECORDS"

ADD CONSTRAINT "CC1058255334652" PRIMARY KEY

("KEYCOL");

下面,我们就开始介绍这些提高DB2 UDB Web 应用程序性能的规则。我会先介绍提高数据库性能的基本规则,然后列出一些适合以Borland Data Provider开发ASP.NET应用时的性能规则。

规则1:按需检索

如果你只能记得住一条规则,那一定要记住这一条:按需检索。如果你是“幸存者”电视节目的忠实观众,您会记得其中的参赛者通过限额分配来保证每个人都有充足的食物。这种做法对于数据库开发同样适用。如果你的应用程序能够按需运行,那么就会合理地将数据库和网络资源留给其他应用。这听起来很简单,但我们还是来看一个例子。

假设有一个包含10,000 行记录和10 个字段的表,以及一个需要显示所有记录但只显示3个字段的Web页。很多开发者经常图省事而使用"select *"语句选择所有的字段:

select * from GLENN.LOTSOFRECORDS

这种做法应当避免,而应力求只检索需要的字段。可以在SQL 语句中定义要检索的字段,例如:

select

KEYCOL, COL1, COL2, COL7

from

GLENN.LOTSOFRECORDS

在本文附带的源程序中,有两个ASP.NET页面:一个是RetrievingAllFields.aspx,它执行第一个查询;另一个是RetrievingLimitedFields.aspx ,它执行第二个查询,即只检索需要的字段。

用TimeDiff类进行检测,执行第一个查询用了1.622 秒,执行第二个查询用了1.311秒。后者用时只是前者的80%,不仅用时少,而且还减少了Web应用程序和数据库服务器之间的网络数据堵塞。

这个例子只限制了检索的字段,你还可以使用WHERE语句限制检索的记录数。WHERE 语句可以限制服务器返回的记录数(见程序清单3)。要记住,通过网络发送的记录数据越少,对应用程序、数据库、用户和网络越有好处。

规则2:优化数据库

有时候你的Web应用程序可能运行得不错,但你想让它更好。一个简单的减少搜索时间的方法是为特定字段创建索引。如果有一个查询是要搜索某个价格范围内的产品(见程序清单3),但你没有为价格字段定义索引,那么返回数据就会多花一些时间。而一旦建立了索引,DB2会很快返回你想要的结果。

程序清单3. 利用索引进行数据库搜索

SELECT

PRODUCTCODE, PRODUCTNAME, DESCRIPTION, UNITPRICE

FROM

GLENN.PRODUCTLIST

WHERE

UNITPRICE > 20.00

优化数据库不只是为搜索字段创建索引这么一条,你应当尽可能多地搜集相关的DB2信息以使应用程序运行得更好。经常访问IBM发者园地(IBM DB2 Developer Domain)和comp.databases.ibm-db2(comp.databases.ibm-db2)等一些相关的Web站点或新闻组,对于保持DB2开发技巧不断更新是一个很好的办法。

你还应当努力熟悉DB2附带的工具,例如DB2索引建议器(Index Advisor)。DB2索引建议器可以根据你递交的查询和所连接的数据库返回最佳索引列表。

规则3:使用DB2 UDB的OLAP功能改善分页

在ASP.NET中,一个最常见的操作是表格分页显示。ASP.NET中DataGrid组件的默认设置是将表格需要的所有记录都返回客户端,然后再根据选择的页显示相应记录(见图1 和程序清单4)。

图1 DataGrid的分页设置

程序清单4 使用DataGrid 内建分页机制

TimeDiff diff = new TimeDiff();

private DataSet GetProductsDataSet() {

diff.Start();

string connString = ConfigurationSettings.AppSettings["database"];

BdpConnection conn = new BdpConnection(connString);

BdpDataAdapter da = new BdpDataAdapter("select KEYCOL, " +

"COL1, COL2, COL7 FROM GLENN.LOTSOFRECORDS "+

"ORDER BY KEYCOL ASC", conn);

DataSet ds = new DataSet();

da.Fill(ds, "Table1");

diff.Stop();

return ds;

}

private void BindToTheData()

{

dataGrid1.DataSource = GetProductsDataSet();

dataGrid1.DataMember = "Table1";

dataGrid1.DataBind();

label1.Text = diff.TimeDifferenceText;

}

private void Page_Load(object sender, System.EventArgs e)

{

if (!IsPostBack) {

BindToTheData();

}

}

private void dataGrid1_PageIndexChanged(object source,

System.Web.UI.WebControls.DataGridPageChangedEventArgs e)

{

//Change the active page of the data

dataGrid1.CurrentPageIndex = e.NewPageIndex;

BindToTheData();

}

如果以LOTSOFRECORDS表为例,从数据库中检索1万条记录很快就会占尽网络带宽,尤其是一次只需要查看10条或20条记录时。用户会发现加载ASP.NET页面的时间太长,而且很可能会超时。试想,如果数据库有数百万条记录,那你的应用程序就会运行缓慢只至停止执行。因此,需要采取更好的方法来检索数据。

还好,DataGrid组件允许定制分页。你可以计算哪些记录需要显示,然后只从服务器检索出相应的记录。

后来版本的DB2 UDB都具有一个很大的特点,那就是OLAP函数,它允许你进行各种记录检索。例如,只检索某些记录可以执行如下的查询:

SELECT * FROM

(SELECT KEYCOL, COL1, COL2, COL7, rownumber()

over(ORDER BY KEYCOL ASC) AS rn

FROM GLENN.LOTSOFRECORDS

ORDER BY KEYCOL ASC) AS a1

WHERE a1.rn BETWEEN 100 AND 120

使用DataGrid定制分页时,需要获得特定页的DataSet。GetDataByPage方法可以检索结果集而不管分页数和分页大小:

private DataSet GetDataByPage(int PageNo, int PageSize, out int NumberOfPages)

{

int startRecord = (PageNo - 1) * PageSize + 1;

int endRecord = startRecord + PageSize - 1;

string connString = ConfigurationSettings.AppSettings["database"];

BdpConnection conn = new BdpConnection(connString);

conn.Open();

//Get the number of Pages

string sRecordCount = "select count(*) from GLENN.LOTSOFRECORDS";

BdpCommand cmdGetRecordCount = new BdpCommand(sRecordCount, conn);

int intRecordCount = (int)cmdGetRecordCount.ExecuteScalar();

NumberOfPages = intRecordCount / PageSize;

if (intRecordCount % PageSize > 0)

NumberOfPages++;

//Get the data specifically for the page

string sSQL =

"SELECT * FROM " +

" (SELECT KEYCOL, COL1, COL2, COL7, rownumber() " +

" over(ORDER BY KEYCOL ASC) AS rn " +

" FROM GLENN.LOTSOFRECORDS " +

" ORDER BY KEYCOL ASC) AS a1 " +

" WHERE a1.rn BETWEEN ? AND ?";

BdpCommand cmdSel = new BdpCommand(sSQL, conn);

BdpParameter prmStart =

cmdSel.Parameters.Add("StartRecord", BdpType.Int32);

prmStart.Value = startRecord;

BdpParameter prmEnd =

cmdSel.Parameters.Add("EndRecord", BdpType.Int32);

prmEnd.Value = endRecord;

BdpDataAdapter da = new BdpDataAdapter(cmdSel, conn);

DataSet ds = new DataSet();

da.Fill(ds, "Table1");

diff.Stop();

return ds;

}

private void LoadSingleDataPage(int pageNo)

{

//Display the Page contents

int PageCount;

DataSet dsData = GetDataByPage(pageNo+1,

dataGrid1.PageSize, out PageCount);

dataGrid1.VirtualItemCount = PageCount * dataGrid1.PageSize;

dataGrid1.CurrentPageIndex = pageNo;

dataGrid1.DataSource = dsData;

dataGrid1.DataBind();

}

private void Page_Load(object sender, System.EventArgs e)

{

if (!IsPostBack)

{

LoadSingleDataPage(0);

}

}

private void dataGrid1_PageIndexChanged(object source,

System.Web.UI.WebControls.DataGridPageChangedEventArgs e)

{

LoadSingleDataPage(e.NewPageIndex);

}

通过定制分页,将检索记录数从1万条限制到20条,占用的网络资源约为最初的0.2%。在我的本机用TimeDiff 类检测,使用定制分页时用时0.5~0.7秒,而使用默认分页机制时用时为0.9~1.5秒。

规则4:使用存储过程

如果发送一条SQL语句到DB2服务器,其执行过程如下:

(1)DB2 UDB Server对SQL语句进行语法检查;

(2)生成存储过程执行计划;

(3)数据返回应用程序。

当使用存储过程时,前两步过程已经完成。存储过程经编译后,调用时只将存储过程名和参数传递给数据库服务器。因此,执行时间的减少赢得了性能上的优势。但这种优势只有当返回的结果集非常大时才有所体现。

下面举一个例子。先建立一个PRODUCTLIST表:

CREATE TABLE "GLENN "."PRODUCTLIST" (

"PRODUCTCODE" VARCHAR(20) NOT NULL ,

"PRODUCTNAME" VARCHAR(50) NOT NULL ,

"DESCRIPTION" VARCHAR(255) ,

"UNITPRICE" DOUBLE NOT NULL ,

"CATEGORYCODE" INTEGER ,

"IMAGEURL" CHAR(150) )

IN "USERSPACE1" ;

COMMENT ON TABLE "GLENN "."PRODUCTLIST" IS 'A list of Products in the Shopping Cart';

-- DDL Statements for primary key on Table "GLENN "."PRODUCTLIST"

ALTER TABLE "GLENN "."PRODUCTLIST"

ADD CONSTRAINT "CC1053568050795" PRIMARY KEY

("PRODUCTCODE");

然后进行如下查询:

SELECT PRODUCTLIST.PRODUCTCODE, PRODUCTLIST.PRODUCTNAME, PRODUCTLIST.DESCRIPTION,

PRODUCTLIST.UNITPRICE, PRODUCTLIST.IMAGEURL

FROM GLENN.PRODUCTLIST AS PRODUCTLIST

WHERE PRODUCTLIST.CATEGORYCODE = 2;

以上查询很容易转为存储过程。DB2开发中心(DB2 Development Center)包含一个优秀的存储过程向导,它可以让你轻松生成存储过程(见图2)。

图2 利用DB2开发中心的存储过程向导创建一个新的存储过程

存储过程向导启动后,你只需要选择相应的表、字段和规则,存储过程的创建过程由向导来完成。

存储过程向导的一个极为有用之处是可以利用它轻松创建存储过程的输入参数。创建一个SQL 存储过程(DB2 还能创建Java存储过程),可以选择“Category Code”作为存储过程的输入参数(见图3)。

图3 创建存储过程输入参数

完成后,所创建的存储过程如下:

CREATE PROCEDURE GLENN.GETPRODUCTSINCATEGORY ( IN CATCODE INTEGER )

DYNAMIC RESULT SETS 1

------------------------------------------------------------------------

-- SQL Stored Procedure

------------------------------------------------------------------------

P1: BEGIN

-- Declare cursor

DECLARE cursor1 CURSOR WITH RETURN FOR

SELECT PRODUCTLIST.PRODUCTCODE, PRODUCTLIST.PRODUCTNAME, PRODUCTLIST.DESCRIPTION,

PRODUCTLIST.UNITPRICE, PRODUCTLIST.IMAGEURL

FROM GLENN.PRODUCTLIST AS PRODUCTLIST

WHERE PRODUCTLIST.CATEGORYCODE = CATCODE;

-- Cursor left open for client application

OPEN cursor1;

END P1

使用存储过程向导时,可以让存储过程返回不止一条查询的结果。通过这种方法,可以发挥存储过程的最大优势。如果结果集很小,使用存储过程可能要慢于使用查询。你应当不断对数据访问作性能测试。

使用Borland Data Provider调用存储过程与调用查询有所不同,主要是BdpDataAdapter的BdpCommand对象必须要将CommandType设为CommandType.StoredProcedure 并将CommandText设为存储过程名。此外,还需要在BdpCommand对象的参数集合中定义存储过程参数。

一旦定义好参数,只需使用BdpDataAdapter的Fill 方法填充DataSet:

private void GetProductsViaStoredProcedure(int CategoryCode) {

cmdGetDataViaStoredProc.Parameters[0].Value = CategoryCode;

BdpDataAdapter da = new BdpDataAdapter(cmdGetDataViaStoredProc);

da.Fill(dsProducts, "Products");

dataGrid1.DataBind();

}

规则5:尽可能使用缓存

ASP.NET最大的特点之一是缓存。缓存的原理很简单,将经常访问的内容储存在内存中,访问时不需要再到数据库或通过网络去检索数据。访问内存中的信息总是要比通过其它过程或网络访问资源要快。

那么如何使用缓存呢?在ASP.NET中有几种方法。一种方法是在ASP.NET页面头部定义一个页面指示标识,让其自动管理页面缓存。如果你读过我的上一篇文章《利用IBM DB2 UDB建立ASP.NET站点》(Build ASP.NET Web Sites with IBM DB2 Universal Database),应当很熟悉这种技术。通过包含OutputCache指令,可以缓存整页:

<%@ Page language="c#" Debug="true" Codebehind="WebForm1.aspx.cs" AutoEventWireup="false"

Inherits="MyNewWebApplication.WebForm1" %>

<%@ OutputCache Duration="300" VaryByParam="None" VaryByCustom="browser" %>

另一种方法是使用Pages对象内建的Cache 对象。将内容放入缓存时,使用Cache 对象的Insert方法;从缓存取出内容时,使用Cache对象的默认集合。

private void Page_Load(object sender, System.EventArgs e)

{

TimeDiff diff = new TimeDiff();

diff.Start();

string retrievalMethod;

DataSet CategoriesDataSet;

if (Cache["Categories"] == null) {

retrievalMethod = "Database";

connShopping.Open();

CategoriesDataSet = new DataSet();

daCategories.Fill(CategoriesDataSet, "Categories");

Cache.Insert("Categories", CategoriesDataSet);

connShopping.Close();

} else {

retrievalMethod = "Cache";

CategoriesDataSet = (DataSet)Cache["Categories"];

}

diff.Stop();

lblDetails.Text = "Retrieval from the " + retrievalMethod +

" in " + diff.TimeDifferenceText + " seconds";

dataGrid1.DataSource = CategoriesDataSet;

dataGrid1.DataMember = "Categories";

dataGrid1.DataBind();

}

当页面第一次被请求时,我在本机测试的数据检索用时为0.9秒,但从缓存中检索数据几乎没用什么时间,因为TimeDiff类的检测结果为00:00:00。可见,使用缓存是加速Web应用简单而有效的方法。

如果需要定时更新数据,还以使用缓存对象的过期策略。只需要使用Insert 方法的重载版本,指定过期时间即可。下面是每隔6小时自动刷新“Categories”缓存的一种方法:

Cache.Insert("Categories", CategoriesDataSet, null,

System.Web.Caching.Cache.NoAbsoluteExpiration,

TimeSpan.FromHours(6));

总结

在完美的世界里,应用程序总是执行快速,支持无限用户,而且不占任何网络资源。但我们并非置身其中,因此你应当很好地利用本文所概括的有效规则,努力提高Web应用程序的性能。

注:版权属原作者所有,部分内容有删节和修改。错误与遗漏难免,批评指正请联系lucentoff@sohu.com

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有