Преобразование XML в JSON с использованием PHP с сохранением массивов - PullRequest
3 голосов
/ 24 августа 2011

Мне нужно преобразовать XML-документ в JSON, чтобы легко получить доступ к данным в JavaScript.В настоящее время я использую этот метод для преобразования XML в JSON:

json_encode(new SimpleXMLElement($xml, LIBXML_NOCDATA));

Однако я столкнулся с проблемой, когда элемент содержит только 1 дочерний элемент.Когда он анализируется с помощью SimpleXML, он обрабатывается как объект, а не как массив.Я хочу, чтобы они всегда обрабатывались как массивы, если только элемент не содержит только текст.

Пример:

$xml = <<<END
<xml>
  <TESTS>
    <TEST>TEXT HERE</TEST>
  </TESTS>
</xml>
END;

echo json_encode(new SimpleXMLElement($xml, LIBXML_NOCDATA));

Это выводит:

{"TESTS":{"TEST":"TEXT HERE"}}

Если я добавлю еще одинэлемент ниже, тогда вывод будет таким, каким я его хочу:

$xml = <<<END
<xml>
  <TESTS>
    <TEST>TEXT HERE</TEST>
    <TEST>MORE TEXT HERE</TEST>
  </TESTS>
</xml>
END;

echo json_encode(new SimpleXMLElement($xml, LIBXML_NOCDATA));

Вывод:

{"TESTS":{"TEST":["TEXT HERE","TEXT HERE"]}}

Обратите внимание, как элементы содержатся внутри массива JSON вместо объекта JSON.Есть ли способ заставить элементы анализироваться как массивы?

Ответы [ 2 ]

2 голосов
/ 05 сентября 2011

Мне пришлось столкнуться с такой же ситуацией.Вот быстрое первое решение проблемы - отлично работает для вашего примера.

class JsonWithArrays
{
    protected $root, $callback;

    public function __construct( SimpleXMLElement $root )
    {
        $this->root = $root;
    }

    /**
     * Set a callback to return if a node should be represented as an array
     * under any circumstances.
     *
     * The callback receives two parameters to react to: the SimpleXMLNode in
     * question, and the nesting level of that node.
     */
    public function use_callback_forcing_array ( $callback )
    {
        $this->callback = $callback;
        return $this;
    }

    public function to_json ()
    {
        $transformed = $this->transform_subnodes( $this->root, new stdClass(), 0 );
        return json_encode( $transformed );
    }

    protected function transform_subnodes ( SimpleXMLElement $parent, stdClass $transformed_parent, $nesting_level )
    {
        $nesting_level++;

        foreach( $parent->children() as $node )
        {
            $name = $node->getName();
            $value = (string) $node;

            if ( count( $node->children() ) > 0 )
            {
                $transformed_parent->$name = new stdClass();
                $this->transform_subnodes( $node, $transformed_parent->$name, $nesting_level );
            }
            elseif ( count( $parent->$name ) > 1 or $this->force_array( $node, $nesting_level ) )
            {
                $transformed_parent->{$name}[] = $value;
            }
            else
            {
                $transformed_parent->$name = $value;
            }
        }

        return $transformed_parent;
    }

    protected function force_array ( $node, $nesting_level )
    {
        if ( is_callable( $this->callback ) )
        {
            return call_user_func( $this->callback, $node, $nesting_level );
        }
        else
        {
            return false;
        }
    }
}

$xml = <<<END
<xml> 
  <TESTS> 
    <TEST>TEXT HERE</TEST> 
  </TESTS> 
</xml> 
END;

$xml2 = <<<END
<xml> 
  <TESTS> 
    <TEST>TEXT HERE</TEST> 
    <TEST>MORE TEXT HERE</TEST> 
  </TESTS> 
</xml> 
END;

// Callback using the node name. Could just as well be done using the nesting
// level.
function cb_testnode_as_array( SimpleXMLElement $node, $nesting_level )
{
    return $node->getName() == 'TEST';
}

$transform = new JsonWithArrays( new SimpleXMLElement($xml, LIBXML_NOCDATA) );
echo $transform
    ->use_callback_forcing_array( 'cb_testnode_as_array' )
    ->to_json();

echo PHP_EOL;

$transform2 = new JsonWithArrays( new SimpleXMLElement($xml2, LIBXML_NOCDATA) );
echo $transform2
    ->use_callback_forcing_array( 'cb_testnode_as_array' )
    ->to_json();
0 голосов
/ 30 августа 2011
echo json_encode(json_decode(json_encode(new SimpleXMLElement($xml, LIBXML_NOCDATA)), true));

На самом деле это очень глупо, но сначала, если все преобразовать в объект, затем декодировать в массив и преобразовать в json, как стиль массива ..))

...