第一個XML文檔

學XSL(一) (二) (三) (四) (五) (六) (七) (八)  (返回) 

 

2000-05-03· 禹希初·CPCW

第一個XML文檔

  隨着Internet的發展,越來越多的信息進入互聯網,信息的交換、檢索、保存及再利用等迫切的需求使HTML這種最常用的標記語言已越來越捉襟見肘。HTML將數據內容與表現融爲一體,可修改性、數據可檢索性差,而XML借鑑了HTML與數據庫、程序語言的優點,將內容與表現分開,不僅使檢索更爲方便,更主要的是用戶之間數據的交換更加方便,可重用性更強。

  XML是一種元標記語言,沒有許多固定的標記,爲WEB開發人員提供了更大的靈活性。當我們使用HTML時,標記只是簡單的表示內容的顯示形式,而與表示的內容沒有任何關聯,爲文檔的進一步處理帶來極大的不便。比如要表示個人簡歷,用HTML的表示方式如下: 

< HTML >

< BODY >

< TABLE border=1 cellspacing=0 >

< TH >姓名< TD >禹希初< TH >性別< TD >男< TH >生日< TD >1977.5

< TR >

< TH >技能< TD colspan=5 >數據庫設計與維護、WEB開發

< /TABLE >

< /BODY >

< /HTML >

  在這裏,我們無法從標記TH、TD得知其內容表示什麼,如果用XML,相應的文檔(文件名:個人簡歷.xml)就可寫成如下形式: 

< ?xml version="1.0" encoding="GB2312"? >

< resume >

< name >禹希初< /name >

< sex >男< /sex >

< birthday >1977.5< /birthday >

< skill >數據庫設計與維護、WEB開發< /skill >

< /resume >

說明:

  version──規定了XML文檔的版本,此處只能是1.0;

  encoding── 此處規定了XML文檔的編碼類型,此處取值爲“GB2312”,也就是“簡體中文”。

  對比兩例,使用XML我們可以做到自定義標記,用標記表明內容的含義。這樣在Internet上交流資料時,爲用計算機處理文檔提供了極大的方便,同時我們閱讀源文件時也不會被一大堆格式弄得暈頭轉向。

  然而,由於XML並沒有爲標記規定顯示方式,如果我們在遊覽器中查看以上兩個文檔(建議使用IE5.0或更新版本),我們將看到xml文檔並沒有以諸如表格的方式來顯示。難道我們就不能像HTML一樣顯示文檔嗎?回答是否定的。以個人簡歷爲例,需要另建一個格式文件說明各個標記的顯示方式,其內容如下(假設文件名爲resume.css): 

resume{ display: block;}

name{ display: block; font-size:120%;}

sex{ display:block; text-indent:2em}

birthday{ display:block; text-indent:2em}

skill{ display:block; text-indent:2em}

說明:

  以上均爲CSS樣式,建議讀者參考有關資料熟悉CSS,在以後學習中必須用到,此處由於篇幅關係不作介紹。建立文件resume.css後,在個人簡歷.xml文件的第一行後添加以下文字: 

< ?xml:stylesheet type="text/css" href="resume.css"? >

說明:

  此處表示引用一個外部CSS樣式文件,其中type規定樣式類型(可取值爲text/css或text/xsl),href規定文件路徑。

  保存文件,再以IE5.0打開文件。怎麼樣?格式有些不一樣吧。好象還不令人滿意,文檔內容是清晰了,但顯示效果比HTML編寫的文檔就差得多了,XML編寫的文檔就只能以這種方式顯示嗎?!

提示:

  1. 爲了更好的理解與掌握XML,建議大家熟悉HTML 4.0與CSS 2.0語法;掌握JavaScript、VBscript中至少一種;編程經驗、對數據庫理論與SQL的瞭解均能使大家在學習XML時獲益。

  2. XML文檔中標記必須成對出現,如果是空標記也必須有前加“/”的同名標記結束,或使用此種文式< xml_mark/ >表示空標記。

  3. XML以及下週將要介紹的XSL文檔,屬性值必須用雙引號(")或單引號(')括起來。

  4. XML文檔必須是好結構的(XSL文檔也是XML文檔中一種),也就是說標記必須有結束標記、標記可以嵌套但不可交叉,如 

< outer >< inner >< /inner >< inner/ >< /outer >

是合法的,而下面的形式 

< outer >< inner >< /outer >< /inner >

則是錯誤的。如果XML文檔在瀏覽時出錯,多半是違反了上面提到的規則。

跟我學XSL(二)(返回標題)

2000-05-04· 禹希初·CPCW

XSL入門

  上期我們講到用CSS(層疊樣式表)來格式化XML文檔,其效果並不很令人滿意。實際上CSS用來格式化HTML標記比較合適些,只是因爲它簡單纔在上例中採用。

  XML在更多的時候只是一種數據文件,怎樣將它變爲我們日常所看到的HTML格式那樣的文件呢?如果我們將XML文件比作結構化的原料的話,那麼XSL就好比“篩子“與“模子“,篩子選取自己需要的原料,這些原料再通過模子形成最終的產品:HTML。

  這個模子大致是這樣:我們先設計好表現的頁面,再將其中需要從XML中獲取數據來填充內容的部分“挖掉“,然後用XSL語句從XML中篩出相關的數據來填充。一言以譬之:這XSL實際上就是HTML的一個“殼子“,XML數據利用這個“殼“來生成“傳統“的HTML。

  XML在展開時是一個樹形結構,我們將樹形結構中自定義標記稱爲節點,節點之間存在父子、兄弟關係,我們要訪問其中的結點從根結點就要以”/”來層層進入。

  在XSL這個殼中,我們要從原料庫??XML裏提取相關的數據,就要用到XSL提供的模式化查詢語言。所謂模式化查詢語言,就是通過相關的模式匹配規則表達式從XML裏提取數據的特定語句,即我們上所說的“篩子“。

參考微軟的”XSL開發者指南”,我們大致可將模式語言分爲三種: 

選擇模式 

< xsl:for-each >、< xsl:value-of >,和 < xsl:apply-templates >

測試模式 

< xsl:if > 和< xsl:when >

匹配模式 

< xsl:template > 

我們現在就分別對之進行介紹。

一、 選擇模式

  選擇模式語句將數據從XML中提取出來,是一種簡單獲得數據的方法,這幾個標記都有一個select屬性,選取XML中特定的結點名的數據。 

1、< xsl:for-each >

如在XML中有這樣的數據:

< author >

< name >小禹< /name >

< name >春華< /name >

< name >秋實< /name >

< /author >

  我們要讀取這三個作者名字,是一個一個地按”author/name”方法來讀取嗎,可有多個這樣的name呀?如果有一種程序性的語句來循環讀取有多好啊!

  想得很對,XSL提供了這樣的具有程序語言性質的語句< XSL:for-each >

  用它讀取這三個作者名字的方法如下:

< xsl:for-each select=”author/name” >

…….

< ./xsl:for-each >

select,顧名思義:選取,它可以選定XML中特定唯一的標記,也可以選擇某一類相同的標記,我們稱之爲結點集。 

語法: 

< xsl:for-each select="pattern" order-by="sort-criteria-list">

屬性: 

1. select

  根據XSL樣式查詢考察上下文以決定哪類結點集(滿足select條件)使用此樣式描述。作爲一種簡化的表示就是,如果你想對文檔中的某一種標記的內容的顯示方式進行格式化,就可以將讓select等於此元素的標記名。例如欲對標記xml_mark進行格式化,即可用如下方式表示: 

< xsl:for-each select="xml_mark" >

< !--樣式定義-- >

< /xsl:for-each >

2. order-by

  以分號(;)分隔、作爲排序標準的列表。在列表元素前添加加號(+)表示按此標記的內容以升序排序,添加減號(-)表示逆序排序。作爲一種簡化的表示就是,排序標準列表就是由select規定的標記的子標記的序列,每個標記之間以(;)分隔。 

2、< xsl:value-of >

< xsl:for-each >模式只是選取節點,並沒有取出節點的值,好比猴子只是爬到了樹的某個枝幹上,那麼就用< xsl:value-of >來摘“勝利果實“吧! 

語法: 

< xsl:value-of select="pattern" >提取節點的值 

屬性: 

select用來與當前上下文匹配的XSL式樣。簡單的講,如果要在XSL文檔某處插入某個XML標記(假定是xml_mark標記)的內容,可用如下方式表示: 

< xsl:value-of select="xml_mark" >< /xsl:value-of >

或 

< xsl:value-of select="xml_mark"/ >

示例:

  此處仍以上期的個人簡歷的作爲例子,我們需要對文件(個人簡歷.xml)作一定修改,確切的說是將其中的第二行 

< ?xml:stylesheet type="text/css" href="resume.css"? >

修改爲 

< ?xml:stylesheet type="text/xsl" href="resume.xsl"? >

然後建立一個新文件:resume.xsl,其內容如下: 

< ?xml version="1.0" encoding="GB2312"? >

< HTML xmlns:xsl="http://www.w3.org/TR/WD-xsl" >

< HEAD >

< TITLE >個人簡歷< /TITLE >

< /HEAD >< BODY > 

< xsl:for-each select="resume" >

< P/ >

< TABLE border="1" cellspacing="0" >

< CAPTION style="font-size: 150%; font-weight: bold" >

個人簡歷 

< /CAPTION >

< TR >

< TH >姓名< /TH >< TD >< xsl:value-of select="name"/ >< /TD >

< TH >性別< /TH >< TD >< xsl:value-of select="sex"/ >< /TD >

< TH >生日< /TH >< TD >< xsl:value-of select="birthday"/ >< /TD >

< /TR >

< TR >

< TH >技能< /TH >< TD colspan="5" >< xsl:value-of select="skill"/ >< /TD >

< /TR >

< /TABLE >

< /xsl:for-each >

< /BODY >

< /HTML >

  完成這些以後再來讓我們看一下辛勤勞動的成果,怎麼樣?效果不錯吧。更酷還在後頭呢。現在我們對文件(個人簡歷.xml)作進一步的修改: 

1. 在標記< resume >前添加一個新標記< document >; 

2. 將標記對< resume >< /resume >之間的內容(包括這一對標記)複製並粘貼在其後,並在最後用< document >結束。 

3. 以Notepad.exe打開文件resume.xsl,在標記< HTML >之後添加文字:< xsl:for-each select="document" >;在標記< /HTML >之前添加文字:< /xsl:for-each >,保存文件。

4. 在瀏覽器中打開文件(個人簡歷.xml)。看到了什麼?兩份個人簡歷!

  就這樣,利用XML我們可以編寫內容與樣式完成分離的文檔!當然,XSL文件比一般的HTML文件要複雜一些,然而一旦完成則可用於格式化所有同類的XML文檔。

注:如果拷貝代碼,請將空格刪除 

跟我學XSL(三)  (返回標題) 

2000-05-05· 禹希初·CPCW

XSL模板與匹配模式

  經過前幾日的學習,我們學習了XHTML文檔的編寫和以及三個XSL元素,已能編寫相當靈活的XSL文檔,今天將學習的是XSL模板的編寫。我們都知道,短的文檔、程序十分好讀,但當規模增大後,其複雜性也以更快的速度增加。

  前面我們學了< xsl:for-each >、< xsl:value-of >等,可以用它們對XML數據實現簡單的格式化輸出,但如果遇到比較複雜的XML格式輸出,將XSL按照要求依次寫下來的話,一是設計困難,可擴展性差,不利於人員之間的分工協作;另則,可修改性很差,可能會出現牽一髮而動全軍的情況,不利於維護。程序中模塊化設計逐步細化的方法在這裏得到了應用! 

  XSL模板將XSL的設計細化成一個個模板(塊),最後再將這些模板(塊)組合成一個完整的XSL;好比船與集裝箱,我們不是將所有的貨物一件件地堆起來,而是裝在各自的集裝箱中,然後再在船上將這些集裝箱堆放起來。這種方法可以使你先從整體上考慮整個XSL的設計,然後將一些表現形式細化成不同的模塊,再具體設計這些模塊,最後將它們整合在一起,這樣,將宏觀與微觀結合起來,符合人們條理化、規範化要求。

裝集裝箱??書寫模板(塊):< xsl:template >

< xsl:template >

語法: 

< xsl:template match="node-context" language="language-name" >

屬性: 

match ── 確定什麼樣的情況下執行此模板。作爲一種簡化的說明,在此處使用標記的名字;其中最上層模板必須將match設爲“/”

language ── 確定在此模板中執行什麼腳本語言,其取值與HTML中的SCRIPT標記的LANGUAGE屬性的取值相同,缺省值是Jscript

< xsl:template >用match屬性從XML選取滿足條件的節點,徵對這些特定的節點形成一個特定輸出形式的模板。

吊集裝箱上船-??調用模板(塊):< xsl:apply-templates >

< xsl:apply-templates >

語法: 

< xsl:apply-templates select="pattern" order-by="sort-criteria-list" >

屬性: 

select ── 確定在此上下文環境中應執行什麼模板,即選取用< xsl:template >標記建立的模板(塊)。 

order-by ── 以分號(;)分隔的排序標準,通常是子標記的序列

示例:

  以個人簡歷爲例,爲便於處理我們希望“技能”中每一項都用標記對< skill >< /skill >括起來,有多少項技能就有多少個這種標記對,經過修改後的個人簡歷XML文檔內容如下: 

< ?xml version="1.0" encoding="GB2312"? >

< ?xml:stylesheet type="text/xsl" href="resume_template.xsl"? >

< document >

< resume >

< name >禹希初< /name >

< sex >男< /sex >

< birthday >1977.5< /birthday >

< skill >數據庫設計與維護< /skill >

< skill >WEB開發< /skill >

< /resume >

< /document >

  然後,建立一個新XSL文件resume_template.xsl,採用模板的形式,其內容如下: 

< ?xml version="1.0" encoding="GB2312"? >

< xsl:stylesheet xmlns:xsl="http://www.w3.org/TR/WD-xsl" >

< !--根模板-- >

< xsl:template match="/" >

< HTML >< HEAD >< TITLE >個人簡歷< /TITLE >< /HEAD >

< BODY >

< xsl:apply-templates select="document/resume"/ >

< /BODY >

< /HTML >

< /xsl:template >

< !--簡歷模板-- >

< xsl:template match="resume" >

< TABLE border="1" cellspacing="0" >

< CAPTION >個人簡歷( 

< xsl:eval >formatIndex(childNumber(this), "I")< /xsl:eval >

)< /CAPTION >

< xsl:apply-templates select="name" / >

< xsl:apply-templates select="sex" / >

< xsl:apply-templates select="birthday" / >

< TR/ >

< TD >技能< /TD >< TD COLSPAN="5" >

< TABLE cellspacing="0" >

< xsl:apply-templates select="skill"/ >

< /TABLE >

< /TD >

< /TABLE >

< BR/ >

< /xsl:template >

< !--姓名模板-- >

< xsl:template match="name" >< TD >姓名< /TD >

< TD >< xsl:value-of/ >< /TD >

< /xsl:template >

< !--性別模板-- >

< xsl:template match="sex" >< TD >性別< /TD >

< TD >< xsl:value-of/ >< /TD >

< /xsl:template >

< !--生日模板-- >

< xsl:template match="birthday" >< TD >生日< /TD >

< TD >< xsl:value-of/ >< /TD >

< /xsl:template >

< !--技能模板-- >

< xsl:template match="skill" >

< TR >< TD >< xsl:value-of/ >< /TD >< /TR >

< /xsl:template >

< /xsl:stylesheet >

  保存文件,打開文件(個人簡歷.xml),效果令人滿意吧。其實要做到同樣的效果,用前面三週介紹的方法也可做,但你得把它作爲一整體考慮。

  在上面的XSL文件中,我們將性別、生日、技能等數據項分別用模板來單獨寫,再用< xsl:apply-template >來調用,這樣,即使你日後要對這些模板作相應的修改與擴充也很方便,不致於出現互相干擾、混雜不清的情況。這種從上至下、逐層細化的設計方法,極大地減少工作複雜程度,也大大減少了差錯的產生,可以實現多人的協作設計。

注意:

  如果XML文檔中不同標記有同名的子標記,在爲其編寫模板時,應把父標記作爲其前綴,格式爲(parent_mark/child_mark)。

  模板文件必須有一個根模板,其屬性match是“/”。

注:如果拷貝源代碼,請將空格刪除 

跟我學XSL(四)  (返回標題) 

2000-05-06· 禹希初·CPCW

測試模式 

  XML技術的優勢之一就在於數據輸出的可選擇性,即選擇需要的數據輸出。前面我們所講到的選擇模式語句:<xsl:for-each>、<xsl:value-of>及<xsl:apply-template>只是簡單的選取通過”/”符號層層到達的節點,如果我們對XML數據不需要全部輸出,而只需要其中的滿足某條件的部分數據,“蘿蔔青菜、各取所需“,那麼條件判斷<xsl:if>與多條件判斷<xsl:choose>及<xsl:when>則迎合了這種需要,如果你對程序設計熟悉的話,會覺得它們似曾相識。 

XSL中的IF,首先,介紹XSL元素<xsl:if>的語法結構: 

<xsl:if> 

語法: 

<xsl:if expr="script-expression" language="language-name" test="pattern"> 

屬性: 

  expr ──腳本語言表達式,計算結果爲“真”或“假”;如果結果爲“真”,且通過test,則在輸出中顯示其中內容(可省略此項屬性)。 

  language ──expr屬性中表達式的腳本語言類型,其取值與HTML標記SCRIPT的LANGUAGE屬性的取值相同,缺省爲“JScript”test ──源數據測試條件。 

示例: 

此處以一份報表爲例,文件名爲report.xml,其內容如下: 

<?xml version="1.0" encoding="GB2312"?> 

<?xml:stylesheet type="text/xsl" href="report.xsl"?> 

<document> 

<report> 

<class> 

甲班 

</class><q1>50</q1><q2>70</q2> 

<q3>30</q3><q4>10</q4></report> 

<report><class> 

乙班 

</class><q1>20</q1><q2>30</q2> 

<q3>40</q3><q4>50</q4></report> 

<report><class> 

丙班 

</class><q1>70</q1><q2>40</q2> 

<q3>20</q3><q4>10</q4></report> 

</document> 

  我們採用XSL模板結合今天所學的<xsl:if>,爲其編寫一個XSL文檔,要求季度產量小於等於20的用紅色表示,文件名爲report.xsl,內容如下: 

<?xml version="1.0" encoding="GB2312"?> 

<xsl:stylesheet xmlns:xsl="http://www.w3.org/TR/WD-xsl"> 

<xsl:template match="/"> 

<HTML><HEAD><TITLE>1999年生產統計</TITLE></HEAD> 

<BODY><xsl:apply-templates select="document ;"/></BODY> 

</HTML> 

</xsl:template> 

<xsl:template match="document"> 

<H3>1999年生產統計</H3> 

<TABLE border="1" cellspacing="0"> 

<TH>班組</TH><TH> 

一季度 

</TH><TH> 

二季度 

</TH> 

<TH> 

三季度 

</TH><TH> 

四季度 

</TH> 

<xsl:apply-templates select="report"/> 

</TABLE> 

</xsl:template> 

<xsl:template match="report"> 

<TR> 

<TD><xsl:value-of select="class"/></TD> 

<TD><xsl:apply-templates select="q1"/></TD> 

<TD><xsl:apply-templates select="q2"/></TD> 

<TD><xsl:apply-templates select="q3"/></TD> 

<TD><xsl:apply-templates select="q4"/></TD> 

</TR> 

</xsl:template> 

<xsl:template match="q1|q2|q3|q4"> 

<!--此處測試產量,如小於等於20則添加一STYLE屬性color,其值爲red(紅色)--> 

<xsl:if test=".[value() $le$ 20]"> 

<xsl:attribute name="style">color:red</xsl:attribute> 

</xsl:if> 

<xsl:value-of/> 

</xsl:template> 

</xsl:stylesheet> 

說明: 

q1|q2|q3|q4 ──標記q1、q2、q3、q3均用此模板確定輸出 

$le$ ──是關係運算符中的“小於等於”,其它關係有小於($lt$)、大於($gt$)、大於等於($ge$)、等於($eq$)、不等於($ne$)等 

. ──表示引用當前標記 

[ ] ──表示篩選,只有滿足篩選條件的標記才能被選取 

value() ──XSL函數,其他常用XSL函數有text()、end()、index()等。 

  下期,我們將學習XSL的另外三個元素,可對同一數據進行多次測試,根據不同條件產生相應輸出。 

跟我學XSL(五)  (返回標題) 

2000-05-07· 禹希初·CPCW

XSL中的choose

  上期我們學習了XSL元素< xsl:if >,已能通過測試XML數據的值來決定不同的輸出形式(不知你嘗試過沒有,實際上< xsl:for-each >也可部分實現< xsl:if >的功能),但有時,我們希望對同一數據同時測試多個條件,根據不同條件輸出相應結果。當然,我們可以用if,假如我們只有if可用的話。幸好我們有一個更好的選擇,那就是用< xsl:choose >。下面介紹相關元素的語法: 

< xsl:choose >

語法:< xsl:choose >

屬性:無,表示一個多選測試的開始 

< xsl:when >

語法: 

< xsl:when expr="script-expression" language="language-name" test="pattern" >

屬性: 

expr ── 腳本語言表達式,計算結果爲“真”或“假”;如果結果爲“真”,且通過test,則在輸出中顯示其中內容(可省略此項屬性)。 

language ── expr屬性中表達式的腳本語言類型,其取值與HTML標記SCRIPT的LANGUAGE屬性的取值相同,缺省爲“JScript”。 

test ── 源數據測試條件。 

< xsl:otherwise >

語法:< xsl:otherwise >

屬性:無,在一個多選測試中,如果沒有不滿足< xsl:when >規定的條件,如果在最後有此標記,則輸出此標記中的內容。

示例:

  此處以學生成績單爲例,要求按成績的高低給出優秀( >85)、一般(70~85)、起格(60~69)、不及格(< 60),而不是顯示分數。其中成績單的XML文檔(文件名:grade.xml)如下: 

< ?xml version="1.0" encoding="GB2312"? >

< ?xml:stylesheet type="text/xsl" href="grade.xsl"? >

< document >

< grade >

< name >大胖< /name >< english >80< /english >

< math >90< /math >< chymest >90< /chymest >

< /grade >

< grade >

< name >小花< /name >< english >98< /english >

< math >70< /math >< chymest >85< /chymest >

< /grade >

< /document >

  爲實現按分數分等級顯示,其XSL文檔(文件名:grade.xsl)內容如下: 

< ?xml version="1.0" encoding="GB2312"? >

< xsl:stylesheet xmlns:xsl="http://www.w3.org/TR/WD-xsl" >

< xsl:template match="/" >

< HTML >

< HEAD >< TITLE >成績單< /TITLE >< /HEAD >

< BODY >< xsl:apply-templates select="document"/ >< /BODY >

< /HTML >

< /xsl:template >

< xsl:template match="document" >

< TABLE border="1" cellspacing="0" >

< TH >姓名< /TH > < TH >英語< /TH >< TH >數學< /TH >< TH >化學< /TH >

< xsl:apply-templates select="grade"/ >

< /TABLE >

< /xsl:template >

< xsl:template match="grade" >

< TR >

< TD >< xsl:apply-templates select="name"/ >< /TD >

< TD >< xsl:apply-templates select="english"/ >< /TD >

< TD >< xsl:apply-templates select="math"/ >< /TD >

< TD >< xsl:apply-templates select="chymest"/ >< /TD >

< /TR >

< /xsl:template >

< xsl:template match="name" >< xsl:value-of/ >< /xsl:template >

< xsl:template match="english|math|chymest" >

< xsl:choose >

< xsl:when test=".[value() $gt$ 85]" >優秀< /xsl:when >

< xsl:when test=".[value() $gt$ 70]" >一般< /xsl:when >

< xsl:when test=".[value() $gt$ 60]" >起格< /xsl:when >

< xsl:otherwise >不起格< /xsl:otherwise >

< /xsl:choose >

< /xsl:template >

< /xsl:stylesheet >

說明:

  在< xsl:choose >選擇中,從第一個< xsl:when >開始,逐個測試,直到滿足一個測試條件就將其中的內容輸出,不再測試後面的條件;如果不滿足任何一個條件,則輸出< xsl:otherwise >中的內容。

  標記對< xsl:when >< /xsl:when >與< xsl:otherwise >< /xsl:otherwise >中可嵌套< xsl:if >或< xsl:choose >。

注:如果拷貝源代碼,請將空格刪除

跟我學XSL(六)  (返回標題) 

2000-05-08· 禹希初·CPCW

運算符、關係與邏輯運算符,特殊字符等

  今天我們學習XSL的各種運算符,滿足編寫更大靈活性的XSL樣式的要求。

表一、運算符與特殊字符

運算符 描述 

/ 選擇子元素,返回左側元素的直接子元素;如果“/”位於最左側表示選擇根結點的直接子元素 

// 遞歸下降,不論深度,搜索指定的元素;如果位於最左側表示從根結點出發遞歸下降搜索指定元素 

. 表示當前元素 

* 通配符,選擇任意元素,不考慮名字 

@ 取得屬性值,作爲屬性名的前綴 

@* 通配符,選擇任意屬性,不考慮名字 

: 名字作用範圍分隔符,將名字作用範圍前綴與元素或屬性名分隔開來 

!* 在相關節點上應用指定方法 

( )* 分組,明確指定優先順序 

[ ] 應用過濾樣式 

[ ]* 下標運算符,用於在集合中指示元素 

表二、邏輯運算符

比較運算符 可選方式 描述 

and $and$或 邏輯與 

or $or$或|| 邏輯或 

not() $not$ 邏輯非 

表三、關係運算符 

= $eq$ 相等 

= $ieq$ 相等(不區分大小寫) 

!= $ne$ 不等 

$ine$ 不等(不區分大小寫) 

< $lt$ 小於 

$ilt$ 小於(不區分大小寫) 

< = $le$ 小於等於 

$ile$ 小於等於(不區分大小寫) 

> $gt$ 大於 

$igt$ 大於(不區分大小寫) 

>= $ge$ 大於等於 

$ige$ 大於等於(不區分大小寫) 

$all$ 集合運算符,如果集合中所有項目均滿足條件則返回“真” 

$any$ 集合運算符,如果集合中任意項目滿足條件則返回“真” 

| 集合運算符,返回兩個集合的聯合 

示例一:

  從個人簡歷中尋找具有具有“WEB開發”技能的人的姓名與E-Mail。假設文檔結構如下所示: 

< document >

< resume >

< name >name< /name >

< sex >sex< /sex >

< birthday >birthday< /birthday >

< skill >skill1< /skill >

< skill >skill2< /skill2 >

… 

< skill >skilln< /skill >

< /resume >

< resume >

… 

< /resume >

… 

< /document >

  爲從以上結構的個人簡歷中尋找出所有具有“WEB開發”技能的人的姓名與E-Mail的XSL文檔結構如下: 

< TABLE border="1" cellspacing="0" >

< TH >姓名< /TH >< TH >E-Mail< /TH >

< xsl:for-each select="resume [ $any$ skill='WEB開發' ]" >

< TR >< TD >< xsl:value-of select="name"/ >< /TD >

< TD >< xsl:value-of select="E-Mail"/ >< /TD >

< /TR >

< /xsl:for-each >

< /TABLE >

說明: 

1.[ ] ───表示選擇條件,只有滿足條件的個人簡歷才被顯示; 

2.$any$ ──由於每個人有多種技能,故加$any$作爲前綴,以使每個人所有技能都能被比較; 

3.skill='WEB開發' ──篩選條件

示例二、

  仍上面的XML文檔爲例,如果欲選擇1977/1/1之前出生的人的姓名、技能與E-Mail,相應的XSL文檔結構如下(假定生日格式爲yyyy/mm/dd): 

< TABLE border="1" cellspacing="0" >

< TH >姓名< /TH >< TH >技能< /TH >< TH >E-Mail< /TH >

< xsl:for-each select="resume [ birthday $lt$ '1977/1/1' ]" >

< TR >

< TD >< xsl:value-of select="name"/ >< /TD >

< TD >< xsl:value-of select="skill [0]"/ >

< xsl:for-each select="skill [index() >0]" >、 

< xsl:value-of select="."/ >

< /xsl:for-each >

< /TD >

< TD >< xsl:value-of select="E-Mail"/ >< /TD >

< /TR >

< /xsl:for-each >

< /TABLE >

說明: 

1. birthday $lt$ '1977/1/1' ── 搜索條件,在此處使用“< ”會錯誤,故使用“$lt$”表示小於。 

2. skill [0] ─── 表示選擇skill的第一項 

3. skill [index() >0]───表示選擇skill的第二項以後(包括第二項)的項目 

4. xsl:value-of select="." ─── 表示選擇當前標記的值

  相信大家應該注意到,在前面以及本次的例子中出現了一些函數,如index()、formatIndex()、childNumber(),也許大家還不完全明瞭其中的含義吧?敬請關注下一課。

注:如果拷貝源代碼,請將空格刪除 

跟我學XSL(七)  (返回標題) 

2000-05-09· 禹希初·CPCW

XSL函數一

  本期學習XSL樣式方法,即可用於XSL元素< xsl:for-each >、< xsl:value-of >、< xsl:template >的select屬性、< xsl:apply-templates >的match屬性、< xsl:if >、< xsl:when >的test屬性中,對元素的範圍進行篩選,從而提供更大的靈活性。

  XML與DHTML(動態HTML)一樣,這些節點都是一個個對象,而且這些對象都是有層次的,從根節點開始構成一顆層次清淅的樹狀結構,這就形成了文檔對象模型DOM,通過對象的屬性、方法來達到訪問控制XML節點的目的。

  我們這裏不打算就XML的DOM逐一詳細闡述,因爲這完全可以寫成一個篇幅較多的教程,我們先就一些常見的方法作一些討論,以期對DOM的對象方法有一個大致的瞭解。

  注:從本期開始,所有示例不再提供完整源代碼,如有不明白之處,請仔細閱讀前面七期、並動手練手。

一、end()

含義:返回集合中最後一個元素。

示例:輸出最後一份簡歷

假定XML文件格式爲:

……< resume >…< /resume >……< resume >…< /resume >……

相應XSL文件內容爲: 

< xsl:for-each select="resume[end()]" >……< /xsl:for-each >

或 

< xsl:templates match="resume[end()]" >……< /xsl:templates >

或 

< xsl:apply-template select="resume[end()]" >……< /xsl:apply-template >

二、index()

含義:返回該元素在集合中的位置,返回值是一整數,其中第一個元素返回0

示例:返回前面三份簡歷 

resume[index() $le$ 3]

注意:index()是與父元素相關的,請看下例: 

< x >

< y/ >

< y/ >

< /x >

< x >

< y/ >

< y/ >

< /x >

返回所有< x >中的第一個< y >

x/y[index()=0] 或x/y[0]

三、nodeName()

含義:返回元素的名字,即標記名

示例:選擇任意元素,假如其名字(即標記名)等於“name”

*[nodeName()='name'] 或 *[name]

四、number()

含義:將值轉換爲數值形式,如果不是數值則返回空,要求參數

示例:年齡(age)小於30歲的人的簡歷(resume)

resume[number(age) $lt$ 30] 或resume[age $lt$ 30]

五、nodeType()

含義:返回結點類型,結果爲是數值。以下是返回值列表:

結點類型 結點類型值 結點的字符形式描述 

Element 1 'element' 

Element Attribute 2 'attribute' 

Markup-Delimited Region of Text 3 'text' 

Processing Instruction 7 'processing_instruction' 

Comment 8 'comment' 

Document Entity 9 'document' 

六、value()

含義:返回元素或屬性的值

示例:value()是元素或屬性的缺省方法,以下表示是等價 

name!value()="NAME"與name="NAME"

@attr="attribute_value"與@attr="attribute_value"

注:@是屬性前綴,@attr表示是屬性attr

七、attribute()

含義:返回所有屬性結點的集合,等價於“@*”

示例:尋找所有的resume元素,滿足條件至少有一個屬性的值爲“ABC”

resume[$any$ attribute()='ABC']或resume[$any$ @*='ABC']

尋找所有的resume元素,滿足條件至少有一個子元素有一個屬性的值爲“ABC”

resume[$any$ */attribute()='ABC']或resume[$any$ */@*='ABC']

八、comment()

含義:返回所有註釋結點 

示例:例如 

resume[$any$ comment()='禹希初的簡歷']

表示尋找含有註釋語句 

< !--禹希初的簡歷-- >

的元素< resume >

九、cdata()

含義:返回所有CDATA類型的結點的集合

示例:例如 

resume[$any$ cdata()='禹希初的簡歷']

表示尋找含有下述語句(必須是直接子結點) 

< ![CDATA[禹希初的簡歷]] >

的元素< resume >

十、node()

含義:返回當前上下文環境中除根結點和屬性結點以外的所有結點的集合,等價於

“* | pi() | comment() | text()”

示例:尋找所有元素resume,其最後一個結點的名字爲"skill"

resume[node()[end()]!nodeName()='skill']

尋找所有resume元素的第一個結點:resume/node()[0]

十一、textnode()

含義:返回所有文本類型的結點的集合

示例:尋找每一個p元素的第二個文本結點 

p/textnode(1)或p!textnode(1)

十二、text()

含義:返回所有表示文本字符串的結點的集合,等價於"cdata()|textnode()";

  本期的內容就介紹至此,另有一個函數date()在本人的機器上一試就發生錯誤使瀏覽器自動關閉,還有一個函數pi()本人尚未找到適當的應用方法,就不介紹了,下期將講述如何XSL中使用腳本。

跟我學XSL(八)  (返回標題) 

2000-05-10· 禹希初·CPCW

腳本與XSL的結合及應用

有時,我們可能會希望XML文檔輸出時能對其中內容加上一些統計信息或者如編號什麼,利用前面的知識就不太容易實現了。今天將介紹兩個新元素< xsl:eval >與< xsl:script >,使我們能輕鬆處理這個難題。 

< xsl:eval >

含義: 

計算腳本表達式,輸出一個文本字符串 

語法: 

< xsl:eval language="language-name" >

屬性: 

language ── 規定所用腳本語言的名字,可用的屬性有“JavaScript”、“JScript”、“VBScript”、“VBS”等,缺省爲“JScript”。 

< xsl:script >

含義: 

聲明全局變量或定義函數。 

語法: 

< xsl:script language="language-name" >

屬性:

同< xsl:eval >

示例:

  不知大家對於第四期《跟我學XML》中的例子是否還有印象?其中的XML文檔並沒有對簡歷編號,但輸出中卻加上了大寫的羅馬數字序號。今天將再舉一稍爲複雜一些的例子:

  假如我們編寫一份年終生產統計表,其中需要小計一項,常規的作法是事先將其算出來,現在不必了,我們可以只給出單項統計,顯示時再統計小計一項。請找出《跟我學XML》的第五期,XML文件不必修改,對XSL文件的修改如下: 

…… 

< xsl:template match="document" >

…… 

< xsl:apply-templates select="report"/ >

< TR >< TD >小計< /TD >

< TD >< xsl:eval >total(this,"q1")< /xsl:eval >< /TD >

< TD >< xsl:eval >total(this,"q2")< /xsl:eval >< /TD >

< TD >< xsl:eval >total(this,"q3")< /xsl:eval >< /TD >

< TD >< xsl:eval >total(this,"q4")< /xsl:eval >< /TD >

< /TR >

< /TABLE >

< xsl:script >

function total(node,q){

temp=0; 

mark='/document/report/'+q;

v=node.selectNodes(mark);

for(t=v.nextNode();t;t=v.nextNode()){

temp+=Number(t.text);

}

return temp; //小計值 

< /xsl:script >

< /xsl:template >

說明:

  黑體部分爲添加部分,注意添加部分分爲兩部分,< xsl:script >< /xsl:script >必須置於< /TABLE >之後,切記。 

selectNodes() ── 是XMLDOMObject的一個方法,返回文檔中所有滿足條件的結點的集合,條件與< xsl:for-each >和select屬性的取值採用同樣的寫法,可以加篩選、下標等限制,如:

尋找一季度產量大於等於50的班組 

/document/report/q1[value() $ge$ 50]

以上寫法還有一個更簡單的寫法 

//q1[value() $ge$ 50]

//表示從根結點出發遍歷所有結點,尋找滿足條件的結點,如果文檔內有同名但意義不同的結點則不能用此種方法,非不得已不建議使用。以此爲例,如果希望統計年總產量,則可以下述字符串尋找結點(建議使用最後一種,此種描述將精確找到需要彙總的數據) 

//*[value() $gt$ 0]

或 

//(q1|q2|q3|q4)

或 

/document/report/(q1|q2|q3|q4)

nextNode() ── 返回結點集中的下一個結點 

Number() ── 將提供的參數轉換爲數值

  下期介紹XSL函數2,用於< xsl:script >及< xsl:eval >中,以及< xsl:if >和< xsl:when >的expr屬性。建議讀者熟悉JavaScript、JScript、VBScript中至少一種,否你能用XSL完成的工作將十分有限。由於篇幅關係,此處不作詳細介紹。 

注:如果拷貝源代碼,請將空格刪除 

XLink介紹 

2000-03-02· 蔡依純 譯·中文XML

  如果在這個世界上每部電腦中的每個資料塊都能夠被識別、定址及連結,那將是多麼令人振奮的事呢!Xink就是說明如何在網絡上做到這點的規格文件。

  HTML的連結

  由於WWW的風行,大部分的人都熟悉超文字連結的基本觀念,亦即是“一個連結具有兩個端點及一個方向,兩個端點我們通常稱爲Anchor。這個連結可以從一個起點連到任何一個網路資源的目標點,這些連結可能是一張圖片,一卷影片,一段聲音,一個程式,一篇HTML的文件,或其中的元素等等”。

  假如你常使用瀏覽器,你一定會知道,當你在連結上按滑鼠的右鍵時,就會出現一個選單,而其中一個選擇是“在新視窗開啓”。

  在HTML中,對於這些連結都是用一個「Anchor」的標誌所做成。

  如果你是撰寫HTML的高手,你一定知道在head標誌裏有個特別的標誌叫做link,它提供了多種的連結功能:比方像是連結到樣式表(新的瀏覽器會提供使用者多一點選擇,讓其自行選擇使用哪一個樣式表),或是連結到有音樂的地方(當下載完畢後可自動地播放),抑或是連結到另一個網頁(可使瀏覽器預先載入該網頁,以省去等待的時間)等。

  以上所描述的都是連結的功能,事實上,有很多其他的超文字連結功能是HTML沒有辦法做到的,即使如早期(1960年末至1970年初期)的超文字連結系統所提供的連結功能,HTML亦沒辦法做到。然而,HTML之所以會成功,是因爲它使用最簡單的方式去使用超連結,而且使用者並不 要特別的編輯器(Editor)便可製作網頁。

  多用途的連結!

  有時當我們看到「連結」時,我們發現其實它看起並不像是連結,但事實上,爲什麼呢?許多的事物包括電子化地址、識別字、位址或是查詢等,都可以有連結這個動作。當你想到兩個資料間有“relationship”、“role”、“pointer”或是“has a”等關聯性時,在XML(eXtensibleMarkupLanguage)中我們可以用「連結」的方法來加以標示。

  讀者們應有一個觀念,在HTML4.0規格中,所定義的超文字連結標準,並不是就是連結的所有特性,所以在這裏我們將對連結做更清楚的定義:

  1.一個連結可以有一個、二個甚至是很多個ends,而每一個end都可以是指標,事實上連結本身可以很明確地指定是外部(externally)連結,因而連接到每一個end。

  2.一個連結並不 一定要有方向,但它也可以有很多或是沒有方向,除此之外,連結的方向不 要一定得從“context”到“part”才行。

  3.連結的end可以是某一範圍,不一定是要連結到整個resource或某一地方。

  4.連結點的表現方式乃視連結資料的角色而定,「擷取」和「取代」只是可能性的行爲之一。是故,使用者設定連結爲擷取資料並取代畫面只是連結功能的表現之一。

  XML連結對我們有什麼用呢?簡言之,像是資料模型化、資料交換、超文字連結或是任何不同關係的資料型態上,都可以派上用場。

  XLinks有一重要功能就是建立「topicmaps」,這是一種依據metadata連結到種種不同網路資源的方式。Topicmaps允許不同的資料有外在的註解(External Annotation)。因此,我們可以說Topicmaps是有結構性的metadata,而依據各特性關聯主題,可以連結到不同的網路資源。

  XLinks

XLink定義了幾種常用的連結型態:Simple、Extended、Group和Document。 

1.Simple的用法比較接近在HTML內a標誌的用法(如下HTML的寫法所示)。 

2.Extended的用法包含arc和locator的元素,並允許各種種類的擴充連結。 

3.Group和document的用法,是讓羣組連結到一些特別的文件。

以下我們將說明Simple的語法,有兩種方法可以知道一個連結是否是Xlink: 

1.直接使用simple字眼; 

2.以xlink:type表示。

讓我們來看看下面Simple的例子: 

1.以HTML的寫法如下: 

2.以Xlink的寫法:

  同樣一個連結,另一種Xlink也可以寫成:

  xmlns:xlink屬性是一個名稱 域(Namespaces)宣告的例子,而所謂的名稱 域,是方便使用者同時使用數個文件型別宣告(Document TypeDeclaration)之機制,至於這樣的做法,我們相信在未來之HTML版本中,也會朝這一方向努力。

  也許我們覺得很奇怪,爲什麼這些arcs會接在一起,而形成extendedlink呢?有些人可能比較喜歡(圖二)的寫法,因爲這種連結並沒有“from”和“to”,只有href的地址。

  在locator的定義中包含兩個原性:href和role。href屬性包含URL,role屬性則包含了如何去使用連結的關鍵字。

  Xlink擴展了URIs的使用,其中actuate和show對於Xlink來說是非常重要的屬性,原因如下:

  1.actuate的屬性可以說明Link是自動地或人工地(user-command)追蹤。

  2.show的屬性說明另一端的連結是 要顯示一個新(new)的視窗,或是取代(replace)成目前的視窗,或是 要剖析的(若該連結連結到XML文件)。

  對於HTML的高手來說,上述href屬性的用法較不平常,這是因爲含有查詢條件的正確URL,而它的主要目的是查詢文件。

  這一種URL使用了一個很特別的語法,就是在URL之前放問號(?)來查詢,許多指向資料庫的連結都用這個技巧。

  W3C正在發展中的Xpointer是一種特殊的查詢,也是Xlink推展的一部份。Xpointer的語法是?xptr=加上XML PathExpression【像是id(phone2)】。從上面的例子中看來,id指到唯一識別字phone2。(這個跟xlink:arc的from和to的用意一樣)。 

XMLPathExpression可讓我們以不同種類的標準來選擇某一元素樹狀結構式(Elementtreeofstructure data)特殊分枝。

 


[此貼子已經被作者於2004-4-7 18:24:33編輯過]
<iframe name="google_ads_frame" marginwidth="0" marginheight="0" src="http://pagead2.googlesyndication.com/pagead/ads?client=ca-pub-1683803802533351&dt=1175594866212&lmt=1175594866&alt_color=FFFFFF&prev_fmts=336x280_as%2C728x15_0ads_al&format=468x60_as&output=html&channel=0016543752&url=http%3A%2F%2Fbbs.xml.org.cn%2Fdispbbs.asp%3FboardID%3D8%26ID%3D6411&color_bg=FFFFFF&color_text=000000&color_link=000000&color_url=0066FF&color_border=0099FF&ad_type=image&ref=http%3A%2F%2Fbbs.xml.org.cn%2Fdispbbs.asp%3FboardID%3D8%26ID%3D19725&cc=2&u_h=768&u_w=1024&u_ah=738&u_aw=1024&u_cd=16&u_tz=480&u_his=2&u_java=true" frameborder="0" width="468" scrolling="no" height="60" allowtransparency="allowtransparency"></iframe>

----------------------------------------------
<?xml version="1.0" encoding="gb2312"?>
<個人簽名>
  <Website inf="提供下載">
www.koolle.com
</Website>
<Email>aloning(at)gmail.com</Email>
  <Qq >29519680</Qq>
</個人簽名>

 
發佈了55 篇原創文章 · 獲贊 1 · 訪問量 8萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章