JsonUtil.php 4.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. <?php
  2. namespace loyalsoft;
  3. /**
  4. * Json转换工具,兼容UTF8格式
  5. * @author gwang
  6. * @version: <br/>
  7. * v1.3.0 优化decode方法, 检查是否json是合法json的同时, 将结果传递出来.
  8. * 去掉没有用到 json_last_error_msg()函数.
  9. * 去掉preg_replace的e参数用法, 完全使用preg_replace_callback.
  10. * 将v1.1.0中的改动移动到独立的方法中, 不干扰decode方法了.
  11. * gwang 2016.4.28 <br/>
  12. * v1.1.0 (未完成测试)在json_decode中集成反序列化的功能.如果参数中给出了目标类型/对象,
  13. * 则返回目标类型的实例. Ps.数据还不能处理 gwang 2017.xx.xx <br/>
  14. * v1.0.2 encode时增加判断,防止对string二次编码. gwang 2016.10.17 <br/>
  15. * v1.0.1 decode时加上了判断,如果不是json格式则返回原串. gwang 2016.4.28 <br/>
  16. * v1.0.0 file copy and created. gwang 2016.4.28 <br/>
  17. */
  18. class JsonUtil {
  19. /**
  20. * 对给定字符串尝试进行解析
  21. * @param string $string
  22. * @param mixed $obj (out) 解析出的对象
  23. * @return boolean 解析是否成功
  24. */
  25. private static function is_json($string, &$obj) {
  26. if (!is_string($string)) { # 不是字符串类型的对象
  27. return false;
  28. }
  29. if (!preg_match('/^[[|{]/', ltrim($string))) { # 字符串开头不是[或{ 则判定为非json格式
  30. // CLogUtil::debuglog('捕获不符合json格式的字符串' . $string);
  31. return false;
  32. }
  33. ob_start(); # 截获意外输出
  34. $obj = json_decode($string); # 解析json
  35. ob_end_clean(); # 抛弃输出
  36. return (json_last_error() == JSON_ERROR_NONE); # 返回
  37. }
  38. /**
  39. * 修正对字符串中中文的编码, Ps.(json_encode之后中文会转换为\u7956\u56fd形式的编码)
  40. * @param mixed $obj
  41. * @return string
  42. */
  43. public static function encode($obj) {
  44. $json = is_string($obj) ? $obj : json_encode($obj); # 防止对字符串二次编码
  45. if ($GLOBALS['OS'] == "linux") { # 还原中文字符串
  46. return preg_replace_callback('#\\\u([0-9a-f]{4})#i', function ($matches) {
  47. return iconv('UCS-2LE', 'UTF-8', pack('H4', $matches[1]));
  48. }, $json);
  49. } else if ($GLOBALS['OS'] == "win32") { # 还原中文字符串
  50. return preg_replace_callback('#\\\u([0-9a-f]{4})#i', function ($matches) {
  51. return iconv('UCS-2BE', 'UTF-8', pack('H4', $matches[1]));
  52. }, $json);
  53. }
  54. return $json;
  55. }
  56. /**
  57. * 对于异常的参数,记录到日志
  58. * @param type $string
  59. * @return obj
  60. */
  61. public static function decode($string) {
  62. $obj = $string;
  63. if (!self::is_json($string, $obj)) {
  64. // CLogUtil_HP::log("decoding: " . gettype($string) . " " # json解析失败,记录日志
  65. // . substr(var_export($string, true), 0, 30) . '...', "JsonUtil");
  66. // CLogUtil_HP::log(DebugHelper::get_call_stack(), "JsonUtil");
  67. }
  68. return $obj;
  69. }
  70. /**
  71. * 研发中..., 想着直接一步到位, 反序列化成目标类型的对象. -by gwang 2017年8月8日 09:49:09
  72. * @param string $str json字符串
  73. * @param mixed $tarType 目标类型
  74. * @return obj
  75. */
  76. public static function decode2Object($str, $tarType = null) {
  77. $obj = self::decode($str);
  78. if ($tarType != null) {
  79. $c = 'stdClass';
  80. if (is_string($tarType)) { # 类型名称
  81. $c = $tarType;
  82. } else if (is_object($tarType)) { # 目标类型的对象,数据还不能处理
  83. $c = get_class($tarType);
  84. }
  85. $desObj = new $c;
  86. CommUtil::loadObject($obj, $desObj); # 装到目标类型的箱子中
  87. $obj = $desObj;
  88. }
  89. return $obj;
  90. }
  91. }