PHP SimpleXML 中的 XML 到 JSON 转换


XML to JSON conversion in PHP SimpleXML

            $data = "<QRYRESULT>
            <ISSUCCESS>Y</ISSUCCESS>
            <EBLCUSTOMER ACCOUNTNO='11111'>
            <CUSTACCTNO>121212</CUSTACCTNO>
            <ACCTSTATUS>active</ACCTSTATUS>
            <CCYDESC>BDT</CCYDESC>
            <BALANCE>9999</BALANCE>
            <AVAILABLEBALANCE>99</AVAILABLEBALANCE>
            <CUSTOMERNAME>cus_name</CUSTOMERNAME>
            <AMOUNTONHOLD>1000</AMOUNTONHOLD>
            <ODLIMIT>99</ODLIMIT>
            </EBLCUSTOMER>
            </QRYRESULT>";

这是我尝试转换的 XML 字符串。我使用了 folloung 代码。

           $result = str_replace(array("'n", "'r", "'t"), '', $data);
           $xml = simplexml_load_string($result);
           $object = new stdclass();
           $object->webservice[] = $xml;
           $result = json_encode($object);
           header('content-Type: application/json');
           echo $result;

我正在获得以下 json 数据。

  {
     "webservice": [
       {
        "ISSUCCESS": "Y",
        "CUSTSUMMARY": {
            "@attributes": {
                "ACCOUNT": "11111"
            },
            "IDACCOUNT": "1010101",
            "CODACCTCURR": "BDT",
            "NUMBALANCE": "99999",
            "ACCTDESC": "22222",
            "PRDNAME": "name"
            }
         }
      ]
   }

但我不想要"@attributes"。我想要如下所示的输出:

{
  "QRYRESULT": {
   "ISSUCCESS": "Y",
   "EBLCUSTOMER": {
    "-ACCOUNTNO": "11111",
    "CUSTACCTNO": "121212",
    "ACCTSTATUS": "active",
    "CCYDESC": "BDT",
    "BALANCE": "9999",
     "AVAILABLEBALANCE": "99",
     "CUSTOMERNAME": "cus_name",
     "AMOUNTONHOLD": "1000",
    "ODLIMIT": "99"
    }
   }
  }

我该怎么做?

你不希望在 JSON 中编码 "@attributes "字段,但这是 PHP JSON 序列化 SimpleXMLElement 的标准方式。

正如你所说,你想要改变这一点,你需要改变PHP JSON序列化对象的方式。这可以通过自己实现 JsonSerializable with SimpleXMLElement 来实现,然后根据需要提供 JSON 序列化:

class JsonSerializer extends SimpleXmlElement implements JsonSerializable
{
    /**
     * SimpleXMLElement JSON serialization
     *
     * @return null|string
     *
     * @link http://php.net/JsonSerializable.jsonSerialize
     * @see JsonSerializable::jsonSerialize
     */
    function jsonSerialize()
    {
        // jishan's SimpleXMLElement JSON serialization ...
        return $serialized;
    }
}

例如,将属性用作像所有子元素一样的字段。

然后,您可以轻松集成它,例如代替

$xml = simplexml_load_string($result);

您可以使用

$xml = simplexml_load_string($result, 'JsonSerializer');

或者只是

$xml = new JsonSerializer($result);

函数的其余部分工作相同,但只是按照您的意愿序列化。

例:

$result = str_replace(array("'n", "'r", "'t"), '', $data);
$xml = new JsonSerializer($result);
$object = new stdclass();
$object->webservice[] = $xml;
$result = json_encode($object, JSON_PRETTY_PRINT);
header('content-Type: application/json');
echo $result;

输出:

{
    "webservice": [
        {
            "EBLCUSTOMER": {
                "ACCOUNTNO": "11111",
                "CUSTACCTNO": "121212",
                "ACCTSTATUS": "active",
                "CCYDESC": "BDT",
                "BALANCE": "9999",
                "AVAILABLEBALANCE": "99",
                "CUSTOMERNAME": "cus_name",
                "AMOUNTONHOLD": "1000",
                "ODLIMIT": "99"
            }
        }
    ]
}

上面示例的序列化函数为:

function jsonSerialize()
{
    // text node (or mixed node represented as text or self closing tag)
    if (!count($this)) {
        return $this[0] == $this
            ? trim($this) : null ;
    }
    // process all child elements and their attributes
    foreach ($this as $tag => $element) {
        // attributes first
        foreach ($element->attributes() as $name => $value) {
            $array[$tag][$name] = $value;
        }
        // child elements second
        foreach($element as $name => $value) {
            $array[$tag][$name] = $value;
        }
    }
    return $array;
}

这里有一些注意事项:

  • 在序列化中,您必须注意自己的元素类型。对于没有子元素的单个元素,在顶部进行区分。如果需要对这些属性进行处理,则需要添加它。
  • trim($this)可能已经使您免于您试图抓住的问题 $result = str_replace(array("'n", "'r", "'t"), '', $data); .在任何情况下,SimpleXMLElement都会JSON序列化"'r"字符(SimpleXMLElement使用"'n"作为中断)。此外,您可能对 XML 中的空格规范化规则感兴趣。
  • 如果属性与子元素同名,则它将被子元素覆盖。
  • 如果子元素跟在另一个具有相同名称的子元素后面,它将被覆盖。

最后两点只是为了保持示例代码的简单性。一种与SimpleXMLElement的标准PHP JSON序列化相一致的方法在我的一系列博客文章中给出。

此过程

的基础知识和示例性的 JsonSerialize 实现可在第三篇文章中找到:PHP 中的 SimpleXML 和 JSON Encode – Part III and End。

另一个相关问题是:

  • PHP 在有一个子级时将 XML 转换为 JSON 组
$fileContents= file_get_contents("https://www.feedforall.com/sample.xml");
$fileContents = str_replace(array("'n", "'r", "'t"), '', $fileContents);
$fileContents = trim(str_replace('"', "'", $fileContents));
$simpleXml = simplexml_load_string($fileContents);
$json = json_encode($simpleXml);
$array = json_decode($json,TRUE); // convert the JSON-encoded string to a PHP variable
return $array;

我没有更好的例子: