PageRenderTime 50ms CodeModel.GetById 25ms RepoModel.GetById 1ms app.codeStats 0ms

/wind/utility/WindJson.php

https://github.com/dapachong/windframework
PHP | 317 lines | 245 code | 10 blank | 62 comment | 56 complexity | cbb8ecb24bf769f6ab0cabe97aa7b54c MD5 | raw file
  1. <?php
  2. Wind::import('WIND:utility.WindConvert');
  3. /**
  4. * json格式转换类
  5. *
  6. * 支持json转php类型,以及php类型转json.
  7. * @author Long.shi <long.shi@adlibaba-inc.com> 2011-10-19
  8. * @copyright ©2003-2103 phpwind.com
  9. * @license http://www.windframework.com
  10. * @version $Id$
  11. * @package utility
  12. */
  13. class WindJson {
  14. const JSON_SLICE = 1;
  15. const JSON_IN_STR = 2;
  16. const JSON_IN_ARR = 4;
  17. const JSON_IN_OBJ = 8;
  18. const JSON_IN_CMT = 16;
  19. /**
  20. * 将数据用json加密
  21. *
  22. * @param mixed $value 要加密的值
  23. * @param string $charset
  24. * @return string
  25. */
  26. public static function encode($source, $charset = 'utf8') {
  27. switch (gettype($source)) {
  28. case 'boolean':
  29. $source = $source ? 'true' : 'false';
  30. break;
  31. case 'NULL':
  32. $source = 'null';
  33. break;
  34. case 'integer':
  35. $source = (int) $source;
  36. break;
  37. case 'double':
  38. case 'float':
  39. $source = (float) $source;
  40. break;
  41. case 'string':
  42. $source = WindConvert::convert($source, 'utf8', $charset);
  43. $source = self::stringToJson($source);
  44. break;
  45. case 'array':
  46. $source = WindConvert::convert($source, 'utf8', $charset);
  47. $source = self::arrayToJson($source);
  48. break;
  49. case 'object':
  50. $source = WindConvert::convert($source, 'utf8', $charset);
  51. $source = self::objectToJson($source);
  52. break;
  53. default:
  54. break;
  55. }
  56. return $source;
  57. }
  58. /**
  59. * 将json格式数据解密
  60. *
  61. * @param string $str
  62. * @param boolean $toArray
  63. * @param string $charset
  64. * @return mixed
  65. */
  66. public static function decode($str, $toArray = true, $charset = 'utf8') {
  67. $str = self::_reduceString($str);
  68. $_str = strtolower($str);
  69. if ('true' == $_str) {
  70. return true;
  71. } elseif ('false' == $_str) {
  72. return false;
  73. } elseif ('null' == $_str) {
  74. return null;
  75. } elseif (is_numeric($str)) {
  76. return (float) $str == (integer) $str ? (integer) $str : (float) $str;
  77. } elseif (preg_match('/^("|\').+(\1)$/s', $_str, $matche) && $matche[1] == $matche[2]) {
  78. $str = self::jsonToString($str);
  79. } elseif (preg_match('/^\[.*\]$/s', $_str) || preg_match('/^\{.*\}$/s', $_str)) {
  80. $str = self::complexConvert($str, $toArray);
  81. }
  82. WindConvert::convert($str, $charset, 'utf8');
  83. return $str;
  84. }
  85. /**
  86. * 将json格式转成php string类型
  87. *
  88. * @param string $string json字符串
  89. * @return Ambigous <string, unknown>
  90. */
  91. protected static function jsonToString($string) {
  92. $delim = substr($string, 0, 1);
  93. $chrs = substr($string, 1, -1);
  94. $decodeStr = '';
  95. for ($c = 0, $length = strlen($chrs); $c < $length; ++$c) {
  96. $compare = substr($chrs, $c, 2);
  97. $ordCode = ord($chrs{$c});
  98. if ('\b' == $compare) {
  99. $decodeStr .= chr(0x08);
  100. ++$c;
  101. } elseif ('\t' == $compare) {
  102. $decodeStr .= chr(0x09);
  103. ++$c;
  104. } elseif ('\n' == $compare) {
  105. $decodeStr .= chr(0x0A);
  106. ++$c;
  107. } elseif ('\f' == $compare) {
  108. $decodeStr .= chr(0x0C);
  109. ++$c;
  110. } elseif ('\r' == $compare) {
  111. $decodeStr .= chr(0x0D);
  112. ++$c;
  113. } elseif (in_array($compare, array('\\"', '\\\'', '\\\\', '\\/'))) {
  114. if (('"' == $delim && '\\\'' != $compare) || ("'" == $delim && '\\"' != $compare)) {
  115. $decodeStr .= $chrs{++$c};
  116. }
  117. } elseif (preg_match('/\\\u[0-9A-F]{4}/i', substr($chrs, $c, 6))) {
  118. $utf16 = chr(hexdec(substr($chrs, ($c + 2), 2))) . chr(hexdec(substr($chrs, ($c + 4), 2)));
  119. $decodeStr .= WindConvert::utf16beToUTF8($utf16); //self::utf16beToUTF8($utf16);
  120. $c += 5;
  121. } elseif (0x20 <= $ordCode && 0x7F >= $ordCode) {
  122. $decodeStr .= $chrs{$c};
  123. } elseif (0xC0 == ($ordCode & 0xE0)) {
  124. $decodeStr .= substr($chrs, $c, 2);
  125. ++$c;
  126. } elseif (0xE0 == ($ordCode & 0xF0)) {
  127. $decodeStr .= substr($chrs, $c, 3);
  128. $c += 2;
  129. } elseif (0xF0 == ($ordCode & 0xF8)) {
  130. $decodeStr .= substr($chrs, $c, 4);
  131. $c += 3;
  132. } elseif (0xF8 == ($ordCode & 0xFC)) {
  133. $decodeStr .= substr($chrs, $c, 5);
  134. $c += 4;
  135. } elseif (0xFC == ($ordCode & 0xFE)) {
  136. $decodeStr .= substr($chrs, $c, 6);
  137. $c += 5;
  138. }
  139. }
  140. return $decodeStr;
  141. }
  142. /**
  143. * 复杂的json格式转换,支持object array格式
  144. *
  145. * @param string $str
  146. * @param boolean $toArray
  147. * @return Ambigous <multitype:, stdClass>|multitype:|Ambigous <mixed, boolean, NULL, number, multitype:, stdClass, Ambigous, string, unknown>|boolean
  148. */
  149. protected static function complexConvert($str, $toArray = true) {
  150. if ('[' == $str{0}) {
  151. $stk = array(self::JSON_IN_ARR);
  152. $arr = array();
  153. } else {
  154. $obj = $toArray ? array() : new stdClass();
  155. $stk = array(self::JSON_IN_OBJ);
  156. }
  157. array_push($stk, array('what' => self::JSON_SLICE, 'where' => 0, 'delim' => false));
  158. $chrs = substr($str, 1, -1);
  159. $chrs = self::_reduceString($chrs);
  160. if ('' == $chrs) {
  161. return self::JSON_IN_ARR == reset($stk) ? $arr : $obj;
  162. }
  163. for ($c = 0, $length = strlen($chrs); $c <= $length; ++$c) {
  164. $top = end($stk);
  165. $substr_chrs_c_2 = substr($chrs, $c, 2);
  166. if (($c == $length) || (($chrs{$c} == ',') && ($top['what'] == self::JSON_SLICE))) {
  167. $slice = substr($chrs, $top['where'], ($c - $top['where']));
  168. array_push($stk, array('what' => self::JSON_SLICE, 'where' => ($c + 1), 'delim' => false));
  169. if (reset($stk) == self::JSON_IN_ARR) {
  170. array_push($arr, self::decode($slice, $toArray));
  171. } elseif (reset($stk) == self::JSON_IN_OBJ) {
  172. if (preg_match('/^\s*(["\'].*[^\\\]["\'])\s*:\s*(\S.*),?$/Uis', $slice, $parts)) {
  173. $key = self::decode($parts[1], $toArray);
  174. $toArray ? $obj[$key] = self::decode($parts[2], $toArray) : $obj->$key = self::decode($parts[2],
  175. $toArray);
  176. } elseif (preg_match('/^\s*(\w+)\s*:\s*(\S.*),?$/Uis', $slice, $parts)) {
  177. $toArray ? $obj[$parts[1]] = self::decode($parts[2], $toArray) : $obj->$parts[1] = self::decode(
  178. $parts[2], $toArray);
  179. }
  180. }
  181. } elseif ((($chrs{$c} == '"') || ($chrs{$c} == "'")) && ($top['what'] != self::JSON_IN_STR)) {
  182. array_push($stk, array('what' => self::JSON_IN_STR, 'where' => $c, 'delim' => $chrs{$c}));
  183. } elseif (($chrs{$c} == $top['delim']) && ($top['what'] == self::JSON_IN_STR) && (($chrs{$c - 1} != "\\") || ($chrs{$c - 1} == "\\" && $chrs{$c - 2} == "\\"))) {
  184. array_pop($stk);
  185. } elseif (($chrs{$c} == '[') && in_array($top['what'],
  186. array(self::JSON_SLICE, self::JSON_IN_ARR, self::JSON_IN_OBJ))) {
  187. array_push($stk, array('what' => self::JSON_IN_ARR, 'where' => $c, 'delim' => false));
  188. } elseif (($chrs{$c} == ']') && ($top['what'] == self::JSON_IN_ARR)) {
  189. array_pop($stk);
  190. } elseif (($chrs{$c} == '{') && in_array($top['what'],
  191. array(self::JSON_SLICE, self::JSON_IN_ARR, self::JSON_IN_OBJ))) {
  192. array_push($stk, array('what' => self::JSON_IN_OBJ, 'where' => $c, 'delim' => false));
  193. } elseif (($chrs{$c} == '}') && ($top['what'] == self::JSON_IN_OBJ)) {
  194. array_pop($stk);
  195. } elseif (($substr_chrs_c_2 == '/*') && in_array($top['what'],
  196. array(self::JSON_SLICE, self::JSON_IN_ARR, self::JSON_IN_OBJ))) {
  197. array_push($stk, array('what' => self::JSON_IN_CMT, 'where' => ++$c, 'delim' => false));
  198. } elseif (($substr_chrs_c_2 == '*/') && ($top['what'] == self::JSON_IN_CMT)) {
  199. array_pop($stk);
  200. for ($i = $top['where']; $i <= ++$c; ++$i) {
  201. $chrs = substr_replace($chrs, ' ', $i, 1);
  202. }
  203. }
  204. }
  205. if (self::JSON_IN_ARR == reset($stk)) {
  206. return $arr;
  207. } elseif (self::JSON_IN_OBJ == reset($stk)) {
  208. return $obj;
  209. }
  210. return false;
  211. }
  212. /**
  213. * 将字符串转化成json格式对象
  214. *
  215. * @param string $string
  216. * @return string
  217. */
  218. protected static function stringToJson($string) {
  219. $ascii = '';
  220. $strlen = strlen($string);
  221. for ($c = 0; $c < $strlen; ++$c) {
  222. $b = $string{$c};
  223. $ordVar = ord($string{$c});
  224. if (0x08 == $ordVar) {
  225. $ascii .= '\b';
  226. } elseif (0x09 == $ordVar) {
  227. $ascii .= '\t';
  228. } elseif (0x0A == $ordVar) {
  229. $ascii .= '\n';
  230. } elseif (0x0C == $ordVar) {
  231. $ascii .= '\f';
  232. } elseif (0x0D == $ordVar) {
  233. $ascii .= '\r';
  234. } elseif (in_array($ordVar, array(0x22, 0x2F, 0x5C))) {
  235. $ascii .= '\\' . $string{$c};
  236. } elseif (0x20 <= $ordVar && 0x7F >= $ordVar) {
  237. $ascii .= $string{$c}; //ASCII
  238. } elseif (0xC0 == ($ordVar & 0xE0)) {
  239. $char = pack('C*', $ordVar, ord($string{++$c}));
  240. $ascii .= sprintf('\u%04s', bin2hex(WindConvert::utf8ToUTF16BE($char)));
  241. } elseif (0xE0 == ($ordVar & 0xF0)) {
  242. $char = pack('C*', $ordVar, ord($string{++$c}), ord($string{++$c}));
  243. $ascii .= sprintf('\u%04s', bin2hex(WindConvert::utf8ToUTF16BE($char)));
  244. } elseif (0xF0 == ($ordVar & 0xF8)) {
  245. $char = pack('C*', $ordVar, ord($string{++$c}), ord($string{++$c}), ord($string{++$c}));
  246. $ascii .= sprintf('\u%04s', bin2hex(WindConvert::utf8ToUTF16BE($char)));
  247. } elseif (0xF8 == ($ordVar & 0xFC)) {
  248. $char = pack('C*', $ordVar, ord($string{++$c}), ord($string{++$c}), ord($string{++$c}),
  249. ord($string{++$c}));
  250. $ascii .= sprintf('\u%04s', bin2hex(WindConvert::utf8ToUTF16BE($char)));
  251. } elseif (0xFC == ($ordVar & 0xFE)) {
  252. $char = pack('C*', $ordVar, ord($string{++$c}), ord($string{++$c}), ord($string{++$c}),
  253. ord($string{++$c}), ord($string{++$c}));
  254. $ascii .= sprintf('\u%04s', bin2hex(WindConvert::utf8ToUTF16BE($char)));
  255. }
  256. }
  257. return '"' . $ascii . '"';
  258. }
  259. /**
  260. * 将数组转化成json格式对象
  261. *
  262. * @param array $array
  263. * @return string
  264. */
  265. protected static function arrayToJson(array $array) {
  266. if (is_array($array) && count($array) && (array_keys($array) !== range(0, sizeof($array) - 1))) {
  267. return '{' . join(',', array_map(array('WindJson', '_nameValue'), array_keys($array), array_values($array))) . '}';
  268. }
  269. return '[' . join(',', array_map(array('WindJson', 'encode'), $array)) . ']';
  270. }
  271. /**
  272. * 将对象转化成json格式对象
  273. *
  274. * @param string $object
  275. * @return string
  276. */
  277. protected static function objectToJson($object) {
  278. if ($object instanceof Traversable) {
  279. $vars = array();
  280. foreach ($object as $k => $v) {
  281. $vars[$k] = $v;
  282. }
  283. } else {
  284. $vars = get_object_vars($object);
  285. }
  286. return '{' . join(',', array_map(array('WindJson', '_nameValue'), array_keys($vars), array_values($vars))) . '}';
  287. }
  288. /**
  289. * @param string $str
  290. * @return string
  291. */
  292. private static function _reduceString($str) {
  293. return trim(preg_replace(array('#^\s*//(.+)$#m', '#^\s*/\*(.+)\*/#Us', '#/\*(.+)\*/\s*$#Us'), '', $str));
  294. }
  295. /**
  296. * callback函数,用于数组或对象加密
  297. *
  298. * @param mixed $name
  299. * @param mixed $value
  300. */
  301. private static function _nameValue($name, $value) {
  302. return self::encode(strval($name)) . ':' . self::encode($value);
  303. }
  304. }
  305. ?>