Как преобразовать массив в SimpleXML

Ответы:

209

короткий:

<?php

$test_array = array (
  'bla' => 'blub',
  'foo' => 'bar',
  'another_array' => array (
    'stack' => 'overflow',
  ),
);
$xml = new SimpleXMLElement('<root/>');
array_walk_recursive($test_array, array ($xml, 'addChild'));
print $xml->asXML();

результаты в

<?xml version="1.0"?>
<root>
  <blub>bla</blub>
  <bar>foo</bar>
  <overflow>stack</overflow>
</root>

ключи и значения меняются местами - это можно исправить с array_flip()помощью метода array_walk. array_walk_recursiveтребует PHP 5. вы можете использовать array_walkвместо этого, но вы не получите 'stack' => 'overflow'в XML тогда.

Топор.
источник
53
Это не будет работать, если $ test_array имеет значение «more_another_array», например «another_array», поскольку ключ «another_array» не преобразуется. Следовательно, у вас будет несколько <overflow> stack </ overflow> '.
Понять
11
Не array_flipбудет работать, так как не может переворачивать массивы (как another_arrayвнутри основного массива).
Лоде
14
Где находится элемент "another_array" xml? Все
сплющено
2
Отлично работал, когда я добавил array_flip перед array_walk_recursive. Спасибо.
Майк Перселл
12
Downvoting, потому что array_flipработает, только если массив не содержит одинаковых значений.
Мартейн
388

Вот код php 5.2, который преобразует массив любой глубины в документ XML:

Array
(
    ['total_stud']=> 500
    [0] => Array
        (
            [student] => Array
                (
                    [id] => 1
                    [name] => abc
                    [address] => Array
                        (
                            [city]=>Pune
                            [zip]=>411006
                        )                       
                )
        )
    [1] => Array
        (
            [student] => Array
                (
                    [id] => 2
                    [name] => xyz
                    [address] => Array
                        (
                            [city]=>Mumbai
                            [zip]=>400906
                        )   
                )

        )
)

сгенерированный XML будет выглядеть так:

<?xml version="1.0"?>
<student_info>
    <total_stud>500</total_stud>
    <student>
        <id>1</id>
        <name>abc</name>
        <address>
            <city>Pune</city>
            <zip>411006</zip>
        </address>
    </student>
    <student>
        <id>1</id>
        <name>abc</name>
        <address>
            <city>Mumbai</city>
            <zip>400906</zip>
        </address>
    </student>
</student_info>

PHP-фрагмент

<?php
// function defination to convert array to xml
function array_to_xml( $data, &$xml_data ) {
    foreach( $data as $key => $value ) {
        if( is_array($value) ) {
            if( is_numeric($key) ){
                $key = 'item'.$key; //dealing with <0/>..<n/> issues
            }
            $subnode = $xml_data->addChild($key);
            array_to_xml($value, $subnode);
        } else {
            $xml_data->addChild("$key",htmlspecialchars("$value"));
        }
     }
}

// initializing or creating array
$data = array('total_stud' => 500);

// creating object of SimpleXMLElement
$xml_data = new SimpleXMLElement('<?xml version="1.0"?><data></data>');

// function call to convert array to xml
array_to_xml($data,$xml_data);

//saving generated xml file; 
$result = $xml_data->asXML('/file/path/name.xml');

?>

Документация по SimpleXMLElement::asXML использованию в этом фрагменте

Hanmant
источник
мы использовали это 'echo $ xml_student_info-> asXML ();' чтобы отобразить этот XML напрямую, но он работает для больших массивов данных. если данные массива меньше десяти элементов, формат xml не отображается, а только в простом формате. мы использовали эту функцию header («Content-type: text / xml»); отображать формат в XML.
Бхарат Ходвадия
1
В этом примере явным образом экранируются специальные символы в текстовых данных элемента с использованием htmlspecialchars, но SimpleXMLElement :: addChild автоматически переводит специальные символы xml в свои объекты char, поэтому htmlspecialchars можно не указывать. Интересно, что это, по-видимому, не приводит к получению дважды экранированных данных.
Мбайтон
1
Пустые значения массива [(string) ""] будут заменены на пустой SimpleXML-узел вместо того, чтобы оставаться пустым.
Джонатан
3
@ Алекс, ваш Edit # 5 делает пример неудачным. Он вставляет <item $ x> перед каждой записью <student>, что делает вывод XML не тем, что задумал автор. Возможно, приведите пример проблемы, которую вы пытаетесь устранить, и мы можем найти другое решение для обоих случаев. Мне потребовалось некоторое время, прежде чем я понял, что авторский код был изменен.
Николас Блазген
1
Если опубликовать два ответа, этот измененный ответ сломал мои запросы, как он добавил <itemN></itemN>. Эта редакция: stackoverflow.com/revisions/5965940/2 была моим победителем
zanderwar
124

Ответы, представленные здесь, только преобразовывают массив в XML с узлами, вы не можете установить атрибуты. Я написал функцию php, которая позволяет вам преобразовывать массив в php, а также устанавливать атрибуты для определенных узлов в xml. Недостатком здесь является то, что вы должны создать массив определенным образом с несколькими соглашениями (только если вы хотите использовать атрибуты)

Следующий пример позволит вам также установить атрибуты в XML.

Источник можно найти здесь: https://github.com/digitickets/lalit/blob/master/src/Array2XML.php

<?php    
$books = array(
    '@attributes' => array(
        'type' => 'fiction'
    ),
    'book' => array(
        array(
            '@attributes' => array(
                'author' => 'George Orwell'
            ),
            'title' => '1984'
        ),
        array(
            '@attributes' => array(
                'author' => 'Isaac Asimov'
            ),
            'title' => 'Foundation',
            'price' => '$15.61'
        ),
        array(
            '@attributes' => array(
                'author' => 'Robert A Heinlein'
            ),
            'title' => 'Stranger in a Strange Land',
            'price' => array(
                '@attributes' => array(
                    'discount' => '10%'
                ),
                '@value' => '$18.00'
            )
        )
    )
);
/* creates 
<books type="fiction">
  <book author="George Orwell">
    <title>1984</title>
  </book>
  <book author="Isaac Asimov">
    <title>Foundation</title>
    <price>$15.61</price>
  </book>
  <book author="Robert A Heinlein">
    <title>Stranger in a Strange Land</title>
    <price discount="10%">$18.00</price>
  </book>
</books>
*/
?>
Лалита
источник
9
Меня беспокоит, что никто не отреагировал на это. Этот класс действительно полезен, так как он делает противоположное тому, что сгенерирует simpleXMLElement. Так что это дает вам возможность использовать SimpleXMLElement в обоих направлениях.
FMaz008
4
Я бы отметил это как ответ, а не текущий. Текущий ответ не строит рекурсивные массивы
Александр IY
2
Хороший класс. Я изменил линию 128 , if(!is_array($arr)) {чтобы if(!is_array($arr) && $arr !== '') {таким образом , что она не будет добавлять новый текстовый узел для пустых строк и , следовательно , сохраняет сокращенную пустой формат тегов т.е. 'tag'=>''это <tag/>вместо<tag></tag>
user1433150
Это лучший ответ на данный момент. Также это имеет правильную структуру нескольких элементов с одинаковым ключом: 1-й - это ключ имени узла, затем он содержит массив с числовыми ключами. (противоположность ответа Hanmant)
Васил Попов
1
Нашел github от автора @Legionar github.com/digitickets/lalit/blob/master/src/Array2XML.php
Дэрил Тео
57

Я нашел все ответы, чтобы использовать слишком много кода. Вот простой способ сделать это:

function to_xml(SimpleXMLElement $object, array $data)
{   
    foreach ($data as $key => $value) {
        if (is_array($value)) {
            $new_object = $object->addChild($key);
            to_xml($new_object, $value);
        } else {
            // if the key is an integer, it needs text with it to actually work.
            if ($key == (int) $key) {
                $key = "key_$key";
            }

            $object->addChild($key, $value);
        }   
    }   
}   

Тогда это просто вопрос отправки массива в функцию, которая использует рекурсию, поэтому он будет обрабатывать многомерный массив:

$xml = new SimpleXMLElement('<rootTag/>');
to_xml($xml, $my_array);

Теперь $ xml содержит красивый объект XML, основанный на вашем массиве, именно так, как вы его написали.

print $xml->asXML();
Фрэнсис Льюис
источник
9
Я люблю это решение больше всего. Хотя, было бы неплохо добавить тест на цифровые клавиши, как: if ( is_numeric( $key ) ) $key = "numeric_$key"; .
Wout
@ Wout Хороший улов. Добавлен. Я сделал проверку приведения типа вместо is_numeric, потому что is_numeric может дать некоторые, хотя и технически ожидаемые результаты, которые действительно отбросят вас.
Фрэнсис Льюис
Я использую эту функцию, но изменил $xml = new SimpleXMLElement('<?xml version="1.0" encoding="UTF-8" ?><rootTag/>');для правильной кодировки UTF-8.
Даантье
Мне также больше всего нравится это решение, простое оно делает :-) Одно замечание: вы можете захотеть изменить $object->addChild($key, $value);его, $object->addChild($key, htmlspecialchars($value));чтобы предотвратить сбой, когда $ value содержит такие символы, как «&», для которых требуется XML-кодировка.
Лев
38
<? PHP
функция array_to_xml (массив $ arr, SimpleXMLElement $ xml)
{
    foreach ($ arr as $ k => $ v) {
        is_array ($ v)
            ? array_to_xml ($ v, $ xml-> addChild ($ k))
            : $ xml-> addChild ($ k, $ v);
    }
    вернуть $ xml;
}

$ test_array = array (
    'bla' => 'blub',
    'foo' => 'bar',
    'another_array' => массив (
        'stack' => 'overflow',
    ),
);

echo array_to_xml ($ test_array, new SimpleXMLElement ('<root />')) -> asXML ();
onokazu
источник
1
это не удастся, если ваш массив содержит внутренний массив с числовыми индексами. <0> ... </ 0> не является допустимым XML.
Adriano Varoli Piazza
@AdrianoVaroliPiazza просто добавьте что-то вроде $k = (is_numeric($k)) ? 'item' : $k;внутриforeach()
AlienWebguy
Если один из ключей в массиве называется «тело», он не работает - точнее, ключ игнорируется и пересекается. Пытаюсь выяснить почему.
Бамбакс
@Bambax Единственная причина, по которой я могу придумать, заключается в том, что XML-код анализируется как HTML на более позднем этапе.
Brilliand
16

Из PHP 5.4

function array2xml($data, $root = null){
    $xml = new SimpleXMLElement($root ? '<' . $root . '/>' : '<root/>');
    array_walk_recursive($data, function($value, $key)use($xml){
        $xml->addChild($key, $value);
    });
    return $xml->asXML();
}
user492589
источник
Это похоже на прямую копию выбранного ответа, просто помещенную в функцию.
Фаберест
Я бы добавил htmlspecialchars () в часть addChild, например: $ xml-> addChild ($ key, htmlspecialchars ($ value));
Tyreal
15

Еще одно улучшение:

/**
* Converts an array to XML
*
* @param array $array
* @param SimpleXMLElement $xml
* @param string $child_name
*
* @return SimpleXMLElement $xml
*/
public function arrayToXML($array, SimpleXMLElement $xml, $child_name)
{
    foreach ($array as $k => $v) {
        if(is_array($v)) {
            (is_int($k)) ? $this->arrayToXML($v, $xml->addChild($child_name), $v) : $this->arrayToXML($v, $xml->addChild(strtolower($k)), $child_name);
        } else {
            (is_int($k)) ? $xml->addChild($child_name, $v) : $xml->addChild(strtolower($k), $v);
        }
    }

    return $xml->asXML();
}

Использование:

$this->arrayToXML($array, new SimpleXMLElement('<root/>'), 'child_name_to_replace_numeric_integers');
Syl
источник
Спасибо! Ваша функция возвращает точное содержимое любого n-мерного массива.
besciualex
12

Вот моя запись, простая и чистая ..

function array2xml($array, $xml = false){
    if($xml === false){
        $xml = new SimpleXMLElement('<root/>');
    }
    foreach($array as $key => $value){
        if(is_array($value)){
            array2xml($value, $xml->addChild($key));
        }else{
            $xml->addChild($key, $value);
        }
    }
    return $xml->asXML();
}


header('Content-type: text/xml');
print array2xml($array);
Франс ван Ассельт
источник
8

Так или иначе ... Я взял код onokazu (спасибо!) И добавил возможность повторять теги в XML, он также поддерживает атрибуты, надеюсь, кто-то найдет его полезным!

 <?php

function array_to_xml(array $arr, SimpleXMLElement $xml) {
        foreach ($arr as $k => $v) {

            $attrArr = array();
            $kArray = explode(' ',$k);
            $tag = array_shift($kArray);

            if (count($kArray) > 0) {
                foreach($kArray as $attrValue) {
                    $attrArr[] = explode('=',$attrValue);                   
                }
            }

            if (is_array($v)) {
                if (is_numeric($k)) {
                    array_to_xml($v, $xml);
                } else {
                    $child = $xml->addChild($tag);
                    if (isset($attrArr)) {
                        foreach($attrArr as $attrArrV) {
                            $child->addAttribute($attrArrV[0],$attrArrV[1]);
                        }
                    }                   
                    array_to_xml($v, $child);
                }
            } else {
                $child = $xml->addChild($tag, $v);
                if (isset($attrArr)) {
                    foreach($attrArr as $attrArrV) {
                        $child->addAttribute($attrArrV[0],$attrArrV[1]);
                    }
                }
            }               
        }

        return $xml;
    }

        $test_array = array (
          'bla' => 'blub',
          'foo' => 'bar',
          'another_array' => array (
            array('stack' => 'overflow'),
            array('stack' => 'overflow'),
            array('stack' => 'overflow'),
          ),
          'foo attribute1=value1 attribute2=value2' => 'bar',
        );  

        $xml = array_to_xml($test_array, new SimpleXMLElement('<root/>'))->asXML();

        echo "$xml\n";
        $dom = new DOMDocument;
        $dom->preserveWhiteSpace = FALSE;
        $dom->loadXML($xml);
        $dom->formatOutput = TRUE;
        echo $dom->saveXml();
    ?>
CodePT
источник
Может быть полезно прокомментировать ваши изменения, чтобы сделать код более понятным; еще, приятное дополнение
StormeHawke
Это сработало для меня с WP All Export. У меня было немного изменить is_numeric часть: if (is_numeric($k)) { $i = $k + 1; $child = $xml->addChild("_$i"); array_to_xml($v, $child); }
Успешность
4

Я использую пару функций, которые я написал некоторое время назад, чтобы сгенерировать xml для передачи туда и обратно из PHP, jQuery и т. Д. Ни один из дополнительных каркасов не используется, просто генерируется строка, которую затем можно использовать с SimpleXML (или другим каркасом ) ...

Если это кому-нибудь пригодится, пожалуйста, используйте его :)

function generateXML($tag_in,$value_in="",$attribute_in=""){
    $return = "";
    $attributes_out = "";
    if (is_array($attribute_in)){
        if (count($attribute_in) != 0){
            foreach($attribute_in as $k=>$v):
                $attributes_out .= " ".$k."=\"".$v."\"";
            endforeach;
        }
    }
    return "<".$tag_in."".$attributes_out.((trim($value_in) == "") ? "/>" : ">".$value_in."</".$tag_in.">" );
}

function arrayToXML($array_in){
    $return = "";
    $attributes = array();
    foreach($array_in as $k=>$v):
        if ($k[0] == "@"){
            // attribute...
            $attributes[str_replace("@","",$k)] = $v;
        } else {
            if (is_array($v)){
                $return .= generateXML($k,arrayToXML($v),$attributes);
                $attributes = array();
            } else if (is_bool($v)) {
                $return .= generateXML($k,(($v==true)? "true" : "false"),$attributes);
                $attributes = array();
            } else {
                $return .= generateXML($k,$v,$attributes);
                $attributes = array();
            }
        }
    endforeach;
    return $return;
}   

Всем любви :)

Нейл Инглиш
источник
4

Я хотел код, который будет принимать все элементы внутри массива и обрабатывать их как атрибуты, а все массивы как подэлементы.

Так что для чего-то вроде

array (
'row1' => array ('head_element' =>array("prop1"=>"some value","prop2"=>array("empty"))),
"row2"=> array ("stack"=>"overflow","overflow"=>"overflow")
);

Я бы получил что-то вроде этого

<?xml version="1.0" encoding="utf-8"?>
<someRoot>
  <row1>
    <head_element prop1="some value">
      <prop2 0="empty"/>
    </head_element>
  </row1>
  <row2 stack="overflow" overflow="stack"/>
 </someRoot>

Чтобы добиться этого, код приведен ниже, но будьте очень осторожны, он рекурсивен и может фактически вызвать переполнение стека :)

function addElements(&$xml,$array)
{
$params=array();
foreach($array as $k=>$v)
{
    if(is_array($v))
        addElements($xml->addChild($k), $v);
    else $xml->addAttribute($k,$v);
}

}
function xml_encode($array)
{
if(!is_array($array))
    trigger_error("Type missmatch xml_encode",E_USER_ERROR);
$xml=new SimpleXMLElement('<?xml version=\'1.0\' encoding=\'utf-8\'?><'.key($array).'/>');
addElements($xml,$array[key($array)]);
return $xml->asXML();
} 

Возможно, вы захотите добавить проверки длины массива, чтобы некоторый элемент был установлен внутри части данных, а не как атрибут.

lcornea
источник
4

Основываясь на всем остальном, обрабатывает числовые индексы + атрибуты с помощью префикса @и может внедрить xml в существующие узлы:

Код

function simple_xmlify($arr, SimpleXMLElement $root = null, $el = 'x') {
    // based on, among others http://stackoverflow.com/a/1397164/1037948

    if(!isset($root) || null == $root) $root = new SimpleXMLElement('<' . $el . '/>');

    if(is_array($arr)) {
        foreach($arr as $k => $v) {
            // special: attributes
            if(is_string($k) && $k[0] == '@') $root->addAttribute(substr($k, 1),$v);
            // normal: append
            else simple_xmlify($v, $root->addChild(
                    // fix 'invalid xml name' by prefixing numeric keys
                    is_numeric($k) ? 'n' . $k : $k)
                );
        }
    } else {
        $root[0] = $arr;
    }

    return $root;
}//--   fn  simple_xmlify

использование

// lazy declaration via "queryparam"
$args = 'hello=4&var[]=first&var[]=second&foo=1234&var[5]=fifth&var[sub][]=sub1&var[sub][]=sub2&var[sub][]=sub3&var[@name]=the-name&var[@attr2]=something-else&var[sub][@x]=4.356&var[sub][@y]=-9.2252';
$q = array();
parse_str($val, $q);

$xml = simple_xmlify($q); // dump $xml, or...
$result = get_formatted_xml($xml); // see below

результат

<?xml version="1.0"?>
<x>
  <hello>4</hello>
  <var name="the-name" attr2="something-else">
    <n0>first</n0>
    <n1>second</n1>
    <n5>fifth</n5>
    <sub x="4.356" y="-9.2252">
      <n0>sub1</n0>
      <n1>sub2</n1>
      <n2>sub3</n2>
    </sub>
  </var>
  <foo>1234</foo>
</x>

Бонус: форматирование XML

function get_formatted_xml(SimpleXMLElement $xml, $domver = null, $preserveWhitespace = true, $formatOutput = true) {
    // http://stackoverflow.com/questions/1191167/format-output-of-simplexml-asxml

    // create new wrapper, so we can get formatting options
    $dom = new DOMDocument($domver);
    $dom->preserveWhiteSpace = $preserveWhitespace;
    $dom->formatOutput = $formatOutput;
    // now import the xml (converted to dom format)
    /*
    $ix = dom_import_simplexml($xml);
    $ix = $dom->importNode($ix, true);
    $dom->appendChild($ix);
    */
    $dom->loadXML($xml->asXML());

    // print
    return $dom->saveXML();
}//--   fn  get_formatted_xml
drzaus
источник
Обновленная версия , которая повторяется в качестве дочерних элементов , а не числовые теги: github.com/zaus/forms-3rdparty-xpost/blob/...
drzaus
3

Вот функция, которая добилась цели для меня:

Просто позвоните с чем-то вроде

echo arrayToXml("response",$arrayIWantToConvert);
function arrayToXml($thisNodeName,$input){
        if(is_numeric($thisNodeName))
            throw new Exception("cannot parse into xml. remainder :".print_r($input,true));
        if(!(is_array($input) || is_object($input))){
            return "<$thisNodeName>$input</$thisNodeName>";
        }
        else{
            $newNode="<$thisNodeName>";
            foreach($input as $key=>$value){
                if(is_numeric($key))
                    $key=substr($thisNodeName,0,strlen($thisNodeName)-1);
                $newNode.=arrayToXml3($key,$value);
            }
            $newNode.="</$thisNodeName>";
            return $newNode;
        }
    }
Майк
источник
3

Вы можете использовать XMLParser , над которым я работал.

$xml = XMLParser::encode(array(
    'bla' => 'blub',
    'foo' => 'bar',
    'another_array' => array (
        'stack' => 'overflow',
    )
));
// @$xml instanceof SimpleXMLElement
echo $xml->asXML();

В результате:

<?xml version="1.0"?>
<root>
    <bla>blub</bla>
    <foo>bar</foo>
    <another_array>
        <stack>overflow</stack>
    </another_array>
</root>
jtrumbull
источник
3

Я нашел это решение похоже на исходную проблему

<?php

$test_array = array (
  'bla' => 'blub',
  'foo' => 'bar',
  'another_array' => array (
    'stack' => 'overflow',
  ),
);

class NoSimpleXMLElement extends SimpleXMLElement {
 public function addChild($name,$value) {
  parent::addChild($value,$name);
 }
}
$xml = new NoSimpleXMLElement('<root/>');
array_walk_recursive($test_array, array ($xml, 'addChild'));
print $xml->asXML();
caiofior
источник
3

Большинство из приведенных выше ответов являются правильными. Однако я пришел к этому ответу, который решает проблему совместимости array_walk_recursive, а также проблему числовых ключей. Это также прошло все испытания, которые я сделал:

function arrayToXML(Array $array, SimpleXMLElement &$xml) {

    foreach($array as $key => $value) {

        // None array
        if (!is_array($value)) {
            (is_numeric($key)) ? $xml->addChild("item$key", $value) : $xml->addChild($key, $value);
            continue;
        }   

        // Array
        $xmlChild = (is_numeric($key)) ? $xml->addChild("item$key") : $xml->addChild($key);
        arrayToXML($value, $xmlChild);
    }
}   

Я также добавил тестовый класс для этого, который вы можете найти полезным:

class ArrayToXmlTest extends PHPUnit_Framework_TestCase {

    public function setUp(){ }
    public function tearDown(){ }

    public function testFuncExists() {
        $this->assertTrue(function_exists('arrayToXML'));
    }

    public function testFuncReturnsXml() {
        $array = array(
            'name' => 'ardi',
            'last_name' => 'eshghi',
            'age' => 31,
            'tel' => '0785323435'
        );

        $xmlEl =  new SimpleXMLElement('<root/>');
        arrayToXml($array, $xmlEl);

        $this->assertTrue($xmlEl instanceOf SimpleXMLElement);
    }

    public function testAssocArrayToXml() {

        $array = array(
            'name' => 'ardi',
            'last_name' => 'eshghi',
            'age' => 31,
            'tel' => '0785323435'
        );

        $expectedXmlEl = new SimpleXMLElement('<root/>'); 
        $expectedXmlEl->addChild('name', $array['name']);
        $expectedXmlEl->addChild('last_name', $array['last_name']);
        $expectedXmlEl->addChild('age', $array['age']);
        $expectedXmlEl->addChild('tel', $array['tel']);

        $actualXmlEl =  new SimpleXMLElement('<root/>');
        arrayToXml($array, $actualXmlEl);

        $this->assertEquals($expectedXmlEl->asXML(), $actualXmlEl->asXML());
    }

    public function testNoneAssocArrayToXml() {

        $array = array(
            'ardi',
            'eshghi',
            31,
            '0785323435'
        );

        // Expected xml value
        $expectedXmlEl = new SimpleXMLElement('<root/>'); 
        foreach($array as $key => $value)
            $expectedXmlEl->addChild("item$key", $value);

        // What the function produces       
        $actualXmlEl =  new SimpleXMLElement('<root/>');
        arrayToXml($array, $actualXmlEl);

        $this->assertEquals($expectedXmlEl->asXML(), $actualXmlEl->asXML());
    }

    public function testNestedMixArrayToXml() {

        $testArray = array(
            "goal",
            "nice",
            "funny" => array(
                'name' => 'ardi',
                'tel'   =>'07415517499',
                "vary",
                "fields" => array(
                    'small',
                    'email' => 'ardi.eshghi@gmail.com'
                ),

                'good old days'

            ),

            "notes" => "come on lads lets enjoy this",
            "cast" => array(
                'Tom Cruise',
                'Thomas Muller' => array('age' => 24)
            )
        );

        // Expected xml value
        $expectedXmlEl = new SimpleXMLElement('<root/>'); 
        $expectedXmlEl->addChild('item0', $testArray[0]);
        $expectedXmlEl->addChild('item1', $testArray[1]);
        $childEl = $expectedXmlEl->addChild('funny');
        $childEl->addChild("name", $testArray['funny']['name']);
        $childEl->addChild("tel", $testArray['funny']['tel']);
        $childEl->addChild("item0", "vary");
        $childChildEl = $childEl->addChild("fields");
        $childChildEl->addChild('item0', 'small');
        $childChildEl->addChild('email', $testArray['funny']['fields']['email']);
        $childEl->addChild("item1", 'good old days');
        $expectedXmlEl->addChild('notes', $testArray['notes']);
        $childEl2 = $expectedXmlEl->addChild('cast');
        $childEl2->addChild('item0', 'Tom Cruise');
        $childChildEl2 = $childEl2->addChild('Thomas Muller');
        $childChildEl2->addChild('age', $testArray['cast']['Thomas Muller']['age']);

        // What the function produces       
        $actualXmlEl = new SimpleXMLElement('<root/>');
        arrayToXml($testArray, $actualXmlEl);

        $this->assertEquals($expectedXmlEl->asXML(), $actualXmlEl->asXML());
    }
}      
Арди
источник
3

другое решение:

$marray=array(....);
$options = array(
                "encoding" => "UTF-8",
                "output_type" => "xml", 
                "version" => "simple",
                "escaping" => array("non-ascii, on-print, markup")
                );
$xmlres = xmlrpc_encode_request('root', $marray, $options);
print($xmlres);
Гай Балтар
источник
Это приводит к неожиданному эффекту создания XML в стиле RPC с такими вещами, как methodCall, methodName, скаляры и векторы и т. Д. На самом деле преобразование массива в XML не является простым.
Volomike
3

Если массив является ассоциативным и имеет правильный ключ, было бы легче сначала превратить его в xml. Что-то вроде:

  function array2xml ($array_item) {
    $xml = '';
    foreach($array_item as $element => $value)
    {
        if (is_array($value))
        {
            $xml .= "<$element>".array2xml($value)."</$element>";
        }
        elseif($value == '')
        {
            $xml .= "<$element />";
        }
        else
        {
            $xml .= "<$element>".htmlentities($value)."</$element>";
        }
    }
    return $xml;
}

$simple_xml = simplexml_load_string(array2xml($assoc_array));

Другой путь - сначала создать базовый XML, например

$simple_xml = simplexml_load_string("<array></array>");

а затем для каждой части вашего массива используйте нечто похожее на мой цикл создания текста и вместо этого используйте функции simplexml "addChild" для каждого узла массива.

Я попробую это позже и обновлю этот пост обеими версиями.

Энтони
источник
Этот бит, в котором я упомянул «<массив> </ массив>», заставил меня понять, что строковая версия нуждается в чем-то подобном. В основном массив должен иметь один узел снаружи. Позвольте мне поспать в целом, у меня будет что-то, что сразу же поймает эту первоначальную ошибку.
Энтони
2

Просто отредактируйте функцию выше, когда ключ числовой, добавьте префикс "key_"

// initializing or creating array
$student_info = array(your array data);

// creating object of SimpleXMLElement
$xml_student_info = new SimpleXMLElement("<?xml version=\"1.0\"?><student_info></student_info>");

// function call to convert array to xml
array_to_xml($student,$xml_student_info);

//saving generated xml file
$xml_student_info->asXML('file path and name');


function array_to_xml($student_info, &$xml_student_info) {
     foreach($student_info as $key => $value) {
          if(is_array($value)) {
            if(!is_numeric($key)){
                $subnode = $xml_student_info->addChild("$key");
                array_to_xml($value, $subnode);
            }
            else{
                $subnode = $xml_student_info->addChild("key_$key");
                array_to_xml($value, $subnode);
            }
          }
          else {
               if(!is_numeric($key)){
                    $xml_student_info->addChild("$key","$value");
               }else{
                    $xml_student_info->addChild("key_$key","$value");
               }
          }
     }
}
frankey
источник
1

Вы можете использовать следующую функцию в своем коде напрямую,

    function artoxml($arr, $i=1,$flag=false){
    $sp = "";
    for($j=0;$j<=$i;$j++){
        $sp.=" ";
     }
    foreach($arr as $key=>$val){
        echo "$sp&lt;".$key."&gt;";
        if($i==1) echo "\n";
        if(is_array($val)){
            if(!$flag){echo"\n";}
            artoxml($val,$i+5);
            echo "$sp&lt;/".$key."&gt;\n";
        }else{
              echo "$val"."&lt;/".$key."&gt;\n";
         }
    }

}

Вызовите функцию с первым аргументом в качестве массива, а второй аргумент должен быть равен 1, это будет увеличено для идеального отступа, а третий должен быть истинным.

например, если переменная массива для преобразования равна $ array1, то при вызове вызывающая функция должна быть заключена в <pre>тег.

  artoxml ($ array1,1, правда);   

Пожалуйста, посмотрите исходный код страницы после выполнения файла, потому что символы <и> не будут отображаться на HTML-странице.

JosephVasantPrakash
источник
1
function toXML($data, $obj = false, $dom) {
    $is_first_level = false;
    if($obj === false) {
        $dom = new DomDocument('1.0');
        $obj = $dom;
        $is_first_level = true;
    }

    if(is_array($data)) {
        foreach($data as $key => $item) {
            $this->toXML($item, $obj->appendChild($dom->createElement($key)), $dom);
        }
    }else {
        $obj->appendChild($dom->createTextNode($data));
    }

    if($is_first_level) {
        $obj->formatOutput = true;
        return $obj->saveXML();
    }
    return $obj;
}
Андрей
источник
Это отличный вариант для создания DOMDocument xml. Спасибо @Andrey
altsyset
1
function array2xml(array $data, SimpleXMLElement $object = null, $oldNodeName = 'item')
{
    if (is_null($object)) $object = new SimpleXMLElement('<root/>');
    $isNumbered = true;
    $idx = 0;
    foreach ($data as $key => $x)
        if (is_string($key) || ($idx++ != $key + 0))
            $isNumbered = false;
    foreach ($data as $key => $value)
    {   
        $attribute = preg_match('/^[0-9]/', $key . '') ? $key : null;
        $key = (is_string($key) && !preg_match('/^[0-9]/', $key . '')) ? $key : preg_replace('/s$/', '', $oldNodeName);
        if (is_array($value))
        {
            $new_object = $object->addChild($key);
            if (!$isNumbered && !is_null($attribute)) $new_object->addAttribute('id', $attribute);
            array2xml($value, $new_object, $key);
        }
        else
        {
            if (is_bool($value)) $value = $value ? 'true' : 'false';
            $node = $object->addChild($key, htmlspecialchars($value));
            if (!$isNumbered && !is_null($attribute) && !isset($node->attributes()->id))
                $node->addAttribute('id', $attribute);
        }
    }
    return $object;
}

Эта функция возвращает, например, список тегов <obj> ... </ obj> <obj> ... </ obj> XML для числовых индексов.

Входные данные:

    array(
    'people' => array(
        'dog',
        'cat',
        'life' => array(
            'gum',
            'shoe',
        ),
        'fish',
    ),
    array('yeah'),
)

Вывод:

<root>
    <people>
        <people>dog</people>
        <people>cat</people>
        <life>
            <life>gum</life>
            <life>shoe</life>
        </life>
        <people>fish</people>
        <people>
            <people>yeah</people>
        </people>
    </people>
</root>

Это должно удовлетворить все общие потребности. Возможно, вы можете изменить 3-ю строку на:

$key = is_string($key) ? $key : $oldNodeName . '_' . $key;

или если вы работаете с множественным числом, оканчивающимся на s:

$key = is_string($key) ? $key : preg_replace('/s$/', '', $oldNodeName);
user2381982
источник
1

С FluidXML вы можете сгенерировать, начиная с массива PHP , XML для SimpleXML с ... всего двумя строками кода.

$fluidxml  = fluidxml($array);
$simplexml = simplexml_import_dom($fluidxml->dom());

Пример массива может быть

$array = [ 'doc' => [
              'fruit' => 'orange',
              'cake'  => [
                   '@id' => '123', 
                   '@'   => 'tiramisu' ],
              [ 'pasta' => 'matriciana' ],
              [ 'pasta' => 'boscaiola'  ]
] ];

https://github.com/servo-php/fluidxml

Даниэле Орландо
источник
0

Вы можете использовать xmlrpc_encode для создания XML из массива, если подробный XML не является проблемой. www.php.net/xmlrpc_encode

будьте осторожны, создаваемый xml отличается в случае использования ассоциативных и / или цифровых клавиш

<?php
// /params/param/value/struct/member
// there is a tag "member" for each element
// "member" contains a tag "name". its value is the associative key
$xml1 = xmlrpc_encode(array('a'=>'b','c'=>'d'));
$simplexml1 = simplexml_load_string($xml1);
print_r($xml1);
print_r($simplexml1);

// /params/param/value/array/data
// there is a tag "data" for each element
// "data" doesn't contain the tag "name"
$xml2 = xmlrpc_encode(array('a','b'));
$simplexml2 = simplexml_load_string($xml2);
print_r($xml2);
print_r($simplexml2);
?>
w35l3y
источник
Эта функция не поддерживается и, по сути, не предусмотрена в моих сборках PHP 5.2.16 или PHP 5.3.5. (возвращает «PHP Fatal error: вызов неопределенной функции xmlrpc_encode ()»)
Данортон,
Вы должны раскомментировать следующую строку в php.ini: extension = php_xmlrpc.dll
w35l3y
@ w35l3y Я проверил свои ini. Он даже не содержит этого расширения, и я использую v 5.3.6.
Майк С.
0
function array2xml($array, $xml = false){

    if($xml === false){

        $xml = new SimpleXMLElement('<?xml version=\'1.0\' encoding=\'utf-8\'?><'.key($array).'/>');
        $array = $array[key($array)];

    }
    foreach($array as $key => $value){
        if(is_array($value)){
            $this->array2xml($value, $xml->addChild($key));
        }else{
            $xml->addChild($key, $value);
        }
    }
    return $xml->asXML();
}
Камиль Домбровски
источник
0

Мой ответ, объединяя ответы других. Это должно исправить невозможность компенсации цифровых клавиш:

function array_to_xml($array, $root, $element) {
    $xml = new SimpleXMLElement("<{$root}/>");
    foreach ($array as $value) {
        $elem = $xml->addChild($element);
        xml_recurse_child($elem, $value);
    }
    return $xml;
}

function xml_recurse_child(&$node, $child) {
    foreach ($child as $key=>$value) {
        if(is_array($value)) {
            foreach ($value as $k => $v) {
                if(is_numeric($k)){
                    xml_recurse_child($node, array($key => $v));
                }
                else {
                    $subnode = $node->addChild($key);
                    xml_recurse_child($subnode, $value);
                }
            }
        }
        else {
            $node->addChild($key, $value);
        }
    }   
}

array_to_xml()Функция предполагает , что массив состоит из цифровых клавиш первый. Если массив был начальный элемент, то сбросит foreach()и $elemзаявление от array_to_xml()функции и просто передать $xmlвместо этого.

refeyd
источник
0

Я бы прокомментировал второй ответ с наибольшим количеством голосов, потому что он не сохраняет структуру и генерирует плохой XML, если есть численно проиндексированные внутренние массивы.

Я разработал свою собственную версию на ее основе, потому что мне нужен был простой конвертер между json и xml независимо от структуры данных. Моя версия сохраняет информацию о числовых ключах и структуру исходного массива. Он создает элементы для числовых индексированных значений, оборачивая значения в именованные элементы с ключом-атрибутом, который содержит числовой ключ.

Например

array('test' => array(0 => 'some value', 1 => 'other'))

превращается в

<test><value key="0">some value</value><value key="1">other</value></test>

Моя версия функции array_to_xml (надеюсь, она кому-нибудь поможет :)

function array_to_xml($arr, &$xml) {
    foreach($arr as $key => $value) {
        if(is_array($value)) {
            if(!is_numeric($key)){
                $subnode = $xml->addChild("$key");
            } else {
                $subnode = $xml->addChild("value");
                $subnode->addAttribute('key', $key);                    
            }
            array_to_xml($value, $subnode);
        }
        else {
            if (is_numeric($key)) {
                $xml->addChild("value", $value)->addAttribute('key', $key);
            } else {
                $xml->addChild("$key",$value);
            }
        }
    }
}   
Йоуни Мякеляйнен
источник
0

Вся структура XML определена в массиве $ data:

function array2Xml($data, $xml = null)
{
    if (is_null($xml)) {
        $xml = simplexml_load_string('<' . key($data) . '/>');
        $data = current($data);
        $return = true;
    }
    if (is_array($data)) {
        foreach ($data as $name => $value) {
            array2Xml($value, is_numeric($name) ? $xml : $xml->addChild($name));
        }
    } else {
        $xml->{0} = $data;
    }
    if (!empty($return)) {
        return $xml->asXML();
    }
}
туз
источник
0

Если вы работаете в magento и у вас есть этот тип ассоциативного массива

$test_array = array (
    '0' => array (
            'category_id' => '582',
            'name' => 'Surat',
            'parent_id' => '565',
            'child_id' => '567',
            'active' => '1',
            'level' => '6',
            'position' => '17'
    ),

    '1' => array (
            'category_id' => '567', 
            'name' => 'test',
            'parent_id' => '0',
            'child_id' => '576',
            'active' => '0',
            'level' => '0',
            'position' => '18'
    ),
);

тогда лучше всего конвертировать ассоциативный массив в формат xml. Используйте этот код в файле контроллера.

$this->loadLayout(false);
//header ("content-type: text/xml");
$this->getResponse()->setHeader('Content-Type','text/xml');
$this->renderLayout();

$clArr2xml = new arr2xml($test_array, 'utf-8', 'listdata');
$output = $clArr2xml->get_xml();
print $output; 

class arr2xml
{
var $array = array();
var $xml = '';
var $root_name = '';
var $charset = '';

public function __construct($array, $charset = 'utf-8', $root_name = 'root')
{
    header ("content-type: text/xml");
    $this->array = $array;
    $this->root_name = $root_name;
    $this->charset = $charset;

    if (is_array($array) && count($array) > 0) {
        $this->struct_xml($array);

    } else {
        $this->xml .= "no data";
    }
}

public function struct_xml($array)
{
    foreach ($array as $k => $v) {
        if (is_array($v)) {
            $tag = ereg_replace('^[0-9]{1,}', 'item', $k); // replace numeric key in array to 'data'
            $this->xml .= "<$tag>";
            $this->struct_xml($v);
            $this->xml .= "</$tag>";
        } else {
            $tag = ereg_replace('^[0-9]{1,}', 'item', $k); // replace numeric key in array to 'data'
            $this->xml .= "<$tag><![CDATA[$v]]></$tag>";
        }
    }
}

public function get_xml()
{

    $header = "<?xml version=\"1.0\" encoding=\"" . $this->charset . "\"?><" . $this->root_name . ">";
    $footer = "</" . $this->root_name . ">";

    return $header . $this->xml . $footer;
}
}

Я надеюсь, что это помогает всем.

Бхарат Ходвадия
источник
0

// Structered array for XML convertion.
$data_array = array(
  array(
    '#xml_tag' => 'a',
    '#xml_value' => '',
    '#tag_attributes' => array(
      array(
        'name' => 'a_attr_name',
        'value' => 'a_attr_value',
      ),
    ),
    '#subnode' => array(
      array(
        '#xml_tag' => 'aa',
        '#xml_value' => 'aa_value',
        '#tag_attributes' => array(
          array(
            'name' => 'aa_attr_name',
            'value' => 'aa_attr_value',
          ),
        ),
        '#subnode' => FALSE,
      ),
    ),
  ),
  array(
    '#xml_tag' => 'b',
    '#xml_value' => 'b_value',
    '#tag_attributes' => FALSE,
    '#subnode' => FALSE,
  ),
  array(
    '#xml_tag' => 'c',
    '#xml_value' => 'c_value',
    '#tag_attributes' => array(
      array(
        'name' => 'c_attr_name',
        'value' => 'c_attr_value',
      ),
      array(
        'name' => 'c_attr_name_1',
        'value' => 'c_attr_value_1',
      ),
    ),
    '#subnode' => array(
      array(
        '#xml_tag' => 'ca',  
        '#xml_value' => 'ca_value',
        '#tag_attributes' => FALSE,
        '#subnode' => array(
          array(
            '#xml_tag' => 'caa',
            '#xml_value' => 'caa_value',
            '#tag_attributes' => array(
              array(
                'name' => 'caa_attr_name',
                'value' => 'caa_attr_value',
              ),
            ),
            '#subnode' => FALSE,
          ),
        ),
      ),
    ),
  ),
);


// creating object of SimpleXMLElement
$xml_object = new SimpleXMLElement('<?xml version=\"1.0\"?><student_info></student_info>');


// function call to convert array to xml
array_to_xml($data_array, $xml_object);

// saving generated xml file
$xml_object->asXML('/tmp/test.xml');

/**
 * Converts an structured PHP array to XML.
 *
 * @param Array $data_array
 *   The array data for converting into XML.
 * @param Object $xml_object
 *   The SimpleXMLElement Object
 *
 * @see https://gist.github.com/drupalista-br/9230016
 * 
 */
function array_to_xml($data_array, &$xml_object) {
  foreach($data_array as $node) {
    $subnode = $xml_object->addChild($node['#xml_tag'], $node['#xml_value']);

    if ($node['#tag_attributes']) {
      foreach ($node['#tag_attributes'] as $tag_attributes) {
        $subnode->addAttribute($tag_attributes['name'], $tag_attributes['value']); 
      }
    }

    if ($node['#subnode']) {
      array_to_xml($node['#subnode'], $subnode);
    }
  }
}
Франциско Луз
источник