轉(zhuǎn)帖|其它|編輯:郝浩|2011-03-16 13:28:49.000|閱讀 880 次
概述: Webpart部署后在SPD中打開顯示呈現(xiàn)控件時(shí)出錯(cuò),后查到這篇文章,主要問題是在CreateChildControls中調(diào)用了 this.page.Header,在設(shè)計(jì)視圖時(shí),該對(duì)象可能還不存在,以后編碼時(shí)要注意在CreateChildControls盡量不調(diào)用數(shù)據(jù)庫(kù)/文件系統(tǒng)讀取、變量未附初值、調(diào)用了類似Page的屬性如Session,Page.Request.QueryString等。
# 界面/圖表報(bào)表/文檔/IDE等千款熱門軟控件火熱銷售中 >>
Webpart部署后在SPD中打開顯示呈現(xiàn)控件時(shí)出錯(cuò),后查到這篇文章,主要問題是在CreateChildControls中調(diào)用了this.page.Header,在設(shè)計(jì)視圖時(shí),該對(duì)象可能還不存在,以后編碼時(shí)要注意在CreateChildControls盡量不調(diào)用數(shù)據(jù)庫(kù)/文件系統(tǒng)讀取、變量未附初值、調(diào)用了類似Page的屬性如Session,Page.Request.QueryString等。
在制作控件的時(shí)候容易遇到呈現(xiàn)控件時(shí)出錯(cuò)發(fā)生了未處理的異常。未將對(duì)象引用設(shè)置到對(duì)象的實(shí)例。這樣的錯(cuò)誤,如下圖:(也有可能僅僅只是因?yàn)槲丛O(shè)置runat="server"標(biāo)簽而導(dǎo)致該錯(cuò)誤,請(qǐng)先檢查該項(xiàng)。)
但是在編譯時(shí)/運(yùn)行時(shí)也都是沒有錯(cuò)誤的。
分析
如圖所示的情況我們稱之為設(shè)計(jì)時(shí)以區(qū)別于編譯時(shí)/運(yùn)行時(shí)。
設(shè)計(jì)時(shí):在ASP.NET環(huán)境中也就是當(dāng)我們?cè)赩isual Studio中使用網(wǎng)頁(yè)設(shè)計(jì)器進(jìn)行編輯的時(shí)候。直接的理解就是在ASPX頁(yè)面切換到設(shè)計(jì)的時(shí)候。
編譯時(shí):直接的理解就是當(dāng)你進(jìn)行編譯的時(shí)候,通常這個(gè)時(shí)候的錯(cuò)誤是由類型檢查,參數(shù)匹配等顯式可直接通過語(yǔ)法約束所限制的錯(cuò)誤。
運(yùn)行時(shí):直接的理解就是當(dāng)你進(jìn)行預(yù)覽/運(yùn)行的時(shí)候。通常這個(gè)時(shí)候的錯(cuò)誤則是由具體的異常,邏輯錯(cuò)誤所組成的。
讓我們分析控件在設(shè)計(jì)時(shí)的表現(xiàn),我們的控件在設(shè)計(jì)時(shí)的時(shí)候,VS智能地模擬輸出控件在運(yùn)行時(shí)的樣貌,控件的呈現(xiàn)通過了一定順序的方法,并最終形成了當(dāng)前輸出。按照標(biāo)準(zhǔn),我們應(yīng)該是在Render或RenderContents中對(duì)控件進(jìn)行了輸出的操作(事實(shí)上其他也是可以,但我們通常也不那么做,或者說更多的呈現(xiàn)控件時(shí)出錯(cuò)的異常主要來(lái)自于Render或RenderContents)。
從錯(cuò)誤的提示未將對(duì)象引用設(shè)置到對(duì)象的實(shí)例。從這一句話來(lái)看,也就是說,有一個(gè)或者以上的對(duì)象的實(shí)例在沒有賦初值的情況下就被使用了。
讓我們窺視一下我們的代碼:
protected override void RenderContents(HtmlTextWriter writer)
{
UpButton.Text = Page.Server.HtmlDecode(UpButton.Text);
DownButton.Text = Page.Server.HtmlDecode(DownButton.Text);
base.RenderContents(writer);
}
因?yàn)樵摽丶谠O(shè)計(jì)的時(shí)候需要有一個(gè)向上的按鈕和一個(gè)向下的按鈕,分別需要用兩個(gè)特殊的標(biāo)點(diǎn)符號(hào)向上和向下,而這兩個(gè)符號(hào)需要通過設(shè)置如下所示的代碼編號(hào)才可以正確地被瀏覽器呈現(xiàn):
private string upButtonText = "and;";
private string downButtonText = "or;";
而這兩個(gè)符號(hào)在呈現(xiàn)前卻會(huì)被頁(yè)面進(jìn)行一個(gè)HtmlEncode方法編譯后再輸出,而這兩個(gè)特殊的標(biāo)志卻只能通過直接輸出的方式進(jìn)行呈現(xiàn),也就是說在HtmlEncode之后只能將該特殊標(biāo)志以文本的形式輸出and; or;而不會(huì)輸出向上和向下的箭頭。這時(shí)候我們需要引入它的反向方法Page.Server.HtmlDecode進(jìn)行解碼,注意到這里我們使用了Page實(shí)例,該實(shí)例只有在頁(yè)面真實(shí)存在的情況下才為非空,否則后續(xù)的操作將是對(duì)null的操作,而這樣的操作將會(huì)顯示未將對(duì)象引用設(shè)置到對(duì)象的實(shí)例。這樣的錯(cuò)誤。
設(shè)計(jì)時(shí):我們剛才提到了,設(shè)計(jì)時(shí)僅僅只是模擬頁(yè)面呈現(xiàn)的過程,而頁(yè)面事實(shí)上是不存在的。因此在這個(gè)時(shí)候Page對(duì)象的實(shí)例將為空,后續(xù)的調(diào)用將引發(fā)異常。
假設(shè)說我們只有這個(gè)方法用于處理當(dāng)前所需要的行為,那么我們?cè)谡{(diào)用Page的時(shí)候必將導(dǎo)致null的對(duì)象并致使后續(xù)操作發(fā)生異常。這個(gè)時(shí)候我們引入設(shè)計(jì)模式這個(gè)概念(非DesignPattern而是DesignMode),DesignMode是由Control類的一個(gè)受保護(hù)的(protected)屬性,它獲取一個(gè)值,用以指示該組件當(dāng)前是否處于設(shè)計(jì)模式。這里的設(shè)計(jì)模式也就相當(dāng)于設(shè)計(jì)時(shí)的概念。
因此我們可以將代碼改造為如下形式:
protected override void RenderContents(HtmlTextWriter writer)
{
if (!DesignMode)
{
UpButton.Text = Page.Server.HtmlDecode(UpButton.Text);
DownButton.Text = Page.Server.HtmlDecode(DownButton.Text);
}
base.RenderContents(writer);
}
這樣只有在非設(shè)計(jì)時(shí)的時(shí)候,我們才引入Page對(duì)象的實(shí)例,因此設(shè)計(jì)時(shí)的異常將迎刃而解。
總結(jié)
因此在設(shè)計(jì)控件的時(shí)候,特別是在考慮控件呈現(xiàn)的時(shí)候,為了避免類似異常的發(fā)生,我們應(yīng)該考慮該控件在設(shè)計(jì)時(shí)能夠獲取足夠的資源,對(duì)于未能獲取資源的項(xiàng),我們應(yīng)該顯式將其區(qū)分(如上代碼中使用DesignMode來(lái)判斷是否為設(shè)計(jì)時(shí)要執(zhí)行的代碼)。
擴(kuò)展
剛才我們所見到的情形可以歸結(jié)為在設(shè)計(jì)時(shí)無(wú)法引用具體實(shí)例所導(dǎo)致的異常,類似該異常的還會(huì)有數(shù)據(jù)庫(kù)/文件系統(tǒng)讀取、變量未附初值、調(diào)用了類似Page的屬性如Session,Page.Request.QueryString等。在頁(yè)面設(shè)計(jì)的時(shí)候由于以上部分方案特別是調(diào)用到Page的相關(guān)方法的由于頁(yè)面總是會(huì)存在,因此我們不會(huì)經(jīng)常看到它們出異常,在設(shè)計(jì)控件的時(shí)候我們更應(yīng)該注意。
本站文章除注明轉(zhuǎn)載外,均為本站原創(chuàng)或翻譯。歡迎任何形式的轉(zhuǎn)載,但請(qǐng)務(wù)必注明出處、不得修改原文相關(guān)鏈接,如果存在內(nèi)容上的異議請(qǐng)郵件反饋至chenjj@fc6vip.cn
文章轉(zhuǎn)載自:博客園