используйте var_dump, вы увидите структуру xml как ключи внутри объекта.
Волхв
3
У меня есть некоторые [CDATA[TEXT]]внутри некоторые элементы, и они не разбираются с этим. Он разбирает это как SimpleXMLElement Object. Любой обходной путь для этого?
Вы можете столкнуться с проблемами в разделах CDATA (возвращая всегда ноль). В качестве решения попробуйте $ xml = simplexml_load_string ($ xmlstring, null, LIBXML_NOCDATA); $ json = json_encode ($ xml); $ array = json_decode ($ json, TRUE); (см. stackoverflow.com/a/2970701/413531 ) // блин ... есть ли способ добавить новые строки в комментарии?
Хирнхамстер
4
Мы делаем то же самое, но с simplexml_load_file, и он работает нормально. Спасибо
Thermech
2
Для чего нужен второй параметр (ИСТИНА)?
Мансур Фахад
3
@MansourFahad В json_decode вы можете использовать необязательный второй параметр as TRUE(обычно по умолчанию FALSE) для преобразования входных данных JSON в ассоциативный массив.
Джейк Бэтмен,
16
@ Измаил Мигель, слишком много кода? Тот факт, что вы помещаете все эти функции в одну строку, не означает, что вы используете меньше кода. Это может выглядеть более компактно, но это происходит за счет читабельности.
Jage
44
Преобразование XML-строки ( $buffer) в упрощенный массив без учета атрибутов и группирование дочерних элементов с одинаковыми именами:
В PHP 7 я должен был добавить это: && (is_countable($node) && 1 === count($node) ? $node = array($node) : 1)Но я получил сообщение об ошибке в следующей строке: [] operator not supported for strings.
andreshg112
@ andreshg112: я не могу воспроизвести (работает стабильно PHP 5.3.0 - 7.4.0), поведение не изменилось с давних
hakre
может быть, это из-за моего файла KML (это XML). Я не могу поделиться этим. Я уже импортировал это, но я должен был сделать это другим способом.
andreshg112
Возможно, вы беспокоитесь о пространствах имен XML. Пример только для частей без пространства имен (или по умолчанию, я иногда смешиваю это).
если вы приведете к массиву, вам не нужно json_encodeи json_decode.
Исмаэль Мигель
11
@ В теории Исмаэля приведение к массиву должно быть достаточно. На практике нам нужно также привести все листовые узлы, которые также являются объектами. Наивное приведение оставляет листья как объекты SimpleXML. json_encode рекурсивно сохраняет много работы.
Питер Меллетт
2
Если в вашей $arrayпеременной нет текстовых значений , возможно, это связано с CDATA. Чтобы решить эту проблему, загрузите XML с: new SimpleXMLElement($xml, LIBXML_NOCDATA).
Это не работает так. Этот код не будет конвертировать даже простой 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
// Преобразование XML в массив и SOAP XML в массив
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 arrayforeach($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 nameif(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
Мне понравился этот вопрос, и некоторые ответы были полезны для меня, но мне нужно преобразовать xml в один массив доминирования, поэтому я опубликую свое решение, возможно, кому-то это понадобится позже:
Ответы:
Другой вариант - расширение SimpleXML (я полагаю, оно входит в стандартную комплектацию большинства установок php.)
http://php.net/manual/en/book.simplexml.php
Синтаксис выглядит примерно так для вашего примера
источник
[CDATA[TEXT]]
внутри некоторые элементы, и они не разбираются с этим. Он разбирает это какSimpleXMLElement Object
. Любой обходной путь для этого?легко!
источник
TRUE
(обычно по умолчаниюFALSE
) для преобразования входных данных JSON в ассоциативный массив.Преобразование XML-строки (
$buffer
) в упрощенный массив без учета атрибутов и группирование дочерних элементов с одинаковыми именами:Результат:
Если вы также хотите иметь атрибуты, они доступны через JSON-кодирование / декодирование SimpleXMLElement. Часто это самое простое и быстрое решение:
Результат:
Обратите внимание, что все эти методы работают только в пространстве имен документа XML.
источник
&& (is_countable($node) && 1 === count($node) ? $node = array($node) : 1)
Но я получил сообщение об ошибке в следующей строке:[] operator not supported for strings
.источник
json_encode
иjson_decode
.$array
переменной нет текстовых значений , возможно, это связано с CDATA. Чтобы решить эту проблему, загрузите XML с:new SimpleXMLElement($xml, LIBXML_NOCDATA)
.<?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Удивлен никто не упомянул
xml_parse_into_struct
:источник
См. Https://github.com/gaarf/XML-string-to-PHP-array/blob/master/xmlstr_to_array.php
источник
Метод, используемый в принятых атрибутах отбрасывания ответа при обнаружении дочерних элементов только с текстовым узлом. Например:
Мое решение (и я хотел бы отдать должное здесь, потому что я уверен, что я кое-что адаптировал):
источник
XML в массив
Более подробная информация Посетите https://github.com/sapankumarmohanty/lamp/blob/master/Crate-XML-2-Array
// Преобразование XML в массив и SOAP XML в массив
источник
Мне понравился этот вопрос, и некоторые ответы были полезны для меня, но мне нужно преобразовать xml в один массив доминирования, поэтому я опубликую свое решение, возможно, кому-то это понадобится позже:
источник