明輝手游網(wǎng)中心:是一個免費提供流行視頻軟件教程、在線學習分享的學習平臺!

如何使用XSL與正則表達式來驗證數(shù)據(jù)的有效果性(二)

[摘要]現(xiàn)在繼續(xù)我在上一系列中介紹的例子。我們將實現(xiàn)一個簡單的搜索機制, 也許你的興趣不是在獲取書籍的標題而是在要獲得整個書節(jié)點(note)上。 那么使用下面這個函數(shù)就能夠滿足你的要求了 Public Function GetFilteredElements(activeElement As Varian...

現(xiàn)在繼續(xù)我在上一系列中介紹的例子。我們將實現(xiàn)一個簡單的搜索機制,
也許你的興趣不是在獲取書籍的標題而是在要獲得整個書節(jié)點(note)上。
那么使用下面這個函數(shù)就能夠滿足你的要求了
Public Function GetFilteredElements(activeElement As Variant, RegExpfilter As String, _
Optional queryString As String = "",optional IsGlobal as Boolean=True, _
optional IgnoreCase as Boolean = True) As IXMLDOMNodeList
Dim re As RegExp
Dim filterDoc As DOMDocument
Dim nodelist As IXMLDOMNodeList
Dim filterList As IXMLDOMNodeList
Dim node As IXMLDOMElement
Dim baseNode As IXMLDOMElement
Set re = New RegExp
On Error GoTo ErrorHandler
Select Case TypeName(activeElement)
Case "IXMLDOMElement"
Set baseNode = activeElement
Case "DOMDocument"
Set baseNode = activeElement.documentElement
Case Else
Error 1001
Set GetFilteredElements = Nothing
End Select
re.Pattern = RegExpfilter
re.Global=IsGlobal
re.IgnoreCase=IgnoreCase
If queryString = "" Then
Set filterList = baseNode.selectNodes(".[textnode()] .//*[textnode()]")
Else
Set filterList = baseNode.selectNodes(queryString)
End If
For Each node In filterList
If re.Test(node.Text) Then
node.setAttribute "filter:filteredElementFound", "true"
End If
Next
Set filterList = baseNode.selectNodes(".[@filter:filteredElementFound] .//*[@filter:filteredElementFound]")
For Each node In filterList
node.removeAttribute "filter:filteredElementFound"
Next
Set GetFilteredElements = filterList
Exit Function
ErrorHandler:
If Err.Number = 1001 Then
MsgBox "Document must be an XML document, or a document element."
Else
Error Err.Number
End If
End Function

下面是對這個方法的簡單描述:
GetFilteredElements主要是用來獲取一個XML的文檔或則一個文檔里面的節(jié)點,并且
按照我們的需要轉(zhuǎn)換它(包括所有的該節(jié)點的子節(jié)點),將轉(zhuǎn)換后的節(jié)點放到一個
列表類型IXMLDOMNodeList中.
這個過濾器程序按次序?qū)γ總節(jié)點進行運用,如果一個節(jié)點的文本滿足了表達式的話,
那么這個節(jié)點就被做上一個臨時屬性的標記為filter:filteredElementFound
(當然為了不和你自己的XML文檔里已經(jīng)存在的標志發(fā)生沖突,你可以把這個屬性
改成你需要的東西).
一旦所有的節(jié)點都被檢查完畢后,一個新的列表對象(僅僅只包含那些滿足表達式的
節(jié)點)被創(chuàng)建了。這些臨時的屬性會被刪除掉,然后返回那些節(jié)點。

如果你沒有給這個函數(shù)輸入查詢參數(shù)的話,
那么將獲得整個文檔的所有葉子節(jié)點(就是那些沒有子元素的節(jié)點)
如果你給這個函數(shù)輸入查詢參數(shù)的話,
那么這個函數(shù)將返回滿足條件的節(jié)點或則子節(jié)點
例如,下面的代碼將返回第一本書的標題和描述的節(jié)點和第四本書的
描述節(jié)點
Dim bookXML=new DOMDocument
bookXML.load("bookCatalog.xml")
Set nodelist=GetfilteredElements(bookXML,"xml")
而下面的代碼將返回第一和第四本書的節(jié)點,然后你可以根據(jù)這些節(jié)點獲取它們的
子節(jié)點的屬性。
Set nodelist=GetfilteredElements(bookXML,"xml","http://book")
通常,你應(yīng)該盡可能的定義一個XSL的查詢過濾器(filter)
因為它會只返回你需要的節(jié)點,會大大減少你需要處理的數(shù)據(jù)量。

這個函數(shù)是XML "數(shù)據(jù)庫"的一個運用實例,因為很多SQL的開發(fā)者用熟悉了的
參數(shù)(例如LIKE)在XML中是沒有等效的方法的,但是只要你在XML中能夠熟練使用
正則表達式,你會發(fā)現(xiàn)它能夠?qū)崿F(xiàn)很多LIKE語句能夠?qū)崿F(xiàn)的功能。

使用XSL轉(zhuǎn)換驗證數(shù)據(jù)
正則表達式使用在XSL的轉(zhuǎn)換中是一個強有力的驗證數(shù)據(jù)有效的工具。
例如,你想根據(jù)XML的數(shù)據(jù)生成一個顯示書籍標題和描述的table
當你使用基于DOM版本(就是使用微軟的那個模型XDOM)的時候,XSL已經(jīng)能夠?qū)崿F(xiàn)非常復雜的轉(zhuǎn)換XML為
HTML的功能了。
在XSL的這個節(jié)點中<xsl:script>,允許你使用腳本語言。
你可以在輸出流中插入文本(但是目前你就不能夠把一個DOM節(jié)點輸出到輸出流中).
XSL中默認的腳本語言是JavaScript,
在使用的過程中你需要注意的是,由于"/"在JavaScript中是特殊字符,你需要使用
"http://"將其轉(zhuǎn)意。
代碼如下:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/TR/WD-xsl">
<xsl:script language="JavaScript"><![CDATA][
IsValidBookTopic=/xml/
]]></xsl:script>
<xsl:template match="/">
<xsl:apply-templates select="http://book" />
</xsl:template>
<xsl:template match="book">
<xsl:if expr="IsValidBookTopic.test(this.text)">
<h1><xsl:value-of select="title"/></h1>
<h2><xsl:value-of select="author"/></h2>
<p><xsl:value-of select="description"/></p>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
當然,你也可以改變你的XSL中的查詢參數(shù)(例如上面例子中使用的參數(shù)是xml)
一個直接的方法是生成一個參數(shù)實體"entities"
例如下面的代碼,你需要使用"%"字符來說明一個字符是參數(shù)。
Function SetXSLParameter(XslDoc as DOMDocument ,ParamName as _
String,ParamValue as Variant) as DOMDocument
Dim XslDoc as DOMDocument
Dim ScriptNode as IXMLDOMElement
Dim re as RegExp

Set re=new RegExp
Re.global=True
Re.IgnoreCase=True
Re.pattern="%"+ParamName
For each ScriptNode in xslDoc.selectNodes("xsl:script")
ScriptNode.text=Re.replace(ScriptNode.text,cstr(ParamValue))
Next
Return XslDoc
End Function

SetXSLParameter用來給XSL文檔設(shè)置參數(shù)。
當然你也可以不使用上面這個函數(shù),直接在XSL的腳本里面修改這個參數(shù),
但是如果你使用了上面這個函數(shù)的話,可以讓你的代碼的可擴展性能更好。
注意的是,XSL的代碼將被上面這個函數(shù)所修改
所以如果你想對相同的XSL對象使用不同的過濾器(就是不同的查詢條件的話)
需要使用XSL對象的clone(就是制作一個XSL的副本)方法。
下面就是一個完整的例子
<xsl:stylesheet xmlns:xsl="http://www.w3.org/TR/WD-xsl">
<xsl:script language="JavaScript"><![CDATA][
IsValidBookTopic=/%searchStr/
]]></xsl:script>
<xsl:template match="/">
<xsl:apply-templates select="http://book" />
</xsl:template>
<xsl:template match="book">
<xsl:if expr="IsValidBookTopic.test(this.text)">
<h1><xsl:value-of select="title"/></h1>
<h2><xsl:value-of select="author"/></h2>
<p><xsl:value-of select="description"/></p>
</xsl:if>
</xsl:template>
</xsl:stylesheet>

然后你就可以使用下面這段代碼來修改這個參數(shù)了
呵呵,我是搞asp的,下面這段代碼就是寫在你的asp的程序中的。

Set xmlDoc=new DomDocument
Set xslDoc=new DOMDocument
XmlDoc.load "catalog.xml"
XslDoc.load "catalog.xsl"
SetXSLParameter xslDoc, "searchStr", "pair-o-dice"
Response.write xmlDoc.transformNode(xslDoc)

同樣,使用上面的方法,你就可以設(shè)置其他的參數(shù)了,例如瀏覽器的類型呀
ASP的查詢參數(shù)呀,或則一個基于Web的組件呀。

使用正則表達式給XSL帶來了強大的功能。
雖然,使用這種通用性很強的方法在目前也許你不會看到能夠給你帶來很
大的優(yōu)勢(主要是很少有人寫XML的程序的說),但是在未來的編程過程中,
你將會體會到它的強大威力了。