提高DB2 Web应用程序性能的五条规则.docx
- 文档编号:11616949
- 上传时间:2023-03-28
- 格式:DOCX
- 页数:13
- 大小:21.30KB
提高DB2 Web应用程序性能的五条规则.docx
《提高DB2 Web应用程序性能的五条规则.docx》由会员分享,可在线阅读,更多相关《提高DB2 Web应用程序性能的五条规则.docx(13页珍藏版)》请在冰豆网上搜索。
提高DB2Web应用程序性能的五条规则
优秀的代码行行相似,糟糕的代码各个不同。
规则其实并不重要,重要的是养成良好习惯。
——译者题记
简介
速度和可升级性是网络开发重要的性能要求,而且它们也不难达到。
应用一些简单的规则会提高网络应用的性能。
在本文中,您将了解到使用Borland®C#Builder和IBM®DB2®通用数据库(UDB)8.1开发快速Microsoft®ASP.NETWeb应用程序的5条规则。
分析Web应用程序的性能需要利用一些手段来检测每个操作的性能。
为此,我创建了一个TimeDiff类(见程序清单1),它可以计算数据库操作的用时。
你可以利用TimeDiff类的检测结果作为衡量数据库操作性能的基准,来观察哪些操作最为有效。
我还创建了一个配合TimeDiff类使用的LOTSOFRECORDS表(见程序清单2),它包含10,000条记录,你可以通过对它的操作来观察不同技术之间的性能差异。
DB2具有内部缓冲池,一旦运行一个查询,内部缓冲池启用,因此二次查询的速度较快。
在检测查询速度时,请忽略缓冲池启用前的结果而采用启用后的结果。
程序清单1.TimeDiff类
usingSystem;
namespaceEffeciency
{
///
///这个类用于时间计算。
在例子中,
///我们将利用它检测数据库操作的速度,
///以便作性能上的比较。
///
publicclassTimeDiff
{
DateTimeStartTime;
DateTimeEndTime;
publicTimeDiff(){}
publicvoidStart()
{
StartTime=DateTime.Now;
}
publicvoidStop()
{
EndTime=DateTime.Now;
}
publicstringTimeDifferenceText
{
get
{
TimeSpanTimeDifference=EndTime-StartTime;
returnTimeDifference.ToString();
}
}
}
}
程序清单2.LOTSOFRECORDS的表定义
CREATETABLE"GLENN"."LOTSOFRECORDS"(
"KEYCOL"INTEGERNOTNULL,
"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";
COMMENTONTABLE"GLENN"."LOTSOFRECORDS"IS'TabledesignedtoContainLotsofRecords';
--DDLStatementsforprimarykeyonTable"GLENN"."LOTSOFRECORDS"
ALTERTABLE"GLENN"."LOTSOFRECORDS"
ADDCONSTRAINT"CC1058255334652"PRIMARYKEY
("KEYCOL");
下面,我们就开始介绍这些提高DB2UDBWeb应用程序性能的规则。
我会先介绍提高数据库性能的基本规则,然后列出一些适合以BorlandDataProvider开发ASP.NET应用时的性能规则。
规则1:
按需检索
如果你只能记得住一条规则,那一定要记住这一条:
按需检索。
如果你是“幸存者”电视节目的忠实观众,您会记得其中的参赛者通过限额分配来保证每个人都有充足的食物。
这种做法对于数据库开发同样适用。
如果你的应用程序能够按需运行,那么就会合理地将数据库和网络资源留给其他应用。
这听起来很简单,但我们还是来看一个例子
假设有一个包含10,000行记录和10个字段的表,以及一个需要显示所有记录但只显示3个字段的Web页。
很多开发者经常图省事而使用"select*"语句选择所有的字段:
select*fromGLENN.LOTSOFRECORDS
这种做法应当避免,而应力求只检索需要的字段。
可以在SQL语句中定义要检索的字段,例如:
bordercolorlight="black"bordercolordark="#FFFFFF"align="center"> 9pt"> 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发者园地(IBMDB2DeveloperDomain)和comp.databases.ibm-db2(comp.databases.ibm-db2)等一些相关的Web站点或新闻组,对于保持DB2开发技巧不断更新是一个很好的办法。 你还应当努力熟悉DB2附带的工具,例如DB2索引建议器(IndexAdvisor)。 DB2索引建议器可以根据你递交的查询和所连接的数据库返回最佳索引列表。 规则3: 使用DB2UDB的OLAP功能改善分页 在ASP.NET中,一个最常见的操作是表格分页显示。 ASP.NET中DataGrid组件的默认设置是将表格需要的所有记录都返回客户端,然后再根据选择的页显示相应记录. 程序清单4使用DataGrid内建分页机制 TimeDiffdiff=newTimeDiff(); privateDataSetGetProductsDataSet(){ diff.Start(); stringconnString=ConfigurationSettings.AppSettings["database"]; BdpConnectionconn=newBdpConnection(connString); BdpDataAdapterda=newBdpDataAdapter("selectKEYCOL,"+ "COL1,COL2,COL7FROMGLENN.LOTSOFRECORDS"+ "ORDERBYKEYCOLASC",conn); DataSetds=newDataSet(); da.Fill(ds,"Table1"); diff.Stop(); returnds; } privatevoidBindToTheData() { dataGrid1.DataSource=GetProductsDataSet(); dataGrid1.DataMember="Table1"; dataGrid1.DataBind(); label1.Text=diff.TimeDifferenceText; } privatevoidPage_Load(objectsender,System.EventArgse) { if(! IsPostBack){ BindToTheData(); } } privatevoiddataGrid1_PageIndexChanged(objectsource, System.Web.UI.WebControls.DataGridPageChangedEventArgse) { //Changetheactivepageofthedata dataGrid1.CurrentPageIndex=e.NewPageIndex; BindToTheData(); } 如果以LOTSOFRECORDS表为例,从数据库中检索1万条记录很快就会占尽网络带宽,尤其是一次只需要查看10条或20条记录时。 用户会发现加载ASP.NET页面的时间太长,而且很可能会超时。 试想,如果数据库有数百万条记录,那你的应用程序就会运行缓慢只至停止执行。 因此,需要采取更好的方法来检索数据。 还好,DataGrid组件允许定制分页。 你可以计算哪些记录需要显示,然后只从服务器检索出相应的记录。 后来版本的DB2UDB都具有一个很大的特点,那就是OLAP函数,它允许你进行各种记录检索。 例如,只检索某些记录可以执行如下的查询: SELECT*FROM (SELECTKEYCOL,COL1,COL2,COL7,rownumber() over(ORDERBYKEYCOLASC)ASrn FROMGLENN.LOTSOFRECORDS ORDERBYKEYCOLASC)ASa1 WHEREa1.rnBETWEEN100AND120 使用DataGrid定制分页时,需要获得特定页的DataSet。 GetDataByPage方法可以检索结果集而不管分页数和分页大小: privateDataSetGetDataByPage(intPageNo,intPageSize,outintNumberOfPages) { intstartRecord=(PageNo-1)*PageSize+1; intendRecord=startRecord+PageSize-1; stringconnString=ConfigurationSettings.AppSettings["database"]; BdpConnectionconn=newBdpConnection(connString); conn.Open(); //GetthenumberofPages stringsRecordCount="selectcount(*)fromGLENN.LOTSOFRECORDS"; BdpCommandcmdGetRecordCount=newBdpCommand(sRecordCount,conn); intintRecordCount=(int)cmdGetRecordCount.ExecuteScalar(); NumberOfPages=intRecordCount/PageSize; if(intRecordCount%PageSize>0) NumberOfPages++; //Getthedataspecificallyforthepage stringsSQL= "SELECT*FROM"+ "(SELECTKEYCOL,COL1,COL2,COL7,rownumber()"+ "over(ORDERBYKEYCOLASC)ASrn"+ "FROMGLENN.LOTSOFRECORDS"+ "ORDERBYKEYCOLASC)ASa1"+ "WHEREa1.rnBETWEEN? AND? "; BdpCommandcmdSel=newBdpCommand(sSQL,conn); BdpParameterprmStart= cmdSel.Parameters.Add("StartRecord",BdpType.Int32); prmStart.Value=startRecord; BdpParameterprmEnd= cmdSel.Parameters.Add("EndRecord",BdpType.Int32); prmEnd.Value=endRecord; BdpDataAdapterda=newBdpDataAdapter(cmdSel,conn); DataSetds=newDataSet(); da.Fill(ds,"Table1"); diff.Stop(); returnds; } privatevoidLoadSingleDataPage(intpageNo) { //DisplaythePagecontents intPageCount; DataSetdsData=GetDataByPage(pageNo+1, dataGrid1.PageSize,outPageCount); dataGrid1.VirtualItemCount=PageCount*dataGrid1.PageSize; dataGrid1.CurrentPageIndex=pageNo; dataGrid1.DataSource=dsData; dataGrid1.DataBind(); } privatevoidPage_Load(objectsender,System.EventArgse) { if(! IsPostBack) { LoadSingleDataPage(0); } } privatevoiddataGrid1_PageIndexChanged(objectsource, System.Web.UI.WebControls.DataGridPageChangedEventArgse) { LoadSingleDataPage(e.NewPageIndex); } 通过定制分页,将检索记录数从1万条限制到20条,占用的网络资源约为最初的0.2%。 在我的本机用TimeDiff类检测,使用定制分页时用时0.5~0.7秒,而使用默认分页机制时用时为0.9~1.5秒。 规则4: 使用存储过程 如果发送一条SQL语句到DB2服务器,其执行过程如下: (1)DB2UDBServer对SQL语句进行语法检查; (2)生成存储过程执行计划; (3)数据返回应用程序。 当使用存储过程时,前两步过程已经完成。 存储过程经编译后,调用时只将存储过程名和参数传递给数据库服务器。 因此,执行时间的减少赢得了性能上的优势。 但这种优势只有当返回的结果集非常大时才有所体现。 下面举一个例子。 先建立一个PRODUCTLIST表: CREATETABLE"GLENN"."PRODUCTLIST"( "PRODUCTCODE"VARCHAR(20)NOTNULL, "PRODUCTNAME"VARCHAR(50)NOTNULL, "DESCRIPTION"VARCHAR(255), "UNITPRICE"DOUBLENOTNULL, "CATEGORYCODE"INTEGER, "IMAGEURL"CHAR(150)) IN"USERSPACE1"; COMMENTONTABLE"GLENN"."PRODUCTLIST"IS'AlistofProductsintheShoppingCart'; --DDLStatementsforprimarykeyonTable"GLENN"."PRODUCTLIST" ALTERTABLE"GLENN"."PRODUCTLIST" ADDCONSTRAINT"CC1053568050795"PRIMARYKEY ("PRODUCTCODE"); 然后进行如下查询: SELECTPRODUCTLIST.PRODUCTCODE,PRODUCTLIST.PRODUCTNAME, PRODUCTLIST.DESCRIPTION,PRODUCTLIST.UNITPRICE, PRODUCTLIST.IMAGEURL FROMGLENN.PRODUCTLISTASPRODUCTLIST WHEREPRODUCTLIST.CATEGORYCODE=2; 以上查询很容易转为存储过程。 DB2开发中心(DB2DevelopmentCenter)包含一个优秀的存储过程向导,它可以让你轻松生成存储过程。 存储过程向导启动后,你只需要选择相应的表、字段和规则,存储过程的创建过程由向导来完成。 存储过程向导的一个极为有用之处是可以利用它轻松创建存储过程的输入参数。 创建一个SQL存储过程(DB2还能创建Java存储过程),可以选择“CategoryCode”作为存储过程的输入参数。 完成后,所创建的存储过程如下: CREATEPROCEDUREGLENN.GETPRODUCTSINCATEGORY(INCATCODEINTEGER) DYNAMICRESULTSETS1 ------------------------------------------------------------------------ --SQLStoredProcedure ------------------------------------------------------------------------ P1: BEGIN --Declarecursor DECLAREcursor1CURSORWITHRETURNFOR SELECTPRODUCTLIST.PRODUCTCODE,PRODUCTLIST. PRODUCTNAME,PRODUCTLIST.DESCRIPTION,PRODUCTLIST.UNITPRICE, PRODUCTLIST.IMAGEURL FROMGLENN.PRODUCTLISTASPRODUCTLIST WHEREPRODUCTLIST.CATEGORYCODE=CATCODE; --Cursorleftopenforclientapplication OPENcursor1; ENDP1 使用存储过程向导时,可以让存储过程返回不止一条查询的结果。 通过这种方法,可以发挥存储过程的最大优势。 如果结果集很小,使用存储过程可能要慢于使用查询。 你应当不断对数据访问作性能测试。 使用BorlandDataProvider调用存储过程与调用查询有所不同,主要是BdpDataAdapter的BdpCommand对象必须要将CommandType设为CommandType.StoredProcedure并将CommandText设为存储过程名。 此外,还需要在BdpCommand对象的参数集合中定义存储过程参数。 一旦定义好参数,只需使用BdpDataAdapter的Fill方法填充DataSet: privatevoidGetProductsViaStoredProcedure(intCategoryCode){ cmdGetDataViaStoredProc.Parameters[0].Value=CategoryCode; BdpDataAdapterda=newBdpDataAdapter(cmdGetDataViaStoredProc); da.Fill(dsProducts,"Products"); dataGrid1.DataBind(); } 规则5: 尽可能使用缓存 ASP.NET最大的特点之一是缓存。 缓存的原理很简单,将经常访问的内容储存在内存中,访问时不需要再到数据库或通过网络去检索数据。 访问内存中的信息总是要比通过其它过程或网络访问资源要快。 那么如何使用缓存呢? 在ASP.NET中有几种方法。 一种方法是在ASP.NET页面头部定义一个页面指示标识,让其自动管理页面缓存。 如果你读过我的上一篇文章《利用IBMDB2UDB建立ASP.NET站点》(BuildASP.NETWebSiteswithIBMDB2UniversalDatabase),应当很熟悉这种技术。 通过包含OutputCache指令,可以缓存整页: 另一种方法是使用Pages对象内建的Cache对象。 将内容放入缓存时,使用Cache对象的Insert方法;从缓存取出内容时,使用Cache对象的默认集合。 privatevoidPage_Load(objectsender,System.EventArgse) { TimeDiffdiff=newTimeDiff(); diff.Start(); stringretrievalMethod; DataSetCategoriesDataSet; if(Cache["Categories"]==null){ retrievalMethod="Database"; connShopping.Open(); Cate 如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。 copyright@ 2008-2022 冰点文档网站版权所有 经营许可证编号:鄂ICP备2022015515号-1