Xp को array में php में कैसे कन्वर्ट करें?


166

मैं XML से नीचे PHP सरणी में बदलना चाहते हैं। कैसे मैं यह कर सकता हूँ पर कोई सुझाव?

<aaaa Version="1.0">
   <bbb>
     <cccc>
       <dddd Id="id:pass" />
       <eeee name="hearaman" age="24" />
     </cccc>
   </bbb>
</aaaa>

3
यह भी कि यह प्रश्न आपके अन्य प्रश्न को कैसे अलग बनाता है? stackoverflow.com/questions/6578084/…
गॉर्डन

3
कुछ चीजें उतनी ही अप्रिय हैं जितना कि एक ओपी अपने स्वयं के प्रश्न के गलत उत्तर को स्वीकार करता है।
जॉन

जवाबों:


127

एक अन्य विकल्प सिम्पलेक्स एक्सएमएल एक्सटेंशन है (मेरा मानना ​​है कि यह अधिकांश php इंस्टॉल के साथ मानक आता है।)

http://php.net/manual/en/book.simplexml.php

सिंटैक्स आपके उदाहरण के लिए कुछ इस तरह दिखता है

$xml = new SimpleXMLElement($xmlString);
echo $xml->bbb->cccc->dddd['Id'];
echo $xml->bbb->cccc->eeee['name'];
// or...........
foreach ($xml->bbb->cccc as $element) {
  foreach($element as $key => $val) {
   echo "{$key}: {$val}";
  }
}

82
निष्पक्ष होने के लिए, यह इस सवाल का बिल्कुल जवाब नहीं देता है कि सरणी कैसे प्राप्त करें।
8

इस xml को पार्स करते समय SimpleXML को चूसें: amazon.in/rss/bestsellers/shoes?tag=dealslama-21 यहां तक ​​कि Print_r ऑब्जेक्ट को वास्तव में नहीं बताता है।
रावीसन

var_dump का उपयोग करें, आपको ऑब्जेक्ट के अंदर कुंजियों के रूप में xml संरचना दिखाई देगी।
Magus

3
मेरे पास कुछ [CDATA[TEXT]]तत्वों के अंदर है और वे इसके साथ पार्स नहीं कर रहे हैं। यह एक के रूप में इसे पार्स करता है SimpleXMLElement Object। उसके लिए कोई वर्कअराउंड?
मास्टर ऑफ

यह इस सवाल का जवाब नहीं देता है
बिलाल राशिद

434

आसान!

$xml = simplexml_load_string($xmlstring, "SimpleXMLElement", LIBXML_NOCDATA);
$json = json_encode($xml);
$array = json_decode($json,TRUE);

20
आप सीडीएटीए सेक्शन में परेशानी में पड़ सकते हैं। समाधान के रूप में $ xml = simplexml_load_string ($ xmlstring, null, LIBXML_NOCDATA) आज़माएं; $ json = json_encode ($ xml); $ सरणी = json_decode ($ json, TRUE); (देखें stackoverflow.com/a/2970701/413531 ) // e damn .. क्या टिप्पणी में नई लाइनें जोड़ने का कोई तरीका है?
हिरणमस्टर

4
हम ठीक यही काम करते हैं लेकिन simplexml_load_file के साथ और यह ठीक काम करता है। धन्यवाद
थर्मेक

2
दूसरा पैरामीटर (TRUE) किसके लिए है?
मंसूर फहद

3
@MansourFahad json_decode में आप JSON इनपुट को साहचर्य सरणी में बदलने के लिए वैकल्पिक दूसरे पैरामीटर का उपयोग कर सकते हैं TRUE(आमतौर पर डिफ़ॉल्ट के रूप में FALSE)।
जेक बाथमैन

16
@ इस्माइल मिगुएल बहुत अधिक कोड? सिर्फ इसलिए कि आप उन सभी कार्यों को एक पंक्ति में रखते हैं इसका मतलब यह नहीं है कि आप कम कोड का उपयोग कर रहे हैं। यह अधिक कॉम्पैक्ट लग सकता है लेकिन यह पठनीयता की कीमत पर आता है।
जेजे

43

XML स्ट्रिंग ( $buffer) को एक सरलीकृत सरणी में परिवर्तित करना जो विशेषताओं को अनदेखा करता है और समान नाम वाले बाल-तत्वों को समूहीकृत करता है:

function XML2Array(SimpleXMLElement $parent)
{
    $array = array();

    foreach ($parent as $name => $element) {
        ($node = & $array[$name])
            && (1 === count($node) ? $node = array($node) : 1)
            && $node = & $node[];

        $node = $element->count() ? XML2Array($element) : trim($element);
    }

    return $array;
}

$xml   = simplexml_load_string($buffer);
$array = XML2Array($xml);
$array = array($xml->getName() => $array);

परिणाम:

Array
(
    [aaaa] => Array
        (
            [bbb] => Array
                (
                    [cccc] => Array
                        (
                            [dddd] => 
                            [eeee] => 
                        )

                )

        )

)

यदि आप भी विशेषताएँ रखना चाहते हैं, तो वे JSON एन्कोडिंग / SimpleXMLElement के डिकोडिंग के माध्यम से उपलब्ध हैं। यह अक्सर सबसे आसान त्वरित समाधान होता है:

$xml   = simplexml_load_string($buffer);
$array = json_decode(json_encode((array) $xml), true);
$array = array($xml->getName() => $array);

परिणाम:

Array
(
    [aaaa] => Array
        (
            [@attributes] => Array
                (
                    [Version] => 1.0
                )

            [bbb] => Array
                (
                    [cccc] => Array
                        (
                            [dddd] => Array
                                (
                                    [@attributes] => Array
                                        (
                                            [Id] => id:pass
                                        )

                                )

                            [eeee] => Array
                                (
                                    [@attributes] => Array
                                        (
                                            [name] => hearaman
                                            [age] => 24
                                        )

                                )

                        )

                )

        )

)

ध्यान दें कि ये सभी विधियाँ केवल XML दस्तावेज़ के नामस्थान में काम करती हैं।


संबंधित:
जोंस

PHP 7 में मुझे इसे जोड़ना था: && (is_countable($node) && 1 === count($node) ? $node = array($node) : 1)हालाँकि, मुझे अगली पंक्ति में एक त्रुटि मिली [] operator not supported for strings:।
andreshg112

@ @reshg112: मैं पुन: पेश करने में असमर्थ हूं (स्थिर PHP 5.3.0 - 7.4.0 काम करता है), व्यवहार उम्र के कारण नहीं बदला है, कृपया विभिन्न PHP संस्करणों के सैकड़ों की तुलना करें: 3v4l.org/l4nQN
hakre

शायद यह मेरी KML फ़ाइल (यह एक XML) के कारण है। मैं इसे साझा नहीं कर सकता। मैंने इसे पहले ही आयात कर लिया था लेकिन मुझे इसे दूसरे तरीके से करना था।
andreshg112

शायद आप XML नामस्थान के बारे में चिंतित हैं। उदाहरण केवल भागों w / oa नाम स्थान के लिए है (या डिफ़ॉल्ट एक, मैं कभी-कभी इसे मिलाता हूं)।
हक्रे

24
$array = json_decode(json_encode((array)simplexml_load_string($xml)),true);

1
यदि आप सरणी के लिए जाते हैं, तो आपको जरूरत नहीं है json_encodeऔर json_decode
इस्माइल मिगुएल

11
@ इस्माईल इन थ्योरी, आरेंज टू अरेंजिंग काफी होनी चाहिए। व्यवहार में हमें सभी पत्ती नोड्स को भी डालना होगा, जो कि वस्तुएं भी हैं। एक भोली जाति पत्तियों को SimpleXML ऑब्जेक्ट के रूप में छोड़ देती है। json_encode जातियां पुनरावर्ती रूप से बहुत अधिक लेगवर्क बचाती हैं।
पीटर मेललेट

2
यदि आपके $arrayचर में पाठ मान नहीं हैं , तो यह शायद CDATA के कारण है। इसे हल करने के लिए, के साथ अपने एक्सएमएल लोड: new SimpleXMLElement($xml, LIBXML_NOCDATA)
जोनाथन पेटिटोलस

1
ps। $ xml = str_replace (सरणी ('<! [CDATA [', ']]>'), '', $ xml);
user956584

1
यह उस तरह काम नहीं करता है। यह कोड इस तरह एक भी सरल XML में परिवर्तित नहीं होगा <?xml version="1.0" encoding="UTF-8"?><note a="b"><body c="d">Hello!</body></note>। कोड चलाएं और आप देखेंगे कि <body> c विशेषता खो गई है! यहाँ पूर्ण कोड की जांच करें यदि आप किसी भी बुरा आश्चर्य नहीं करना चाहती github.com/gaarf/XML-string-to-PHP-array/blob/master/... या नीचे मेरी जवाब देखने stackoverflow.com/a/30234924/828366
फ्रांसेस्को कासुला

7

Https://github.com/gaarf/XML-string-to-PHP-array/blob/master/xmlstr_to_array.php देखें

<?php
/**
  * convert xml string to php array - useful to get a serializable value
  *
  * @param string $xmlstr
  * @return array
  *
  * @author Adrien aka Gaarf & contributors
  * @see http://gaarf.info/2009/08/13/xml-string-to-php-array/
*/
function xmlstr_to_array($xmlstr) {
  $doc = new DOMDocument();
  $doc->loadXML($xmlstr);
  $root = $doc->documentElement;
  $output = domnode_to_array($root);
  $output['@root'] = $root->tagName;
  return $output;
}
function domnode_to_array($node) {
  $output = array();
  switch ($node->nodeType) {
    case XML_CDATA_SECTION_NODE:
    case XML_TEXT_NODE:
      $output = trim($node->textContent);
    break;
    case XML_ELEMENT_NODE:
      for ($i=0, $m=$node->childNodes->length; $i<$m; $i++) {
        $child = $node->childNodes->item($i);
        $v = domnode_to_array($child);
        if(isset($child->tagName)) {
          $t = $child->tagName;
          if(!isset($output[$t])) {
            $output[$t] = array();
          }
          $output[$t][] = $v;
        }
        elseif($v || $v === '0') {
          $output = (string) $v;
        }
      }
      if($node->attributes->length && !is_array($output)) { //Has attributes but isn't an array
        $output = array('@content'=>$output); //Change output into an array.
      }
      if(is_array($output)) {
        if($node->attributes->length) {
          $a = array();
          foreach($node->attributes as $attrName => $attrNode) {
            $a[$attrName] = (string) $attrNode->value;
          }
          $output['@attributes'] = $a;
        }
        foreach ($output as $t => $v) {
          if(is_array($v) && count($v)==1 && $t!='@attributes') {
            $output[$t] = $v[0];
          }
        }
      }
    break;
  }
  return $output;
}

7

आश्चर्यचकित किसी ने उल्लेख नहीं किया xml_parse_into_struct:

$simple = "<para><note>simple note</note></para>";
$p = xml_parser_create();
xml_parse_into_struct($p, $simple, $vals, $index);
xml_parser_free($p);
echo "Index array\n";
print_r($index);
echo "\nVals array\n";
print_r($vals);

कभी-कभी मुझे आश्चर्य होता है कि xml_parse_into_struct को डिज़ाइन करते समय PHP XML कार्यान्वयन बनाने वाला डेवलपर क्या सोच रहा था ...
Anibal Sanchez

7

केवल पाठ नोड के साथ बाल तत्वों का सामना करते समय स्वीकृत उत्तर ड्रॉप विशेषताओं में उपयोग की जाने वाली विधि। उदाहरण के लिए:

$xml = '<container><element attribute="123">abcd</element></container>';
print_r(json_decode(json_encode(simplexml_load_string($xml, "SimpleXMLElement", LIBXML_NOCDATA)),1));

Array
(
    [element] => abcd
)

मेरा समाधान (और मेरी इच्छा है कि मैं यहां क्रेडिट दे सकूं क्योंकि मुझे यकीन है कि मैंने इसे कुछ से अनुकूलित किया है):

function XMLtoArray($xml) {
    $previous_value = libxml_use_internal_errors(true);
    $dom = new DOMDocument('1.0', 'UTF-8');
    $dom->preserveWhiteSpace = false; 
    $dom->loadXml($xml);
    libxml_use_internal_errors($previous_value);
    if (libxml_get_errors()) {
        return [];
    }
    return DOMtoArray($dom);
}

function DOMtoArray($root) {
    $result = array();

    if ($root->hasAttributes()) {
        $attrs = $root->attributes;
        foreach ($attrs as $attr) {
            $result['@attributes'][$attr->name] = $attr->value;
        }
    }

    if ($root->hasChildNodes()) {
        $children = $root->childNodes;
        if ($children->length == 1) {
            $child = $children->item(0);
            if (in_array($child->nodeType,[XML_TEXT_NODE,XML_CDATA_SECTION_NODE])) {
                $result['_value'] = $child->nodeValue;
                return count($result) == 1
                    ? $result['_value']
                    : $result;
            }

        }
        $groups = array();
        foreach ($children as $child) {
            if (!isset($result[$child->nodeName])) {
                $result[$child->nodeName] = DOMtoArray($child);
            } else {
                if (!isset($groups[$child->nodeName])) {
                    $result[$child->nodeName] = array($result[$child->nodeName]);
                    $groups[$child->nodeName] = 1;
                }
                $result[$child->nodeName][] = DOMtoArray($child);
            }
        }
    }
    return $result;
}

$xml = '
    <aaaa Version="1.0">
       <bbb>
         <cccc>
           <dddd id="123" />
           <eeee name="john" age="24" />
           <ffff type="employee">Supervisor</ffff>
         </cccc>
       </bbb>
    </aaaa>
';
print_r(XMLtoArray($xml));

Array
(
    [aaaa] => Array
        (
            [@attributes] => Array
                (
                    [Version] => 1.0
                )

            [bbb] => Array
                (
                    [cccc] => Array
                        (
                            [dddd] => Array
                                (
                                    [@attributes] => Array
                                        (
                                            [id] => 123
                                        )

                                )

                            [eeee] => Array
                                (
                                    [@attributes] => Array
                                        (
                                            [name] => john
                                            [age] => 24
                                        )

                                )

                            [ffff] => Array
                                (
                                    [@attributes] => Array
                                        (
                                            [type] => employee
                                        )

                                    [_value] => Supervisor
                                )

                        )

                )

        )

)

2

XML करने के लिए ऐरे

अधिक विवरण https://github.com/sapankumarmohanty/lamp/blob/master/Crate-XML-2-Array

// एक्सएमएल को एरे में बदलें और एसओएपी एक्सएमएल को एरे में

function xml2array($contents, $get_attributes = 1, $priority = 'tag')
    {
        if (!$contents) return array();
        if (!function_exists('xml_parser_create')) {
            // print "'xml_parser_create()' function not found!";
            return array();
        }
        // Get the XML parser of PHP - PHP must have this module for the parser to work
        $parser = xml_parser_create('');
        xml_parser_set_option($parser, XML_OPTION_TARGET_ENCODING, "UTF-8"); // http://minutillo.com/steve/weblog/2004/6/17/php-xml-and-character-encodings-a-tale-of-sadness-rage-and-data-loss
        xml_parser_set_option($parser, XML_OPTION_CASE_FOLDING, 0);
        xml_parser_set_option($parser, XML_OPTION_SKIP_WHITE, 1);
        xml_parse_into_struct($parser, trim($contents) , $xml_values);
        xml_parser_free($parser);
        if (!$xml_values) return; //Hmm...
        // Initializations
        $xml_array = array();
        $parents = array();
        $opened_tags = array();
        $arr = array();
        $current = & $xml_array; //Refference
        // Go through the tags.
        $repeated_tag_index = array(); //Multiple tags with same name will be turned into an array
        foreach($xml_values as $data) {
            unset($attributes, $value); //Remove existing values, or there will be trouble
            // This command will extract these variables into the foreach scope
            // tag(string), type(string), level(int), attributes(array).
            extract($data); //We could use the array by itself, but this cooler.
            $result = array();
            $attributes_data = array();
            if (isset($value)) {
                if ($priority == 'tag') $result = $value;
                else $result['value'] = $value; //Put the value in a assoc array if we are in the 'Attribute' mode
            }
            // Set the attributes too.
            if (isset($attributes) and $get_attributes) {
                foreach($attributes as $attr => $val) {                                   
                                    if ( $attr == 'ResStatus' ) {
                                        $current[$attr][] = $val;
                                    }
                    if ($priority == 'tag') $attributes_data[$attr] = $val;
                    else $result['attr'][$attr] = $val; //Set all the attributes in a array called 'attr'
                }
            }
            // See tag status and do the needed.
                        //echo"<br/> Type:".$type;
            if ($type == "open") { //The starting of the tag '<tag>'
                $parent[$level - 1] = & $current;
                if (!is_array($current) or (!in_array($tag, array_keys($current)))) { //Insert New tag
                    $current[$tag] = $result;
                    if ($attributes_data) $current[$tag . '_attr'] = $attributes_data;
                                        //print_r($current[$tag . '_attr']);
                    $repeated_tag_index[$tag . '_' . $level] = 1;
                    $current = & $current[$tag];
                }
                else { //There was another element with the same tag name
                    if (isset($current[$tag][0])) { //If there is a 0th element it is already an array
                        $current[$tag][$repeated_tag_index[$tag . '_' . $level]] = $result;
                        $repeated_tag_index[$tag . '_' . $level]++;
                    }
                    else { //This section will make the value an array if multiple tags with the same name appear together
                        $current[$tag] = array(
                            $current[$tag],
                            $result
                        ); //This will combine the existing item and the new item together to make an array
                        $repeated_tag_index[$tag . '_' . $level] = 2;
                        if (isset($current[$tag . '_attr'])) { //The attribute of the last(0th) tag must be moved as well
                            $current[$tag]['0_attr'] = $current[$tag . '_attr'];
                            unset($current[$tag . '_attr']);
                        }
                    }
                    $last_item_index = $repeated_tag_index[$tag . '_' . $level] - 1;
                    $current = & $current[$tag][$last_item_index];
                }
            }
            elseif ($type == "complete") { //Tags that ends in 1 line '<tag />'
                // See if the key is already taken.
                if (!isset($current[$tag])) { //New Key
                    $current[$tag] = $result;
                    $repeated_tag_index[$tag . '_' . $level] = 1;
                    if ($priority == 'tag' and $attributes_data) $current[$tag . '_attr'] = $attributes_data;
                }
                else { //If taken, put all things inside a list(array)
                    if (isset($current[$tag][0]) and is_array($current[$tag])) { //If it is already an array...
                        // ...push the new element into that array.
                        $current[$tag][$repeated_tag_index[$tag . '_' . $level]] = $result;
                        if ($priority == 'tag' and $get_attributes and $attributes_data) {
                            $current[$tag][$repeated_tag_index[$tag . '_' . $level] . '_attr'] = $attributes_data;
                        }
                        $repeated_tag_index[$tag . '_' . $level]++;
                    }
                    else { //If it is not an array...
                        $current[$tag] = array(
                            $current[$tag],
                            $result
                        ); //...Make it an array using using the existing value and the new value
                        $repeated_tag_index[$tag . '_' . $level] = 1;
                        if ($priority == 'tag' and $get_attributes) {
                            if (isset($current[$tag . '_attr'])) { //The attribute of the last(0th) tag must be moved as well
                                $current[$tag]['0_attr'] = $current[$tag . '_attr'];
                                unset($current[$tag . '_attr']);
                            }
                            if ($attributes_data) {
                                $current[$tag][$repeated_tag_index[$tag . '_' . $level] . '_attr'] = $attributes_data;
                            }
                        }
                        $repeated_tag_index[$tag . '_' . $level]++; //0 and 1 index is already taken
                    }
                }
            }
            elseif ($type == 'close') { //End of tag '</tag>'
                $current = & $parent[$level - 1];
            }
        }
        return ($xml_array);
    }
    
    // Let's call the this above function xml2array
    
    xml2array($xmlContent, $get_attributes = 3, $priority = 'tag'); // it will work 100% if not ping me @skype: sapan.mohannty
    
//  Enjoy coding

0

मुझे यह सवाल पसंद आया और कुछ उत्तर मेरे लिए मददगार थे, लेकिन मुझे xml को एक वर्चस्व वाली सरणी में बदलने की आवश्यकता है, इसलिए मैं अपना समाधान पोस्ट करूंगा, हो सकता है कि किसी को बाद में इसकी आवश्यकता हो:

<?php
$xml = json_decode(json_encode((array)simplexml_load_string($xml)),1);
$finalItem = getChild($xml);
var_dump($finalItem);

function getChild($xml, $finalItem = []){
    foreach($xml as $key=>$value){
        if(!is_array($value)){
            $finalItem[$key] = $value;
        }else{
            $finalItem = getChild($value, $finalItem);
        }
    }
    return $finalItem;
}
?>  
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.