如何使用数组更新SimpleXMLElement

时间:2013-02-12 16:15:31

标签: php xml arrays simplexml

我有这个适用于Flash Encoder的xml:

<?xml version="1.0" encoding="UTF-8"?>
<flashmedialiveencoder_profile>
    <preset>
        <name></name>
        <description></description>
    </preset>
    <capture>
        <video>
            <device></device> 
            <crossbar_input>1</crossbar_input>
            <frame_rate>25.00</frame_rate>
            <size>
                <width></width>
                <height></height>
            </size>
        </video>
        <audio>
            <device></device> 
            <crossbar_input>2</crossbar_input>
            <sample_rate></sample_rate>
            <channels>1</channels>
            <input_volume>60</input_volume>
        </audio>
    </capture>
    <encode>
        <video>
            <format>H.264</format>
            <datarate></datarate>
            <outputsize></outputsize>
            <advanced>
                <profile></profile>
                <level></level>
                <keyframe_frequency>5 Seconds</keyframe_frequency>
            </advanced>
            <autoadjust>
                <enable>false</enable>
                <maxbuffersize>1</maxbuffersize>
                <dropframes>
                <enable>false</enable>
                </dropframes>
                <degradequality>
                <enable>false</enable>
                <minvideobitrate></minvideobitrate>
                <preservepfq>false</preservepfq>
                </degradequality>
            </autoadjust>
        </video>
        <audio>
            <format>MP3</format>
            <datarate></datarate>
        </audio>
    </encode>
    <restartinterval>
        <days></days>
        <hours></hours>
        <minutes></minutes>
    </restartinterval>
    <reconnectinterval>
        <attempts></attempts>
        <interval></interval>
    </reconnectinterval>
    <output>
        <rtmp>
        <url></url>
        <backup_url></backup_url>
        <stream></stream>
        </rtmp>
    </output>
    <metadata>
        <entry>
        <key>author</key>
        <value></value>
        </entry>
        <entry>
        <key>copyright</key>
        <value></value>
        </entry>
        <entry>
        <key>description</key>
        <value></value>
        </entry>
        <entry>
        <key>keywords</key>
        <value></value>
        </entry>
        <entry>
        <key>rating</key>
        <value></value>
        </entry>
        <entry>
        <key>title</key>
        <value></value>
        </entry>
    </metadata>
    <preview>
        <video>
        <input>
            <zoom>50%</zoom>
        </input>
        <output>
            <zoom>50%</zoom>
        </output>
        </video>
        <audio></audio>
    </preview>
    <log>
        <level>100</level>
        <directory></directory>
    </log>
</flashmedialiveencoder_profile>

我需要根据用户需要的xml类型更新一些数据。所以我想做这样的事情:

$data = array (
    'preset' => array (
        'name' => 'Low Bandwidth (150 Kbps) - H.264',
    ),
    'capture' => array (
        'audio'  => array (
            'sample_rate' => 44100
        )
    ),
    'encode' => array (
        'video' => array (
            'datarate' => '300;',
            'outputsize' => '480x360;',
            'advanced' => array (
                'profile' => 'Main',
                'level' => 2.1,
            )
         ),
         'audio' => array (
            'datarate' => 48
         )
    ),
);

我知道这棵树有效,因为我可以手动完成,但问题是我不想那样做,所以如果将来我需要改变xml中的其他想法,我只有将其添加到阵列配置中。

我该怎么办?我可以写一个递归函数来做它,但我真的不想这样做。还有其他办法吗?我不知道......可能是这样的:

$xml->updateFromArray($data);

我重复一遍,我可以编写该函数并扩展SimpleXMLElement,但如果还有其他本机php方法,那就更好了。

2 个答案:

答案 0 :(得分:2)

考虑$xml,文档元素为SimpleXMLElement$data是您的数组(如问题中所示),如果您担心对儿童进行编号,例如对于元数据,我以这种方式在数组中编号:

'metadata' => array(
    'entry' => array(
        5 => array(
            'value' => 'Sunny Days',
        ),
    ),
),

以下显示如何借助堆栈以非递归方式解决问题:

while (list($set, $node) = array_pop($stack)) {
    if (!is_array($set)) {
        $node[0] = $set;
        continue;
    }

    foreach ($set as $element => $value) {
        $parent = $node->$element;
        if ($parent[0] == NULL) {
            throw new Exception(sprintf("Child-Element '%s' not found.", $element));
        }
        $stack[] = array($value, $parent);
    }
}

一些注意事项:

  • 我更改了具体的异常类型以删除依赖项。
  • $parent[0] == NULL测试元素$parent不为空(比较/看SimpleXML Type Cheatsheet)。
  • 当元素节点被放入堆栈以便稍后检索时,需要使用$node[0]来设置它从堆栈中取出后(编号元素已经在$node中(默认情况下是第一个,要稍后更改,需要将数字0用作偏移量。)

Online Demo


到目前为止,如果到目前为止它们不存在,则不允许创建新元素。要添加新元素,请为不存在的子项添加异常:

        if ($parent[0] == NULL) {
            throw new Exception(sprintf("Child-Element '%s' not found.", $element));
        }

需要替换为添加新子代的一些代码,包括第一个:

        if ($parent[0] == NULL) {
            if (is_int($element)) {
                if ($element != $node->count()) {
                    throw new Exception(sprintf("Element Number out of order: %d unfitting for %d elements so far.", $element, $node->count()));
                }
                $node[] = '';
                $parent = $node->$element;
            } else {
                $parent[0] = $node->addChild($element);
            }
        }

仍然存在的例外是添加新元素但其数量大于现有元素数加1的情况。例如你有4个元素,然后你“添加”数字6的元素,这将无法正常工作。该值从零开始,这通常不应该是任何问题。

Demo

答案 1 :(得分:0)

我已将此对象扩展为SimpleXMLElement,但如果有更好的方法可以做到这一点,我将非常感激:)

/**
 * Object to manipulate and add information to an xml object
 *
 * @package Stream
 * @subpackage SimpleXMLElement
 * @author abraham.sustaita@gmail.com
 * @author Abraham Cruz
 * @version 2 Added functionality to update data with array 
 * @example $this->rows[0] = array (); now can be edited
 */
class Stream_SimpleXMLElement extends SimpleXMLElement 
{
    /** 
     * Method to update the information of the xml
     * from an array
     * 
     * @throws Core_Exception
     * @param array $array
     * @return Stream_SimpleXMLElement
     */
    public function updateFromArray(array $array) {
        foreach ($array as $key => $data) {
            if ($this->$key->count() == 0) {
                throw new Core_Exception("The key {$key} does not exists within the XML object.");
            } else if ($this->$key->count() > 1) {
                foreach ($data as $i => $value) {
                    $tmpData = &$this->$key;
                    $tmpkey = &$tmpData [$i];
                    if (is_array($value)) {
                        foreach ($value as $key2 => $value2) {
                            if ($tmpkey->$key2 instanceof Stream_SimpleXMLElement && is_array($value2)) {
                                $tmpkey->$key2->updateFromArray($value2);
                            } else {
                                $tmpkey->$key2 = $value2;
                            }
                        }
                    } else {
                        $tmpkey = $value;
                    }
                    unset ($tmpData, $tmpkey);
                }
            } else {
                if (is_array($data)) {
                    $this->$key->updateFromArray($data);
                } else {
                    $this->$key = $data;
                }
            }
        }
        return $this;
    }
}

当对象Stream_SimpleXMLElement扩展SimpleXMLElement时,对象的每个子节点都将成为Stream_SimpleXMLElement的实例,因此我们可以使用相同的方法......