To convert correctly an XML for Array without the need to use JSON, the method below can be used for this.
function xmlToArray($xml, $options = array()) {
$defaults = array(
'namespaceSeparator' => ':', // você pode querer que isso seja algo diferente de um cólon
'attributePrefix' => '@', // para distinguir entre os nós e os atributos com o mesmo nome
'alwaysArray' => array(), // array de tags que devem sempre ser array
'autoArray' => true, // só criar arrays para as tags que aparecem mais de uma vez
'textContent' => '$', // chave utilizada para o conteúdo do texto de elementos
'autoText' => true, // pular chave "textContent" se o nó não tem atributos ou nós filho
'keySearch' => false, // pesquisa opcional e substituir na tag e nomes de atributos
'keyReplace' => false // substituir valores por valores acima de busca
);
$options = array_merge($defaults, $options);
$namespaces = $xml->getDocNamespaces();
$namespaces[''] = null; // adiciona namespace base(vazio)
// Obtém os atributos de todos os namespaces
$attributesArray = array();
foreach ($namespaces as $prefix => $namespace) {
foreach ($xml->attributes($namespace) as $attributeName => $attribute) {
// Substituir caracteres no nome do atributo
if ($options['keySearch']) $attributeName =
str_replace($options['keySearch'], $options['keyReplace'], $attributeName);
$attributeKey = $options['attributePrefix']
. ($prefix ? $prefix . $options['namespaceSeparator'] : '')
. $attributeName;
$attributesArray[$attributeKey] = (string)$attribute;
}
}
// Obtém nós filhos de todos os namespaces
$tagsArray = array();
foreach ($namespaces as $prefix => $namespace) {
foreach ($xml->children($namespace) as $childXml) {
// Recursividade em nós filho
$childArray = xmlToArray($childXml, $options);
list($childTagName, $childProperties) = each($childArray);
// Substituir caracteres no nome da tag
if ($options['keySearch']) $childTagName =
str_replace($options['keySearch'], $options['keyReplace'], $childTagName);
// Adiciona um prefixo namespace, se houver
if ($prefix) $childTagName = $prefix . $options['namespaceSeparator'] . $childTagName;
if (!isset($tagsArray[$childTagName])) {
// Só entra com esta chave
// Testa se as tags deste tipo deve ser sempre matrizes, não importa a contagem de elementos
$tagsArray[$childTagName] =
in_array($childTagName, $options['alwaysArray']) || !$options['autoArray']
? array($childProperties) : $childProperties;
} elseif (
is_array($tagsArray[$childTagName]) && array_keys($tagsArray[$childTagName])
=== range(0, count($tagsArray[$childTagName]) - 1)
) {
$tagsArray[$childTagName][] = $childProperties;
} else {
$tagsArray[$childTagName] = array($tagsArray[$childTagName], $childProperties);
}
}
}
// Obtém o texto do nó
$textContentArray = array();
$plainText = trim((string)$xml);
if ($plainText !== '') $textContentArray[$options['textContent']] = $plainText;
$propertiesArray = !$options['autoText'] || $attributesArray || $tagsArray || ($plainText === '')
? array_merge($attributesArray, $tagsArray, $textContentArray) : $plainText;
// Retorna o nó como array
return array(
$xml->getName() => $propertiesArray
);
}
Source
Example of use:
$page = file_get_contents("http://api.elsevier.com/content/search/index:SCIDIR?query=heart+attack&apiKey=20d0c7953f56925f725afe204869dadb&xml-decode=true&httpAccept=application%2Fxml");
$xml = simplexml_load_string($page);
$arrayData = xmlToArray($xml);
echo "<pre>";
print_r($arrayData);
Codepad.Viper-7 Demo
Thanks! I’m only having trouble with this solution, everything works ok but how can I count the entries of the array? I am trying to use Count($array['search-Results']) but gives the following error: Cannot use Object of type stdClass as array in
– pc_oc
@pc_oc
echo count($arrayData['search-results']);
seems to work for me.– stderr