3

我需要以特定方式打印任意SimpleXML 对象,并对属性节点进行特殊处理。

问题是 SimpleXML 元素和属性似乎使用完全相同的类,属性节点甚至假装支持attributes()方法,而 SimpleXML 隐藏了它的内部,所以似乎没有任何方法可以告诉节点类型(缺少生成 XML并重新分析它)。

两者都给出相同的结果:

$element = new SimpleXMLElement('<foo>test</foo>');
echo $element;
print_r($element);

$element = new SimpleXMLElement('<foo attr="test" />');
echo $element['attr'];
print_r($element['attr']);

是否有允许在 SimpleXML 中识别节点类型的隐藏属性/方法?等效于 DOM$node->nodeType$node instanceof DOMAttr? (我不能改用 DOM,对 SimpleXML 的支持是核心要求)。

4

6 回答 6

3

没有内置属性SimpleXMLElement可以让您将它们区分开来。

然而,正如其他人所建议dom_import_simplexml的那样,该函数有时可以动态更改节点,例如,如果您传入一个子节点列表或命名的子节点,它将接受这些并将它们变成第一个元素。

如果它是一个空列表,例如没有从attributes()或不存在的命名子节点返回属性,它会给出一个警告,告诉你一个无效的节点类型已经给出:

警告:dom_import_simplexml():要导入的节点类型无效

因此,如果您需要使用活泼的布尔值true/false来实现这一点,那么它是如何与 Simplexml 一起使用的:

$isElement   = $element->xpath('.') == array($element);

$isAttribute = $element[0] == $element
               and $element->xpath('.') != array($element);

它与属性列表和元素列表类似,我早上刚刚在博客上写过这个,你需要有一些关于评估什么的具体知识,所以我为它创建了一个备忘单:

+------------------+---------------------------------------------+
| TYPE             | TEST                                        |
+------------------+---------------------------------------------+
| Element          | $element->xpath('.') == array($element)     |
+------------------+---------------------------------------------+
| Attribute        | $element[0] == $element                     |
|                  | and $element->xpath('.') != array($element) |
+------------------+---------------------------------------------+
| Attributes       | $element->attributes() === NULL             |
+------------------+---------------------------------------------+
| Elements         | $element[0] != $element                     |
|                  | and $element->attributes() !== NULL         |
+------------------+---------------------------------------------+
| Single           | $element[0] == $element                     |
+------------------+---------------------------------------------+
| Empty List       | $element[0] == NULL                         |
+------------------+---------------------------------------------+
| Document Element | $element->xpath('/*') == array($element)    |
+------------------+---------------------------------------------+
于 2013-02-12T09:29:05.163 回答
2

是的,有办法。好吧,您可以通过 API 检索没有什么优雅的东西,但是在 SimpleXML 内部的某个地方会跟踪它是什么并且它会产生差异,例如,当您调用诸如 getName() 或 asXML() 之类的函数时。

$element = new SimpleXMLElement('<foo>test</foo>');
print_r($element->getName());
print_r($element->asXML());
echo "------------------\n";
$element = new SimpleXMLElement('<foo attr="test" />');
$at = $element['attr'];
print_r($at->getName());
print_r($at->asXML());

foo
<?xml version="1.0"?>
<foo>test</foo>
------------------
attr 
attr="test"

你的代码不会赢得选美比赛,但至少你可以做到。

于 2009-05-15T20:40:43.963 回答
2

使用 palako 指出的内容,这样的功能可能会起作用:

function is_attribute($node) {
    return !($node->asXML()[0] == "<")
}
于 2009-05-18T03:04:23.210 回答
1

您需要SimpleXMLElement::attributes

function xml_out($el) {
    $name = $el->getName();
    echo '<'. $name;

    if (count($el->attributes())) {
        foreach ($el->attributes() as $a=>$v) {
            echo ' '. ((string)$a) . '="' . htmlspecialchars((string)$v) . '"';
        }
    }

    echo '>'. (string)$el;

    if (count($el->children())) {
        foreach($el->children() as $c) {
            xml_out($c);
        }
    }
    echo '</'. $name . '>';

}

可能需要一些调整。

于 2009-04-07T10:29:27.987 回答
1

是否有允许在 SimpleXML 中识别节点类型的隐藏属性/方法?DOM 的 $node->nodeType 或 $node instanceof DOMAttr 的等价物?(我不能使用 DOM,支持 SimpleXML 是核心要求)

答案是否定的……是的。SimpleXML 没有这样的属性,但好消息是:SimpleXML 和 DOM 是同一枚硬币的两个面。(硬币是 libxml ;))您不必选择其中之一,您可以同时使用!你可以SimpleXMLElement变成DOMNode,反之亦然。在您的情况下,您可以执行以下操作:

$element = new SimpleXMLElement('<foo attr="test" />');
$is_attr = (dom_import_simplexml($element['attr'])->nodeType === XML_ATTRIBUTE_NODE);

如果这是您经常做的事情,或者您不想处理 DOM/SimpleXML 杂耍,您可以看看SimpleDOM

$element = new SimpleDOM('<foo attr="test" />');
$is_attr = ($element['attr']->nodeType() === XML_ATTRIBUTE_NODE);
于 2009-11-15T04:49:56.467 回答
0

不幸的是,没有允许在 SimpleXML 中识别节点类型的隐藏属性或方法。SimpleXML 仅使用一个 Class 并且元素没有任何指向其父级的内容。如果您尝试以下反射,您会发现没有什么可以区分元素或属性。

$element = new SimpleXMLElement('<foo>test</foo>');
echo ReflectionObject::export($element);

$element = new SimpleXMLElement('<foo attr="test">test</foo>');
echo ReflectionObject::export($element['attr']);

但是,如果元素具有属性,您可以检测到它。因此,您必须假设传入的所有元素都具有属性。有了这个假设,您就可以将它们区分开来。

$element = new SimpleXMLElement('<foo attr="test">test</foo>');

echo ReflectionObject::export($element);
echo ReflectionObject::export($element['attr']);

这是我能想到的最好的。请记住,没有属性的元素仍然可能返回 false 。

function isNotAttribute($simpleXML)
{
  return (count($simpleXML->attributes()) > 0);
}

$element = new SimpleXMLElement('<foo attr="test">test</foo>');
var_dump(isNotAttribute($element));
var_dump(isNotAttribute($element['attr']));

// returns
bool(true)
bool(false)
于 2009-05-15T15:54:40.283 回答