I've got following XML:
<account>
<id>123</id>
<email></email>
<status>ACTIVE</status>
</account>
I want to have it as an array variable. Therefore I read it with $xml = simplexml_load_file()
. The simplest way to convert simpleXMLElement to an associative array I know is to grind it with: json_decode(json_encode((array) $xml),1);
The problem is that I don't want to get the email
key as an empty array, but rather as a NULL
value. As SimpleXMLElement, it looks like:
public 'email' =>
object(SimpleXMLElement)[205]
whereas in array it looks like:
'email' =>
array (size=0)
empty
I'd like to get:
'email' => NULL
The only way to achieve this I thought of is iterate through all elements and replace empty array with null value. The problem is that my XML is way bigger (above is just to explain the problem) and I'd have to iterate a lot of XML elements (and this would be manual work - I'm looking for something automatic). Maybe I'm missing some options in one of the functions... or maybe there's another trick to do this?
I cannot add a comment, but I think this will work for you, it should be faster then a regex or a loop:
//after you json_encode, before you decode
$str = str_replace(':[]',':null',json_encode($array));
An empty array in JSON is represented by "[]
". Sometimes the arrays are parsed as objects, in that case (or as a fallback) you can replace ":{}
" too.
An empty SimpleXMLElement object
will be casted to an empty array. You can change this by by extending from SimpleXMLElement and implementing the JsonSerializable interface
and casting an it to null.
/**
* Class JsonXMLElement
*/
class JsonXMLElement extends SimpleXMLElement implements JsonSerializable
{
/**
* Specify data which should be serialized to JSON
*
* @return mixed data which can be serialized by json_encode.
*/
public function jsonSerialize()
{
$array = array();
// json encode attributes if any.
if ($attributes = $this->attributes()) {
$array['@attributes'] = iterator_to_array($attributes);
}
// json encode child elements if any. group on duplicate names as an array.
foreach ($this as $name => $element) {
if (isset($array[$name])) {
if (!is_array($array[$name])) {
$array[$name] = [$array[$name]];
}
$array[$name][] = $element;
} else {
$array[$name] = $element;
}
}
// json encode non-whitespace element simplexml text values.
$text = trim($this);
if (strlen($text)) {
if ($array) {
$array['@text'] = $text;
} else {
$array = $text;
}
}
// return empty elements as NULL (self-closing or empty tags)
if (!$array) {
$array = NULL;
}
return $array;
}
}
Then tell simplexml_load_string
to return an object of JsonXMLElement
class
$xml = <<<XML
<account>
<id>123</id>
<email></email>
<status>ACTIVE</status>
</account>
XML;
$obj = simplexml_load_string($xml, 'JsonXMLElement');
// print_r($obj);
print json_encode($obj, true);
/*
* Output...
{
"id": 123,
"email": null,
"status": "ACTIVE"
}
*/
Credit: hakre
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With