gpt4 book ai didi

ruby - 根据命名空间前缀删除 nokogiri 属性

转载 作者:数据小太阳 更新时间:2023-10-29 08:56:43 24 4
gpt4 key购买 nike

我正在使用 nokogiri 来解析 XML 文件。文件中的一些节点具有特定于命名空间的属性:

<metadata xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:opf="http://www.idpf.org/2007/opf">
<dc:identifier id="iden" opf:scheme="ISBN">xxxx</dc:identifier>
<dc:creator opf:role="aut" opf:file-as="Name">xxxx</dc:creator>
<dc:date opf:event="publication">xxxx</dc:date>
<dc:publisher>xxxx</dc:publisher>
<meta name="cover" content="x"/>
</metadata>

我正在尝试删除任何带有“opf”前缀的属性。我遇到过基于部分匹配查找属性 value 的 xpath 解决方案,但是当它是属性名称本身的部分匹配时呢?我尝试了很多没有用的东西。我做了一件简单的事情,只是试图至少提取属性名称,但如果我这样做了:

elements = @doc.at_xpath('//xmlns:metadata').children
elements.each { |el|
el.attributes.each { |attribute|
if attribute[1].namespace_scopes[1].prefix == "opf"
puts attribute[0]
end
}
}

我最终得到:

id
scheme
role
file-as
event
name
content

但我只想要带有“opf”前缀的那些(“opf:scheme”、“opf:role”、“opf:file-as”、“opf:event”),这样它们就可以被删除,而不会触及任何其他属性。我什至试图通过硬编码我知道存在的属性来强制它:

opf_attributes = ["opf:file-as","opf:scheme","opf:role","opf:event"]
elements.each { |el|
opf_attributes.each { |x|
el.remove_attribute(x) if el[x] != nil
}
}

这不是解决此问题的最明智的方法,但这仍然行不通。节点没有任何变化,属性保持原样。 (我不知道这是否值得注意,但如果我改用 remove_attr(x) 方法,我会收到此错误:undefined method 'remove_attr' for #<Nokogiri::XML::Element:0x...>

所以,我的问题是:
有没有更清晰的方法

  1. 根据部分匹配和/或 namespace 前缀查找属性,然后
  2. 从包含它们的节点中删除这些属性?

最佳答案

节点对象有一个remove方法可以将它们从树中删除,所以你可以这样写:

require 'nokogiri'

doc = Nokogiri::XML(DATA)
puts '--- Before'
puts doc.to_s

doc.traverse do |node|
next unless node.respond_to? :attributes
node.attributes.each do |key, val|
val.remove if val&.namespace&.prefix == 'opf'
end
end

puts
puts '--- After'
puts doc.to_s

__END__
<metadata xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:opf="http://www.idpf.org/2007/opf">
<dc:identifier id="iden" opf:scheme="ISBN">xxxx</dc:identifier>
<dc:creator opf:role="aut" opf:file-as="Name">xxxx</dc:creator>
<dc:date opf:event="publication">xxxx</dc:date>
<dc:publisher>xxxx</dc:publisher>
<meta name="cover" content="x"/>
</metadata>

并查看以下输出:

➜  ~ ruby test.rb
--- Before
<?xml version="1.0"?>
<metadata xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:opf="http://www.idpf.org/2007/opf">
<dc:identifier id="iden" opf:scheme="ISBN">xxxx</dc:identifier>
<dc:creator opf:role="aut" opf:file-as="Name">xxxx</dc:creator>
<dc:date opf:event="publication">xxxx</dc:date>
<dc:publisher>xxxx</dc:publisher>
<meta name="cover" content="x"/>
</metadata>

--- After
<?xml version="1.0"?>
<metadata xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:opf="http://www.idpf.org/2007/opf">
<dc:identifier id="iden">xxxx</dc:identifier>
<dc:creator>xxxx</dc:creator>
<dc:date>xxxx</dc:date>
<dc:publisher>xxxx</dc:publisher>
<meta name="cover" content="x"/>
</metadata>

注意 如果您使用的 Ruby 版本不支持 &.,您将需要处理可能为 nil 的命名空间。

关于ruby - 根据命名空间前缀删除 nokogiri 属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53180114/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com