How to sort xml child via attribute value PhP

喜夏-厌秋 提交于 2021-02-05 08:18:57

问题


I have an XML file through PHP that I am trying to list a specific child "word" in ascending order in a div, according to the attribute "word".

I put sort($word); without sucess

Here is my code:

<?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());

}
?>

The output

    <?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>

So, the word "ball" should be listed before "piece", in ascending order as I said above when I parse the xml in a php page.


回答1:


SimpleXML is not good at manipulating nodes so this is a lot easier with 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();

To sort by child element text content you would just have to change the compare function.

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)
        );
    }
);


来源:https://stackoverflow.com/questions/61925360/how-to-sort-xml-child-via-attribute-value-php

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!