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)特殊分枝。
----------------------------------------------
<?xml version="1.0" encoding="gb2312"?>
<個人簽名>
<Website inf="提供下載">
www.koolle.com
</Website>
<Email>aloning(at)gmail.com</Email>
<Qq >29519680</Qq>
</個人簽名>