如何通过属性值 PhP 对 xml child 进行排序

How to sort xml child via attribute value PhP

我通过 PHP 有一个 XML 文件,我试图在 div 中按升序列出特定的 child "word",根据属性 "word".

我把sort($word);没有成功

这是我的代码:

<?php
if(isset($_POST['submitSave'])) {
// Disable errors due to empty xml files
error_reporting(E_ALL & ~E_WARNING);

$domDoc = new DOMDocument('1.0');
$domDoc->preserveWhiteSpace = false;
$domDoc->formatOutput = true;
$xpath = new DOMXpath($domDoc);

// load xml file
try {
    $domDoc->load('./data/expression.xml');
} catch (\Throwable $th) {
    //throw $th;
}

if($domDoc->getElementsByTagName('expression')->length>0){
    // If we already have expression tag defined
    $expression = $domDoc->getElementsByTagName('expression')[0];
}else{
    // If we don't have any expression tag, i.e. file is empty
    $expression = $domDoc->createElement('expression');
}



  // Create child node for product and set id(attribute), word(child), 
  meaning(child)
$vocabulario = $domDoc->createElement('vocabulario');
$vocabulario->setAttribute('word', $_POST['word']);
$word = $domDoc->createElement('word', $_POST['word']);
$meaning = $domDoc->createElement('meaning', $_POST['meaning']);




$domDoc->appendChild($expression);
$expression->appendChild($vocabulario);
$vocabulario->appendChild($word);
$vocabulario->appendChild($meaning);



// convert returned node list into an array
$vocabularios = iterator_to_array(
// fetch all vocabulario elements
$xpath->evaluate('/expression/vocabulario')  
);
// sort the array
usort(
$vocabularios,
static function($a, $b) use ($xpath) {
    // compare the word child element
    return strcasecmp(
        $xpath->evaluate('string(word)', $a),
        $xpath->evaluate('string(word)', $b)
       );
      }
    );

// iterate the sorted array
foreach ($vocabularios as $vocabulario) {
// append node to the expression document element
// that moves nodes from their current position
$domDoc->documentElement->appendChild($vocabulario);
}




file_put_contents('./data/expression.xml', $domDoc->saveXML());

}
?>

输出

    <?xml version="1.0" encoding="UTF-8"?>
<expression>
  <vocabulario word="piece">
    <word>piece</word>
    <meaning>pedaço</meaning>
  </vocabulario>
  <vocabulario word="ball">
    <word>ball</word>
    <meaning>bola</meaning>
  </vocabulario>
</expression>

所以,单词"ball" 应该在"piece" 之前列出,正如我在php 页面中解析xml 时所说的升序排列。

SimpleXML 不擅长操纵节点,因此使用 DOM 更容易。

$document = new DOMDocument();
$document->preserveWhiteSpace = FALSE;
$document->loadXML($xmlString);
$xpath = new DOMXpath($document);

// convert returned node list into an array
$vocabularios = iterator_to_array(
    // fetch all vocabulario elements
    $xpath->evaluate('/expression/vocabulario')  
);
// sort the array
usort(
    $vocabularios,
    static function($a, $b) {
        // compare the word attribute
        return strcasecmp($a->getAttribute('word'), $b->getAttribute('word'));
    }
);

// iterate the sorted array
foreach ($vocabularios as $vocabulario) {
    // append node to the expression document element
    // that moves nodes from their current position
    $document->documentElement->appendChild($vocabulario);
}

$document->formatOutput = TRUE;
echo $document->saveXML();

要按子元素文本内容排序,您只需更改比较函数即可。

usort(
    $vocabularios,
    static function($a, $b) use ($xpath) {
        // compare the word child element
        return strcasecmp(
            $xpath->evaluate('string(word)', $a),
            $xpath->evaluate('string(word)', $b)
        );
    }
);