活动公告

系统通知
05-18 21:22
系统通知
通知:本站资源由网友上传分享,如有违规等问题请到版务模块进行投诉,资源失效请在帖子内回复要求补档,会尽快处理!
10-23 09:31

XQuery实例解析 深入理解XML查询语言的强大功能从简单查询到复杂数据处理通过实际案例全面提升你的编程技能

SunJu_FaceMall

3万

主题

2860

科技点

3万

积分

白金月票

碾压王

积分
32872

塔罗立华奏

<font color=白金月票" /> 发表于 2025-9-6 00:40:01 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

x
引言

XQuery是一种用于查询XML数据的函数式编程语言,作为W3C推荐的标准,它提供了强大的功能来处理半结构化数据。无论是简单的数据检索还是复杂的数据转换,XQuery都能提供高效而优雅的解决方案。本文将通过一系列由浅入深的实例,详细解析XQuery的各种功能和用法,帮助读者从基础到高级全面掌握这门强大的查询语言。

XQuery基础

XQuery数据模型

XQuery基于XDM(XQuery and XPath Data Model),该模型将XML文档视为节点树。主要节点类型包括:

• 元素节点
• 属性节点
• 文本节点
• 命名空间节点
• 处理指令节点
• 注释节点
• 文档节点

基本语法

XQuery的语法类似于SQL,但更加灵活。一个基本的XQuery查询由以下部分组成:
  1. (: 这是一个XQuery注释 :)
  2. for $variable in expression
  3. where condition
  4. order by expression
  5. return expression
复制代码

示例XML文档

下面我们将使用一个示例XML文档,它包含一个简单的书籍集合:
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <bookstore>
  3.   <book category="FICTION">
  4.     <title lang="en">The Great Gatsby</title>
  5.     <author>F. Scott Fitzgerald</author>
  6.     <year>1925</year>
  7.     <price>10.99</price>
  8.   </book>
  9.   <book category="SCIENCE">
  10.     <title lang="en">A Brief History of Time</title>
  11.     <author>Stephen Hawking</author>
  12.     <year>1988</year>
  13.     <price>14.99</price>
  14.   </book>
  15.   <book category="FICTION">
  16.     <title lang="fr">L'Étranger</title>
  17.     <author>Albert Camus</author>
  18.     <year>1942</year>
  19.     <price>12.50</price>
  20.   </book>
  21.   <book category="TECHNOLOGY">
  22.     <title lang="en">Clean Code</title>
  23.     <author>Robert C. Martin</author>
  24.     <year>2008</year>
  25.     <price>35.99</price>
  26.   </book>
  27. </bookstore>
复制代码

简单查询实例

基本选择

示例1:选择整个文档
  1. (: 选择整个XML文档 :)
  2. doc("bookstore.xml")
复制代码

这个查询返回整个XML文档。在实际应用中,你可能需要指定文件的完整路径。

示例2:选择所有书籍
  1. (: 选择所有书籍元素 :)
  2. doc("bookstore.xml")/bookstore/book
复制代码

这个查询使用XPath表达式/bookstore/book来选择所有book元素。

示例3:选择所有书名
  1. (: 选择所有书名 :)
  2. doc("bookstore.xml")/bookstore/book/title
复制代码

这个查询返回所有title元素。

路径表达式和谓词

示例4:使用谓词过滤
  1. (: 选择价格大于15的书籍 :)
  2. doc("bookstore.xml")/bookstore/book[price > 15]
复制代码

这个查询使用谓词[price > 15]来过滤价格大于15的书籍。

示例5:使用属性过滤
  1. (: 选择类别为FICTION的书籍 :)
  2. doc("bookstore.xml")/bookstore/book[@category = "FICTION"]
复制代码

这个查询使用属性谓词[@category = "FICTION"]来选择类别为FICTION的书籍。

条件表达式

示例6:使用条件表达式
  1. (: 根据价格显示不同的消息 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. return
  4.   if ($book/price > 15) then
  5.     <expensive>{concat($book/title, " is expensive.")}</expensive>
  6.   else
  7.     <affordable>{concat($book/title, " is affordable.")}</affordable>
复制代码

这个查询根据书籍的价格返回不同的消息。

排序结果

示例7:按价格排序书籍
  1. (: 按价格升序排列书籍 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. order by $book/price
  4. return $book
复制代码

这个查询按价格升序排列所有书籍。

示例8:多条件排序
  1. (: 先按类别,再按价格排序书籍 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. order by $book/@category, $book/price
  4. return $book
复制代码

这个查询先按类别,再按价格排序书籍。

中级查询实例

FLWOR表达式

FLWOR(For, Let, Where, Order by, Return)是XQuery的核心构造,类似于SQL中的SELECT-FROM-WHERE。

示例9:使用完整的FLWOR表达式
  1. (: 使用FLWOR表达式选择特定类别的书籍并按价格排序 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. let $title := $book/title
  4. where $book/@category = "FICTION" and $book/price < 15
  5. order by $book/price
  6. return
  7.   <book>
  8.     {$title}
  9.     <price>{$book/price}</price>
  10.   </book>
复制代码

这个查询选择类别为FICTION且价格小于15的书籍,按价格排序,并返回只包含标题和价格的新book元素。

连接操作

示例10:使用两个XML文档进行连接

假设我们有另一个XML文档authors.xml:
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <authors>
  3.   <author>
  4.     <name>F. Scott Fitzgerald</name>
  5.     <country>USA</country>
  6.   </author>
  7.   <author>
  8.     <name>Stephen Hawking</name>
  9.     <country>UK</country>
  10.   </author>
  11.   <author>
  12.     <name>Albert Camus</name>
  13.     <country>France</country>
  14.   </author>
  15.   <author>
  16.     <name>Robert C. Martin</name>
  17.     <country>USA</country>
  18.   </author>
  19. </authors>
复制代码

我们可以执行连接操作:
  1. (: 连接书籍和作者信息 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. for $author in doc("authors.xml")/authors/author
  4. where $book/author = $author/name
  5. return
  6.   <bookWithCountry>
  7.     {$book/title}
  8.     <author>{$book/author}</author>
  9.     <country>{$author/country}</country>
  10.   </bookWithCountry>
复制代码

这个查询将书籍与作者的国家信息连接起来。

聚合函数

示例11:计算平均价格
  1. (: 计算所有书籍的平均价格 :)
  2. let $books := doc("bookstore.xml")/bookstore/book
  3. return
  4.   <statistics>
  5.     <averagePrice>{avg($books/price)}</averagePrice>
  6.     <totalBooks>{count($books)}</totalBooks>
  7.     <minPrice>{min($books/price)}</minPrice>
  8.     <maxPrice>{max($books/price)}</maxPrice>
  9.   </statistics>
复制代码

这个查询计算书籍的平均价格、总数、最低价格和最高价格。

示例12:按类别分组统计
  1. (: 按类别统计书籍数量和平均价格 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. group by $category := $book/@category
  4. return
  5.   <categoryStats category="{$category}">
  6.     <count>{count($book)}</count>
  7.     <avgPrice>{avg($book/price)}</avgPrice>
  8.   </categoryStats>
复制代码

这个查询按类别分组,统计每个类别的书籍数量和平均价格。

量词表达式

示例13:使用some量词
  1. (: 检查是否有价格超过30的书籍 :)
  2. if (some $book in doc("bookstore.xml")/bookstore/book satisfies $book/price > 30) then
  3.   <result>There are expensive books.</result>
  4. else
  5.   <result>All books are affordable.</result>
复制代码

这个查询检查是否存在价格超过30的书籍。

示例14:使用every量词
  1. (: 检查所有书籍是否都有价格信息 :)
  2. if (every $book in doc("bookstore.xml")/bookstore/book exists $book/price) then
  3.   <result>All books have price information.</result>
  4. else
  5.   <result>Some books are missing price information.</result>
复制代码

这个查询检查所有书籍是否都有价格信息。

高级查询实例

序列操作

示例15:序列组合和排序
  1. (: 创建一个包含所有价格和年份的混合序列,并排序 :)
  2. let $prices := doc("bookstore.xml")/bookstore/book/price
  3. let $years := doc("bookstore.xml")/bookstore/book/year
  4. let $combined := ($prices, $years)
  5. order by $combined
  6. return
  7.   <sortedValues>
  8.     {for $value in $combined return <value>{$value}</value>}
  9.   </sortedValues>
复制代码

这个查询将所有价格和年份组合成一个序列,并排序。

示例16:序列过滤
  1. (: 过滤出价格在10到20之间的书籍价格 :)
  2. let $prices := doc("bookstore.xml")/bookstore/book/price
  3. return
  4.   <filteredPrices>
  5.     {for $price in $prices[. >= 10 and . <= 20] return <price>{$price}</price>}
  6.   </filteredPrices>
复制代码

这个查询过滤出价格在10到20之间的书籍价格。

节点构造

示例17:构造复杂XML结构
  1. (: 构造一个包含书籍信息的复杂XML结构 :)
  2. <bookReport>
  3.   <generated>{current-dateTime()}</generated>
  4.   <books>
  5.     {
  6.       for $book in doc("bookstore.xml")/bookstore/book
  7.       return
  8.         <book category="{$book/@category}">
  9.           <title>{$book/title/text()}</title>
  10.           <author>{$book/author/text()}</author>
  11.           <year>{$book/year/text()}</year>
  12.           <price currency="USD">{$book/price/text()}</price>
  13.         </book>
  14.     }
  15.   </books>
  16.   <summary>
  17.     <totalBooks>{count(doc("bookstore.xml")/bookstore/book)}</totalBooks>
  18.     <totalValue>{sum(doc("bookstore.xml")/bookstore/book/price)}</totalValue>
  19.   </summary>
  20. </bookReport>
复制代码

这个查询构造了一个包含所有书籍信息和汇总数据的复杂XML结构。

用户定义函数

示例18:定义和使用函数
  1. (: 定义一个计算折扣价的函数 :)
  2. declare function local:discountPrice($price as xs:decimal, $discountRate as xs:decimal) as xs:decimal {
  3.   $price * (1 - $discountRate)
  4. };
  5. (: 使用函数计算所有书籍的折扣价 :)
  6. for $book in doc("bookstore.xml")/bookstore/book
  7. return
  8.   <book>
  9.     <title>{$book/title/text()}</title>
  10.     <originalPrice>{$book/price/text()}</originalPrice>
  11.     <discountedPrice>{local:discountPrice($book/price, 0.1)}</discountedPrice>
  12.   </book>
复制代码

这个查询定义了一个计算折扣价的函数,并使用它计算所有书籍的折扣价。

示例19:递归函数
  1. (: 定义一个计算阶乘的递归函数 :)
  2. declare function local:factorial($n as xs:integer) as xs:integer {
  3.   if ($n = 0) then 1
  4.   else $n * local:factorial($n - 1)
  5. };
  6. (: 计算一些数字的阶乘 :)
  7. <factorials>
  8.   <fact5>{local:factorial(5)}</fact5>
  9.   <fact10>{local:factorial(10)}</fact10>
  10. </factorials>
复制代码

这个查询定义了一个计算阶乘的递归函数,并计算5和10的阶乘。

模块化编程

示例20:创建和使用模块

首先,创建一个名为library.xq的模块文件:
  1. module namespace lib = "http://example.com/library";
  2. (: 定义一个函数来计算书籍的平均价格 :)
  3. declare function lib:averagePrice($books as element(book)*) as xs:decimal {
  4.   avg($books/price)
  5. };
  6. (: 定义一个函数来过滤特定类别的书籍 :)
  7. declare function lib:filterByCategory($books as element(book)*, $category as xs:string) as element(book)* {
  8.   $books[@category = $category]
  9. };
复制代码

然后,在主查询中使用这个模块:
  1. import module namespace lib = "http://example.com/library" at "library.xq";
  2. let $books := doc("bookstore.xml")/bookstore/book
  3. let $fictionBooks := lib:filterByCategory($books, "FICTION")
  4. return
  5.   <report>
  6.     <allBooks>
  7.       <count>{count($books)}</count>
  8.       <avgPrice>{lib:averagePrice($books)}</avgPrice>
  9.     </allBooks>
  10.     <fictionBooks>
  11.       <count>{count($fictionBooks)}</count>
  12.       <avgPrice>{lib:averagePrice($fictionBooks)}</avgPrice>
  13.     </fictionBooks>
  14.   </report>
复制代码

这个示例展示了如何创建和使用XQuery模块来组织代码。

实际应用案例

XML数据转换

示例21:转换XML格式
  1. (: 将书籍数据转换为另一种格式 :)
  2. <inventory>
  3.   <creationDate>{current-date()}</creationDate>
  4.   {
  5.     for $book in doc("bookstore.xml")/bookstore/book
  6.     return
  7.       <item id="{generate-id($book)}">
  8.         <name>{$book/title/text()}</name>
  9.         <details>
  10.           <author>{$book/author/text()}</author>
  11.           <published>{$book/year/text()}</published>
  12.         </details>
  13.         <pricing>
  14.           <list>{$book/price/text()}</list>
  15.           <discount>{round-half-to-even($book/price * 0.9, 2)}</discount>
  16.         </pricing>
  17.       </item>
  18.   }
  19. </inventory>
复制代码

这个查询将原始的书籍数据转换为一种新的格式,包括生成ID、重新组织元素结构和计算折扣价。

HTML生成

示例22:生成HTML表格
  1. (: 生成一个书籍列表的HTML表格 :)
  2. <html>
  3.   <head>
  4.     <title>Book List</title>
  5.     <style>
  6.       table {{ border-collapse: collapse; width: 100%; }}
  7.       th, td {{ border: 1px solid #ddd; padding: 8px; text-align: left; }}
  8.       th {{ background-color: #f2f2f2; }}
  9.       tr:nth-child(even) {{ background-color: #f9f9f9; }}
  10.     </style>
  11.   </head>
  12.   <body>
  13.     <h1>Book List</h1>
  14.     <table>
  15.       <tr>
  16.         <th>Title</th>
  17.         <th>Author</th>
  18.         <th>Category</th>
  19.         <th>Year</th>
  20.         <th>Price</th>
  21.       </tr>
  22.       {
  23.         for $book in doc("bookstore.xml")/bookstore/book
  24.         order by $book/author
  25.         return
  26.           <tr>
  27.             <td>{$book/title/text()}</td>
  28.             <td>{$book/author/text()}</td>
  29.             <td>{$book/@category}</td>
  30.             <td>{$book/year/text()}</td>
  31.             <td>${$book/price/text()}</td>
  32.           </tr>
  33.       }
  34.     </table>
  35.     <p>
  36.       Total books: {count(doc("bookstore.xml")/bookstore/book)}<br/>
  37.       Total value: ${sum(doc("bookstore.xml")/bookstore/book/price)}
  38.     </p>
  39.   </body>
  40. </html>
复制代码

这个查询生成一个完整的HTML页面,包含书籍列表表格和汇总信息。

Web服务集成

示例23:调用REST API并处理结果
  1. (: 调用外部Web服务获取天气数据,并将其与书籍数据结合 :)
  2. let $weather := http:send-request(
  3.   <http:request method="get" href="http://api.example.com/weather?city=New York"/>
  4. )
  5. let $books := doc("bookstore.xml")/bookstore/book
  6. return
  7.   <readingRecommendation>
  8.     <weather>
  9.       <temperature>{$weather//temperature/text()}</temperature>
  10.       <condition>{$weather//condition/text()}</condition>
  11.     </weather>
  12.     <recommendedBooks>
  13.       {
  14.         if ($weather//temperature < 10) then
  15.           (: 推荐适合寒冷天气阅读的书籍 :)
  16.           $books[@category = "FICTION" and number(year) < 1950]
  17.         else if ($weather//temperature > 25) then
  18.           (: 推荐适合暖和天气阅读的书籍 :)
  19.           $books[@category = "SCIENCE"]
  20.         else
  21.           (: 默认推荐 :)
  22.           $books[price < 15]
  23.       }
  24.     </recommendedBooks>
  25.   </readingRecommendation>
复制代码

这个查询调用一个天气API,并根据天气条件推荐不同的书籍。

数据库集成

示例24:查询XML数据库
  1. (: 查询BaseX数据库中的书籍集合,并进行复杂分析 :)
  2. let $books := collection("bookstore")/bookstore/book
  3. return
  4.   <analysis>
  5.     <yearTrends>
  6.       {
  7.         for $year in distinct-values($books/year)
  8.         let $yearBooks := $books[year = $year]
  9.         order by $year
  10.         return
  11.           <year value="{$year}">
  12.             <count>{count($yearBooks)}</count>
  13.             <avgPrice>{avg($yearBooks/price)}</avgPrice>
  14.             <categories>{distinct-values($yearBooks/@category)}</categories>
  15.           </year>
  16.       }
  17.     </yearTrends>
  18.     <priceDistribution>
  19.       <low>{count($books[price < 10])}</low>
  20.       <medium>{count($books[price >= 10 and price < 20])}</medium>
  21.       <high>{count($books[price >= 20])}</high>
  22.     </priceDistribution>
  23.   </analysis>
复制代码

这个查询从BaseX数据库中获取书籍数据,并进行年份趋势分析和价格分布分析。

性能优化技巧

使用索引

示例25:创建和使用索引

在BaseX中,可以创建索引来优化查询:
  1. (: 创建价格索引和类别索引 :)
  2. db:create-index("bookstore", "price", xs:decimal)
  3. db:create-index("bookstore", "category", xs:string)
  4. (: 使用索引优化的查询 :)
  5. for $book in collection("bookstore")/bookstore/book[price > 10 and @category = "FICTION"]
  6. order by $book/price
  7. return $book
复制代码

这个示例展示了如何在BaseX中创建索引,并使用这些索引来优化查询。

避免全文档扫描

示例26:优化路径表达式
  1. (: 不够高效的查询 - 进行了多次路径导航 :)
  2. for $book in doc("large_bookstore.xml")/bookstore/book
  3. where $book/price > 20 and $book/@category = "TECHNOLOGY"
  4. return $book/title
  5. (: 更高效的查询 - 使用变量避免重复导航 :)
  6. for $book in doc("large_bookstore.xml")/bookstore/book
  7. let $price := $book/price
  8. let $category := $book/@category
  9. where $price > 20 and $category = "TECHNOLOGY"
  10. return $book/title
复制代码

这个示例展示了如何通过使用变量来避免重复的路径导航,提高查询效率。

使用适当的函数

示例27:使用高效的函数
  1. (: 不够高效的查询 - 使用了复杂的条件逻辑 :)
  2. for $book in doc("bookstore.xml")/bookstore/book
  3. where if ($book/@category = "FICTION") then $book/price < 15
  4.      else if ($book/@category = "SCIENCE") then $book/price < 20
  5.      else true()
  6. return $book
  7. (: 更高效的查询 - 使用更直接的逻辑 :)
  8. for $book in doc("bookstore.xml")/bookstore/book
  9. where ($book/@category = "FICTION" and $book/price < 15) or
  10.       ($book/@category = "SCIENCE" and $book/price < 20) or
  11.       (not($book/@category = ("FICTION", "SCIENCE")))
  12. return $book
复制代码

这个示例展示了如何简化条件逻辑,提高查询效率。

最佳实践和常见错误

最佳实践

1. 使用有意义的变量名
  1. (: 不好的做法 :)
  2.    for $b in doc("books.xml")/books/book
  3.    return $b
  4.    
  5.    (: 好的做法 :)
  6.    for $book in doc("books.xml")/books/book
  7.    return $book
复制代码

1. 添加注释
  1. (: 这个查询计算每个类别的平均价格 :)
  2.    for $book in doc("bookstore.xml")/bookstore/book
  3.    group by $category := $book/@category
  4.    return
  5.      <categoryStats category="{$category}">
  6.        <avgPrice>{avg($book/price)}</avgPrice>
  7.      </categoryStats>
复制代码

1. 使用模块化编程
  1. (: 将常用函数放在模块中 :)
  2.    module namespace utils = "http://example.com/utils";
  3.    
  4.    declare function utils:formatPrice($price as xs:decimal) as xs:string {
  5.      concat("$", $price)
  6.    };
复制代码

1. 处理空序列
  1. (: 不够健壮的代码 :)
  2.    let $price := doc("bookstore.xml")/bookstore/book[1]/price
  3.    return $price * 1.1
  4.    
  5.    (: 更健壮的代码 - 处理可能的空序列 :)
  6.    let $price := doc("bookstore.xml")/bookstore/book[1]/price
  7.    return if (exists($price)) then $price * 1.1 else 0
复制代码

常见错误

1. 忽略命名空间
  1. (: 错误 - 忽略了命名空间 :)
  2.    for $book in doc("books.xml")//book
  3.    return $book
  4.    
  5.    (: 正确 - 处理命名空间 :)
  6.    declare namespace ns = "http://example.com/books";
  7.    for $book in doc("books.xml")//ns:book
  8.    return $book
复制代码

1. 混淆路径表达式
  1. (: 错误 - 混淆了绝对路径和相对路径 :)
  2.    let $book := doc("bookstore.xml")/bookstore/book[1]
  3.    return /bookstore/book[price > 10]
  4.    
  5.    (: 正确 - 使用上下文项或明确路径 :)
  6.    let $book := doc("bookstore.xml")/bookstore/book[1]
  7.    return $book/../book[price > 10]
复制代码

1. 错误使用FLWOR表达式
  1. (: 错误 - 在FLWOR中混合使用for和let :)
  2.    for $book in doc("bookstore.xml")/bookstore/book
  3.    let $price := $book/price
  4.    where $price > 10
  5.    return $book
  6.    
  7.    (: 正确 - FLWOR表达式结构清晰 :)
  8.    for $book in doc("bookstore.xml")/bookstore/book
  9.    let $price := $book/price
  10.    where $price > 10
  11.    return $book
复制代码

1. 忽略类型检查
  1. (: 错误 - 忽略类型检查可能导致运行时错误 :)
  2.    let $price := doc("bookstore.xml")/bookstore/book[1]/price
  3.    return $price + 10
  4.    
  5.    (: 正确 - 使用类型检查和转换 :)
  6.    let $price := xs:decimal(doc("bookstore.xml")/bookstore/book[1]/price)
  7.    return $price + 10
复制代码

总结与展望

XQuery是一种强大而灵活的XML查询语言,它提供了从简单查询到复杂数据处理的全面功能。通过本文的实例解析,我们深入了解了XQuery的各种特性,包括基本的路径表达式、FLWOR表达式、连接操作、聚合函数、量词表达式、序列操作、节点构造、用户定义函数和模块化编程等。

随着XML数据在各种应用中的持续使用,XQuery的重要性只会增加。未来,我们可以期待XQuery在与JSON和其他数据格式的集成、更强大的流处理能力、与云计算和大数据平台的更好集成以及更高级的分析功能等方面的发展。

通过掌握XQuery,你将拥有处理XML数据的强大工具,能够应对从简单查询到复杂数据转换的各种挑战。继续实践和探索XQuery的各种功能,将有助于你全面提升XML数据处理和编程技能。
「七転び八起き(ななころびやおき)」
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则