xml - Oracle XMLQuery 插入不需要的命名空间

标签 xml oracle11g namespaces qxmlquery

甲骨文 11.2

下面是我在 XMLType 列上运行的 XMLQuery 的简化版本。当我运行查询时,它只是解析并重新创建存储的 XML不需要的 default 和 tsip 命名空间被插入到父元素的子元素中。请注意,不会插入 tsxm 命名空间,这是因为它不等于默认命名空间此查询不执行任何操作并且可以轻松重写,但实际(更大)查询使用相同的方法,所以这就是我的原因以这种格式发布问题。

创建表:

CREATE TABLE XML_DOCUMENT_TMP
(
  DOCUMENT_ID   NUMBER(12)                      NOT NULL,
  XML_DATA      SYS.XMLTYPE                     NOT NULL,
  CREATED_DATE  TIMESTAMP(6)                    NOT NULL
);

插入一些数据(必须具有原样的命名空间):

insert into XML_DOCUMENT_TMP
(document_id,created_date,xml_data)
values(1,sysdate, 
'<patent  xmlns="http://schemas.thomson.com/ts/20041221/tsip" 
    xmlns:tsip="http://schemas.thomson.com/ts/20041221/tsip" 
    xmlns:tsxm="http://schemas.thomson.com/ts/20041221/tsxm"  
    tsip:action="replace" tsip:cc="CA" tsip:se="2715340" tsip:ki="C">
    <accessions tsip:action="replace">
        <accession tsip:src="wila" tsip:type="key">CA-2715340-C</accession>
        <accession tsip:src="tscm" tsip:type="tscmKey">CA-2715340-C-20150804</accession>
    </accessions>
    <claimed tsip:action="replace">
    <    claimsTsxm tsip:lang="en">
            <tsxm:heading tsxm:align="left">We Claim:</tsxm:heading>
            <claimTsxm tsip:no="1" tsxm:num="1" tsip:type="main">1.  power.       </claimTsxm>
      </claimsTsxm>
  </claimed>
</patent>
');

运行XMLQuery:

请注意命名空间通配符的必要性已在 here 中解释过

WITH tmpTable AS (
SELECT * FROM XML_DOCUMENT_TMP cm )
SELECT tt.xml_data ,
XMLQuery('declare default element namespace  "http://schemas.thomson.com/ts/20041221/tsip";
  declare namespace  tsip="http://schemas.thomson.com/ts/20041221/tsip";
  declare namespace  tsxm="http://schemas.thomson.com/ts/20041221/tsxm"; 


  return          
  <patent>{$m/*:patent/@*}
  {
    for $i in $m/*:patent/*
        return    $i
  }
  </patent>' 
        PASSING tt.xml_data as "m"   RETURNING CONTENT) newXml 
 FROM tmpTable tt
 WHERE tt.document_id in (1);

返回:

<patent xmlns="http://schemas.thomson.com/ts/20041221/tsip" xmlns:tsip="http://schemas.thomson.com/ts/20041221/tsip" tsip:action="replace" tsip:cc="CA" tsip:se="2715340" tsip:ki="C">
    <accessions xmlns="http://schemas.thomson.com/ts/20041221/tsip" xmlns:tsip="http://schemas.thomson.com/ts/20041221/tsip" tsip:action="replace">
        <accession tsip:src="wila" tsip:type="key">CA-2715340-C</accession>
        <accession tsip:src="tscm" tsip:type="tscmKey">CA-2715340-C-20150804</accession>
    </accessions>
    <claimed xmlns="http://schemas.thomson.com/ts/20041221/tsip" xmlns:tsip="http://schemas.thomson.com/ts/20041221/tsip" tsip:action="replace">
        <claimsTsxm tsip:lang="en">
            <tsxm:heading xmlns:tsxm="http://schemas.thomson.com/ts/20041221/tsip" tsxm:align="left">We Claim:</tsxm:heading>
            <claimTsxm tsip:no="1" xmlns:tsxm="http://schemas.thomson.com/ts/20041221/tsip" tsxm:num="1" tsip:type="main">1.  power.</claimTsxm>
        </claimsTsxm>
</claimed>

如何删除在加入和声明的元素中创建的不需要的 namespace 。 任何建议表示赞赏。

最佳答案

如果您尝试使用命名空间的各种值,您会发现虽然顶部 <patent>由于您所做的声明,命名空间被声明和包含在子元素级别,此信息未按您期望的方式使用。

XQuery 正在根据在该执行循环中考虑的节点中使用的 namespace 提取 namespace ,独立于整个文档。这就是每次 XQuery 循环时它们都会“重新声明”的原因。

其他文章解释说您正在尝试做的是“解析”数据以及“提取”数据,这在一定程度上是正确的,因此 XSLT 是正确的工具而不是 XQuery。

我发现的一个外部链接有一个 XQuery 方法来剥离 namespace ,因此返回“原始”XML 是 here .

将该代码应用于您的 XQuery 使我能够:

SELECT xmlquery('xquery version "1.0"; (: :)
             declare default element namespace 
                        "http://www.somewherein.uk/ns/1.0"; (: :)

             declare function local:strip-namespace($inputRequest  as element()) as element()
             {
                element {xs:QName(local-name($inputRequest ))}
                {
                  for $child in $inputRequest /(@*,node())
                    return
                      if ($child instance of element())
                      then local:strip-namespace($child)
                      else $child
                }
             }; (: :)

             <patent>
             {
             for $s in /*:patent/*
              return local:strip-namespace($s)
             }
             </patent>' 
             PASSING cmf.XML_DATA 
             RETURNING content)
FROM XML_DOCUMENT_TMP cmf WHERE cmf.DOCUMENT_ID=1

一些进一步的编辑让我到了下面,我认为这就是你所追求的(在 patent 级别定义的命名空间)

SELECT xmlquery('xquery version "1.0"; (: :)
             declare default element namespace 
                        "http://www.somewherein.uk/ns/1.0"; (: :)

             declare function local:strip-namespace($inputRequest as element()) as element()
             {
                element {fn:name($inputRequest)}
                {
                  for $child in $inputRequest /(@*,node())
                    return
                      if ($child instance of element())
                      then local:strip-namespace($child)
                      else $child
                }
             }; (: :)

             <patent>
             {
             for $s in /(*:patent, node())
              return local:strip-namespace($s)
             }
             </patent>' 
             PASSING cmf.XML_DATA 
             RETURNING content)
FROM XML_DOCUMENT_TMP cmf WHERE cmf.DOCUMENT_ID=1;

如下所述,由于 XPath 中的某些问题,导致循环代码中出现一些重复。这也意味着 txsm namespace 被声明了几次; XQuery 声明它是“第一次”,它在遍历该树分支时遇到正在使用它的 namespace ,这意味着如果有 sibling 使用 ns,那么它将被声明多次。通过将声明的显式位置移回父节点,我们可以消除这种情况。

SELECT xmlquery('xquery version "1.0"; (: :)
             declare default element namespace  "http://schemas.thomson.com/ts/20041221/tsip"; (: :)
             declare namespace  tsip="http://schemas.thomson.com/ts/20041221/tsip"; (: :)
             declare namespace  tsxm="http://schemas.thomson.com/ts/20041221/tsxm"; (: :)

             declare function local:strip-namespace($inputRequest as element()) as element()
             {
                element {fn:name($inputRequest)}
                {
                  for $child in $inputRequest /(@*,node())
                    return
                      if ($child instance of element())
                      then local:strip-namespace($child)
                      else $child
                }
             }; (: :)

             <patent xmlns:tsxm="http://schemas.thomson.com/ts/20041221/tsxm" xmlns:tsip="http://schemas.thomson.com/ts/20041221/tsip">
             {
             for $s in /*:patent/*
              return local:strip-namespace($s)
             }
             </patent>' 
             PASSING cmf.XML_DATA
             RETURNING content)
FROM XML_DOCUMENT_TMP cmf WHERE cmf.DOCUMENT_ID=1;

关于xml - Oracle XMLQuery 插入不需要的命名空间,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38653381/

相关文章:

sql - 条件纯sql查询

oracle - ORA-01461 (with > 4k varchar2) 错误 仅在合并语句中。插入或更新工作正常

c# - 持有枚举的命名空间建议?

c# - 根级别的数据无效。第 1 行,位置 1。读取 xml 时

sql - 使用 SQL 从 Oracle 数据库中插入和检索 BLOB 数据的方法有哪些?

xml - 错误十六进制值0x00是无效字符,通过LAN发送XML格式的数据表时

c++ - 命名类时使用命名空间或前缀供应商名称?

r - 当我获取 R 文件时,library(<package>) 到底发生了什么

java - 如何使用 jxls 进行 XML 映射

java - 我可以使用 CURL -u 访问 XML 数据,并使用 HTTP Basic Auth 在浏览器中获取它。如何在 Java 中复制该过程?