首页 > 开发 > PHP > 正文

PHP实现的json类实例

2024-05-04 23:38:09
字体:
来源:转载
供稿:网友

这篇文章主要介绍了PHP实现的json类,实例分析了php实现json类的方法,具有一定参考借鉴价值,需要的朋友可以参考下

本文实例讲述了PHP实现的json类。分享给大家供大家参考。具体如下:

这里注意json_encode只有(PHP 5 >= 5.2.0, PECL json >= 1.2.0)才有

 

 
  1. <?php 
  2. /* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: */ 
  3. /** 
  4. * Converts to and from JSON format. 
  5. * 
  6. * JSON (JavaScript Object Notation) is a lightweight data-interchange 
  7. * format. It is easy for humans to read and write. It is easy for machines 
  8. * to parse and generate. It is based on a subset of the JavaScript 
  9. * Programming Language, Standard ECMA-262 3rd Edition - December 1999. 
  10. * This feature can also be found in Python. JSON is a text format that is 
  11. * completely language independent but uses conventions that are familiar 
  12. * to programmers of the C-family of languages, including C, C++, C#, Java, 
  13. * JavaScript, Perl, TCL, and many others. These properties make JSON an 
  14. * ideal data-interchange language. 
  15. * 
  16. * This package provides a simple encoder and decoder for JSON notation. It 
  17. * is intended for use with client-side Javascript applications that make 
  18. * use of HTTPRequest to perform server communication functions - data can 
  19. * be encoded into JSON notation for use in a client-side javascript, or 
  20. * decoded from incoming Javascript requests. JSON format is native to 
  21. * Javascript, and can be directly eval()'ed with no further parsing 
  22. * overhead 
  23. * 
  24. * All strings should be in ASCII or UTF-8 format! 
  25. * 
  26. * LICENSE: Redistribution and use in source and binary forms, with or 
  27. * without modification, are permitted provided that the following 
  28. * conditions are met: Redistributions of source code must retain the 
  29. * above copyright notice, this list of conditions and the following 
  30. * disclaimer. Redistributions in binary form must reproduce the above 
  31. * copyright notice, this list of conditions and the following disclaimer 
  32. * in the documentation and/or other materials provided with the 
  33. * distribution. 
  34. * 
  35. * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED 
  36. * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 
  37. * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN 
  38. * NO EVENT SHALL CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 
  39. * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 
  40. * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 
  41. * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 
  42. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR 
  43. * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE 
  44. * USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH 
  45. * DAMAGE. 
  46. * 
  47. * @category 
  48. * @package Services_JSON 
  49. * @author Michal Migurski <mike-json@teczno.com> 
  50. * @author Matt Knapp <mdknapp[at]gmail[dot]com> 
  51. * @author Brett Stimmerman <brettstimmerman[at]gmail[dot]com> 
  52. * @copyright 2005 Michal Migurski 
  53. * @version CVS: $Id: JSON.php,v 1.31 2006/06/28 05:54:17 migurski Exp $ 
  54. * @license http://www.opensource.org/licenses/bsd-license.php 
  55. * @link http://pear.php.net/pepr/pepr-proposal-show.php?id=198 
  56. */ 
  57. /** 
  58. * Marker constant for Services_JSON::decode(), used to flag stack state 
  59. */ 
  60. define('SERVICES_JSON_SLICE', 1); 
  61. /** 
  62. * Marker constant for Services_JSON::decode(), used to flag stack state 
  63. */ 
  64. define('SERVICES_JSON_IN_STR', 2); 
  65. /** 
  66. * Marker constant for Services_JSON::decode(), used to flag stack state 
  67. */ 
  68. define('SERVICES_JSON_IN_ARR', 3); 
  69. /** 
  70. * Marker constant for Services_JSON::decode(), used to flag stack state 
  71. */ 
  72. define('SERVICES_JSON_IN_OBJ', 4); 
  73. /** 
  74. * Marker constant for Services_JSON::decode(), used to flag stack state 
  75. */ 
  76. define('SERVICES_JSON_IN_CMT', 5); 
  77. /** 
  78. * Behavior switch for Services_JSON::decode() 
  79. */ 
  80. define('SERVICES_JSON_LOOSE_TYPE', 16); 
  81. /** 
  82. * Behavior switch for Services_JSON::decode() 
  83. */ 
  84. define('SERVICES_JSON_SUPPRESS_ERRORS', 32); 
  85. /** 
  86. * Converts to and from JSON format. 
  87. * 
  88. * Brief example of use: 
  89. * 
  90. * <code> 
  91. * // create a new instance of Services_JSON 
  92. * $json = new Services_JSON(); 
  93. * 
  94. * // convert a complexe value to JSON notation, and send it to the browser 
  95. * $value = array('foo', 'bar', array(1, 2, 'baz'), array(3, array(4))); 
  96. * $output = $json->encode($value); 
  97. * 
  98. * print($output); 
  99. * // prints: ["foo","bar",[1,2,"baz"],[3,[4]]] 
  100. * 
  101. * // accept incoming POST data, assumed to be in JSON notation 
  102. * $input = file_get_contents('php://input', 1000000); 
  103. * $value = $json->decode($input); 
  104. * </code> 
  105. */ 
  106. class Services_JSON 
  107. /** 
  108. * constructs a new JSON instance 
  109. * 
  110. * @param int $use object behavior flags; combine with boolean-OR 
  111. * 
  112. * possible values: 
  113. * - SERVICES_JSON_LOOSE_TYPE: loose typing. 
  114. * "{...}" syntax creates associative arrays 
  115. * instead of objects in decode(). 
  116. * - SERVICES_JSON_SUPPRESS_ERRORS: error suppression. 
  117. * Values which can't be encoded (e.g. resources) 
  118. * appear as NULL instead of throwing errors. 
  119. * By default, a deeply-nested resource will 
  120. * bubble up with an error, so all return values 
  121. * from encode() should be checked with isError() 
  122. */ 
  123. function Services_JSON($use = 0) 
  124. $this->use = $use; 
  125. /** 
  126. * convert a string from one UTF-16 char to one UTF-8 char 
  127. * 
  128. * Normally should be handled by mb_convert_encoding, but 
  129. * provides a slower PHP-only method for installations 
  130. * that lack the multibye string extension. 
  131. * 
  132. * @param string $utf16 UTF-16 character 
  133. * @return string UTF-8 character 
  134. * @access private 
  135. */ 
  136. function utf162utf8($utf16) 
  137. // oh please oh please oh please oh please oh please 
  138. if(function_exists('mb_convert_encoding')) { 
  139. return mb_convert_encoding($utf16, 'UTF-8''UTF-16'); 
  140. $bytes = (ord($utf16{0}) << 8) | ord($utf16{1}); 
  141. switch(true) { 
  142. case ((0x7F & $bytes) == $bytes): 
  143. // this case should never be reached, because we are in ASCII range 
  144. // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  145. return chr(0x7F & $bytes); 
  146. case (0x07FF & $bytes) == $bytes: 
  147. // return a 2-byte UTF-8 character 
  148. // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  149. return chr(0xC0 | (($bytes >> 6) & 0x1F)) 
  150. . chr(0x80 | ($bytes & 0x3F)); 
  151. case (0xFFFF & $bytes) == $bytes: 
  152. // return a 3-byte UTF-8 character 
  153. // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  154. return chr(0xE0 | (($bytes >> 12) & 0x0F)) 
  155. . chr(0x80 | (($bytes >> 6) & 0x3F)) 
  156. . chr(0x80 | ($bytes & 0x3F)); 
  157. // ignoring UTF-32 for now, sorry 
  158. return ''
  159. /** 
  160. * convert a string from one UTF-8 char to one UTF-16 char 
  161. * 
  162. * Normally should be handled by mb_convert_encoding, but 
  163. * provides a slower PHP-only method for installations 
  164. * that lack the multibye string extension. 
  165. * 
  166. * @param string $utf8 UTF-8 character 
  167. * @return string UTF-16 character 
  168. * @access private 
  169. */ 
  170. function utf82utf16($utf8) 
  171. // oh please oh please oh please oh please oh please 
  172. if(function_exists('mb_convert_encoding')) { 
  173. return mb_convert_encoding($utf8, 'UTF-16''UTF-8'); 
  174. switch(strlen($utf8)) { 
  175. case 1: 
  176. // this case should never be reached, because we are in ASCII range 
  177. // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  178. return $utf8; 
  179. case 2: 
  180. // return a UTF-16 character from a 2-byte UTF-8 char 
  181. // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  182. return chr(0x07 & (ord($utf8{0}) >> 2)) 
  183. . chr((0xC0 & (ord($utf8{0}) << 6)) 
  184. | (0x3F & ord($utf8{1}))); 
  185. case 3: 
  186. // return a UTF-16 character from a 3-byte UTF-8 char 
  187. // see: http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  188. return chr((0xF0 & (ord($utf8{0}) << 4)) 
  189. | (0x0F & (ord($utf8{1}) >> 2))) 
  190. . chr((0xC0 & (ord($utf8{1}) << 6)) 
  191. | (0x7F & ord($utf8{2}))); 
  192. // ignoring UTF-32 for now, sorry 
  193. return ''
  194. /** 
  195. * encodes an arbitrary variable into JSON format 
  196. * 
  197. * @param mixed $var any number, boolean, string, array, or object to be encoded. 
  198. * see argument 1 to Services_JSON() above for array-parsing behavior. 
  199. * if var is a strng, note that encode() always expects it 
  200. * to be in ASCII or UTF-8 format! 
  201. * 
  202. * @return mixed JSON string representation of input var or an error if a problem occurs 
  203. * @access public 
  204. */ 
  205. function encode($var
  206. switch (gettype($var)) { 
  207. case 'boolean'
  208. return $var ? 'true' : 'false'
  209. case 'NULL'
  210. return 'null'
  211. case 'integer'
  212. return (int) $var
  213. case 'double'
  214. case 'float'
  215. return (float) $var
  216. case 'string'
  217. // STRINGS ARE EXPECTED TO BE IN ASCII OR UTF-8 FORMAT 
  218. $ascii = ''
  219. $strlen_var = strlen($var); 
  220. /* 
  221. * Iterate over every character in the string, 
  222. * escaping with a slash or encoding to UTF-8 where necessary 
  223. */ 
  224. for ($c = 0; $c < $strlen_var; ++$c) { 
  225. $ord_var_c = ord($var{$c}); 
  226. switch (true) { 
  227. case $ord_var_c == 0x08: 
  228. $ascii .= '/b'
  229. break
  230. case $ord_var_c == 0x09: 
  231. $ascii .= '/t'
  232. break
  233. case $ord_var_c == 0x0A: 
  234. $ascii .= '/n'
  235. break
  236. case $ord_var_c == 0x0C: 
  237. $ascii .= '/f'
  238. break
  239. case $ord_var_c == 0x0D: 
  240. $ascii .= '/r'
  241. break
  242. case $ord_var_c == 0x22: 
  243. case $ord_var_c == 0x2F: 
  244. case $ord_var_c == 0x5C: 
  245. // double quote, slash, slosh 
  246. $ascii .= '//'.$var{$c}; 
  247. break
  248. case (($ord_var_c >= 0x20) && ($ord_var_c <= 0x7F)): 
  249. // characters U-00000000 - U-0000007F (same as ASCII) 
  250. $ascii .= $var{$c}; 
  251. break
  252. case (($ord_var_c & 0xE0) == 0xC0): 
  253. // characters U-00000080 - U-000007FF, mask 110XXXXX 
  254. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  255. $char = pack('C*', $ord_var_c, ord($var{$c + 1})); 
  256. $c += 1; 
  257. $utf16 = $this->utf82utf16($char); 
  258. $ascii .= sprintf('/u%04s', bin2hex($utf16)); 
  259. break
  260. case (($ord_var_c & 0xF0) == 0xE0): 
  261. // characters U-00000800 - U-0000FFFF, mask 1110XXXX 
  262. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  263. $char = pack('C*', $ord_var_c, 
  264. ord($var{$c + 1}), 
  265. ord($var{$c + 2})); 
  266. $c += 2; 
  267. $utf16 = $this->utf82utf16($char); 
  268. $ascii .= sprintf('/u%04s', bin2hex($utf16)); 
  269. break
  270. case (($ord_var_c & 0xF8) == 0xF0): 
  271. // characters U-00010000 - U-001FFFFF, mask 11110XXX 
  272. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  273. $char = pack('C*', $ord_var_c, 
  274. ord($var{$c + 1}), 
  275. ord($var{$c + 2}), 
  276. ord($var{$c + 3})); 
  277. $c += 3; 
  278. $utf16 = $this->utf82utf16($char); 
  279. $ascii .= sprintf('/u%04s', bin2hex($utf16)); 
  280. break
  281. case (($ord_var_c & 0xFC) == 0xF8): 
  282. // characters U-00200000 - U-03FFFFFF, mask 111110XX 
  283. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  284. $char = pack('C*', $ord_var_c, 
  285. ord($var{$c + 1}), 
  286. ord($var{$c + 2}), 
  287. ord($var{$c + 3}), 
  288. ord($var{$c + 4})); 
  289. $c += 4; 
  290. $utf16 = $this->utf82utf16($char); 
  291. $ascii .= sprintf('/u%04s', bin2hex($utf16)); 
  292. break
  293. case (($ord_var_c & 0xFE) == 0xFC): 
  294. // characters U-04000000 - U-7FFFFFFF, mask 1111110X 
  295. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  296. $char = pack('C*', $ord_var_c, 
  297. ord($var{$c + 1}), 
  298. ord($var{$c + 2}), 
  299. ord($var{$c + 3}), 
  300. ord($var{$c + 4}), 
  301. ord($var{$c + 5})); 
  302. $c += 5; 
  303. $utf16 = $this->utf82utf16($char); 
  304. $ascii .= sprintf('/u%04s', bin2hex($utf16)); 
  305. break
  306. return '"'.$ascii.'"'
  307. case 'array'
  308. /* 
  309. * As per JSON spec if any array key is not an integer 
  310. * we must treat the the whole array as an object. We 
  311. * also try to catch a sparsely populated associative 
  312. * array with numeric keys here because some JS engines 
  313. * will create an array with empty indexes up to 
  314. * max_index which can cause memory issues and because 
  315. * the keys, which may be relevant, will be remapped 
  316. * otherwise. 
  317. * 
  318. * As per the ECMA and JSON specification an object may 
  319. * have any string as a property. Unfortunately due to 
  320. * a hole in the ECMA specification if the key is a 
  321. * ECMA reserved word or starts with a digit the 
  322. * parameter is only accessible using ECMAScript's 
  323. * bracket notation. 
  324. */ 
  325. // treat as a JSON object 
  326. if (is_array($var) && count($var) && (array_keys($var) !== range(0, sizeof($var) - 1))) { 
  327. $properties = array_map(array($this'name_value'), 
  328. array_keys($var), 
  329. array_values($var)); 
  330. foreach($properties as $property) { 
  331. if(Services_JSON::isError($property)) { 
  332. return $property; 
  333. return '{' . join(',', $properties) . '}'
  334. // treat it like a regular array 
  335. $elements = array_map(array($this'encode'), $var); 
  336. foreach($elements as $element) { 
  337. if(Services_JSON::isError($element)) { 
  338. return $element; 
  339. return '[' . join(',', $elements) . ']'
  340. case 'object'
  341. $vars = get_object_vars($var); 
  342. $properties = array_map(array($this'name_value'), 
  343. array_keys($vars), 
  344. array_values($vars)); 
  345. foreach($properties as $property) { 
  346. if(Services_JSON::isError($property)) { 
  347. return $property; 
  348. return '{' . join(',', $properties) . '}'
  349. default
  350. return ($this->use & SERVICES_JSON_SUPPRESS_ERRORS) 
  351. 'null' 
  352. new Services_JSON_Error(gettype($var)." can not be encoded as JSON string"); 
  353. /** 
  354. * array-walking function for use in generating JSON-formatted name-value pairs 
  355. * 
  356. * @param string $name name of key to use 
  357. * @param mixed $value reference to an array element to be encoded 
  358. * 
  359. * @return string JSON-formatted name-value pair, like '"name":value' 
  360. * @access private 
  361. */ 
  362. function name_value($name, $value) 
  363. $encoded_value = $this->encode($value); 
  364. if(Services_JSON::isError($encoded_value)) { 
  365. return $encoded_value; 
  366. return $this->encode(strval($name)) . ':' . $encoded_value; 
  367. /** 
  368. * reduce a string by removing leading and trailing comments and whitespace 
  369. * 
  370. * @param $str string string value to strip of comments and whitespace 
  371. * 
  372. * @return string string value stripped of comments and whitespace 
  373. * @access private 
  374. */ 
  375. function reduce_string($str) 
  376. $str = preg_replace(array( 
  377. // eliminate single line comments in '// ...' form 
  378. '#^/s*//(.+)$#m'
  379. // eliminate multi-line comments in '/* ... */' form, at start of string 
  380. '#^/s*//*(.+)/*/#Us'
  381. // eliminate multi-line comments in '/* ... */' form, at end of string 
  382. '#//*(.+)/*//s*$#Us' 
  383. ), '', $str); 
  384. // eliminate extraneous space 
  385. return trim($str); 
  386. /** 
  387. * decodes a JSON string into appropriate variable 
  388. * 
  389. * @param string $str JSON-formatted string 
  390. * 
  391. * @return mixed number, boolean, string, array, or object 
  392. * corresponding to given JSON input string. 
  393. * See argument 1 to Services_JSON() above for object-output behavior. 
  394. * Note that decode() always returns strings 
  395. * in ASCII or UTF-8 format! 
  396. * @access public 
  397. */ 
  398. function decode($str) 
  399. $str = $this->reduce_string($str); 
  400. switch (strtolower($str)) { 
  401. case 'true'
  402. return true
  403. case 'false'
  404. return false
  405. case 'null'
  406. return null
  407. default
  408. $m = array(); 
  409. if (is_numeric($str)) { 
  410. // Lookie-loo, it's a number 
  411. // This would work on its own, but I'm trying to be 
  412. // good about returning integers where appropriate: 
  413. // return (float)$str; 
  414. // Return float or int, as appropriate 
  415. return ((float)$str == (integer)$str) 
  416. ? (integer)$str 
  417. : (float)$str; 
  418. } elseif (preg_match('/^("|/').*(/1)$/s', $str, $m) && $m[1] == $m[2]) { 
  419. // STRINGS RETURNED IN UTF-8 FORMAT 
  420. $delim = substr($str, 0, 1); 
  421. $chrs = substr($str, 1, -1); 
  422. $utf8 = ''
  423. $strlen_chrs = strlen($chrs); 
  424. for ($c = 0; $c < $strlen_chrs; ++$c) { 
  425. $substr_chrs_c_2 = substr($chrs, $c, 2); 
  426. $ord_chrs_c = ord($chrs{$c}); 
  427. switch (true) { 
  428. case $substr_chrs_c_2 == '/b'
  429. $utf8 .= chr(0x08); 
  430. ++$c; 
  431. break
  432. case $substr_chrs_c_2 == '/t'
  433. $utf8 .= chr(0x09); 
  434. ++$c; 
  435. break
  436. case $substr_chrs_c_2 == '/n'
  437. $utf8 .= chr(0x0A); 
  438. ++$c; 
  439. break
  440. case $substr_chrs_c_2 == '/f'
  441. $utf8 .= chr(0x0C); 
  442. ++$c; 
  443. break
  444. case $substr_chrs_c_2 == '/r'
  445. $utf8 .= chr(0x0D); 
  446. ++$c; 
  447. break
  448. case $substr_chrs_c_2 == '//"'
  449. case $substr_chrs_c_2 == '///''
  450. case $substr_chrs_c_2 == '////'
  451. case $substr_chrs_c_2 == '///'
  452. if (($delim == '"' && $substr_chrs_c_2 != '///'') || 
  453. ($delim == "'" && $substr_chrs_c_2 != '//"')) { 
  454. $utf8 .= $chrs{++$c}; 
  455. break
  456. case preg_match('////u[0-9A-F]{4}/i', substr($chrs, $c, 6)): 
  457. // single, escaped unicode character 
  458. $utf16 = chr(hexdec(substr($chrs, ($c + 2), 2))) 
  459. . chr(hexdec(substr($chrs, ($c + 4), 2))); 
  460. $utf8 .= $this->utf162utf8($utf16); 
  461. $c += 5; 
  462. break
  463. case ($ord_chrs_c >= 0x20) && ($ord_chrs_c <= 0x7F): 
  464. $utf8 .= $chrs{$c}; 
  465. break
  466. case ($ord_chrs_c & 0xE0) == 0xC0: 
  467. // characters U-00000080 - U-000007FF, mask 110XXXXX 
  468. //see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  469. $utf8 .= substr($chrs, $c, 2); 
  470. ++$c; 
  471. break
  472. case ($ord_chrs_c & 0xF0) == 0xE0: 
  473. // characters U-00000800 - U-0000FFFF, mask 1110XXXX 
  474. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  475. $utf8 .= substr($chrs, $c, 3); 
  476. $c += 2; 
  477. break
  478. case ($ord_chrs_c & 0xF8) == 0xF0: 
  479. // characters U-00010000 - U-001FFFFF, mask 11110XXX 
  480. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  481. $utf8 .= substr($chrs, $c, 4); 
  482. $c += 3; 
  483. break
  484. case ($ord_chrs_c & 0xFC) == 0xF8: 
  485. // characters U-00200000 - U-03FFFFFF, mask 111110XX 
  486. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  487. $utf8 .= substr($chrs, $c, 5); 
  488. $c += 4; 
  489. break
  490. case ($ord_chrs_c & 0xFE) == 0xFC: 
  491. // characters U-04000000 - U-7FFFFFFF, mask 1111110X 
  492. // see http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8 
  493. $utf8 .= substr($chrs, $c, 6); 
  494. $c += 5; 
  495. break
  496. return $utf8; 
  497. } elseif (preg_match('/^/[.*/]$/s', $str) || preg_match('/^/{.*/}$/s', $str)) { 
  498. // array, or object notation 
  499. if ($str{0} == '[') { 
  500. $stk = array(SERVICES_JSON_IN_ARR); 
  501. $arr = array(); 
  502. else { 
  503. if ($this->use & SERVICES_JSON_LOOSE_TYPE) { 
  504. $stk = array(SERVICES_JSON_IN_OBJ); 
  505. $obj = array(); 
  506. else { 
  507. $stk = array(SERVICES_JSON_IN_OBJ); 
  508. $obj = new stdClass(); 
  509. array_push($stk, array('what' => SERVICES_JSON_SLICE, 
  510. 'where' => 0, 
  511. 'delim' => false)); 
  512. $chrs = substr($str, 1, -1); 
  513. $chrs = $this->reduce_string($chrs); 
  514. if ($chrs == '') { 
  515. if (reset($stk) == SERVICES_JSON_IN_ARR) { 
  516. return $arr; 
  517. else { 
  518. return $obj; 
  519. //print("/nparsing {$chrs}/n"); 
  520. $strlen_chrs = strlen($chrs); 
  521. for ($c = 0; $c <= $strlen_chrs; ++$c) { 
  522. $top = end($stk); 
  523. $substr_chrs_c_2 = substr($chrs, $c, 2); 
  524. if (($c == $strlen_chrs) || (($chrs{$c} == ',') && ($top['what'] == SERVICES_JSON_SLICE))) { 
  525. // found a comma that is not inside a string, array, etc., 
  526. // OR we've reached the end of the character list 
  527. $slice = substr($chrs, $top['where'], ($c - $top['where'])); 
  528. array_push($stk, array('what' => SERVICES_JSON_SLICE, 'where' => ($c + 1), 'delim' => false)); 
  529. //print("Found split at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."/n"); 
  530. if (reset($stk) == SERVICES_JSON_IN_ARR) { 
  531. // we are in an array, so just push an element onto the stack 
  532. array_push($arr, $this->decode($slice)); 
  533. } elseif (reset($stk) == SERVICES_JSON_IN_OBJ) { 
  534. // we are in an object, so figure 
  535. // out the property name and set an 
  536. // element in an associative array, 
  537. // for now 
  538. $parts = array(); 
  539. if (preg_match('/^/s*(["/'].*[^///]["/'])/s*:/s*(/S.*),?$/Uis', $slice, $parts)) { 
  540. // "name":value pair 
  541. $key = $this->decode($parts[1]); 
  542. $val = $this->decode($parts[2]); 
  543. if ($this->use & SERVICES_JSON_LOOSE_TYPE) { 
  544. $obj[$key] = $val; 
  545. else { 
  546. $obj->$key = $val; 
  547. } elseif (preg_match('/^/s*(/w+)/s*:/s*(/S.*),?$/Uis', $slice, $parts)) { 
  548. // name:value pair, where name is unquoted 
  549. $key = $parts[1]; 
  550. $val = $this->decode($parts[2]); 
  551. if ($this->use & SERVICES_JSON_LOOSE_TYPE) { 
  552. $obj[$key] = $val; 
  553. else { 
  554. $obj->$key = $val; 
  555. } elseif ((($chrs{$c} == '"') || ($chrs{$c} == "'")) && ($top['what'] != SERVICES_JSON_IN_STR)) { 
  556. // found a quote, and we are not inside a string 
  557. array_push($stk, array('what' => SERVICES_JSON_IN_STR, 'where' => $c, 'delim' => $chrs{$c})); 
  558. //print("Found start of string at {$c}/n"); 
  559. } elseif (($chrs{$c} == $top['delim']) && 
  560. ($top['what'] == SERVICES_JSON_IN_STR) && 
  561. ((strlen(substr($chrs, 0, $c)) - strlen(rtrim(substr($chrs, 0, $c), '//'))) % 2 != 1)) { 
  562. // found a quote, we're in a string, and it's not escaped 
  563. // we know that it's not escaped becase there is _not_ an 
  564. // odd number of backslashes at the end of the string so far 
  565. array_pop($stk); 
  566. //print("Found end of string at {$c}: ".substr($chrs, $top['where'], (1 + 1 + $c - $top['where']))."/n"); 
  567. } elseif (($chrs{$c} == '[') && 
  568. in_array($top['what'], array(SERVICES_JSON_SLICE, SERVICES_JSON_IN_ARR, SERVICES_JSON_IN_OBJ))) { 
  569. // found a left-bracket, and we are in an array, object, or slice 
  570. array_push($stk, array('what' => SERVICES_JSON_IN_ARR, 'where' => $c, 'delim' => false)); 
  571. //print("Found start of array at {$c}/n"); 
  572. } elseif (($chrs{$c} == ']') && ($top['what'] == SERVICES_JSON_IN_ARR)) { 
  573. // found a right-bracket, and we're in an array 
  574. array_pop($stk); 
  575. //print("Found end of array at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."/n"); 
  576. } elseif (($chrs{$c} == '{') && 
  577. in_array($top['what'], array(SERVICES_JSON_SLICE, SERVICES_JSON_IN_ARR, SERVICES_JSON_IN_OBJ))) { 
  578. // found a left-brace, and we are in an array, object, or slice 
  579. array_push($stk, array('what' => SERVICES_JSON_IN_OBJ, 'where' => $c, 'delim' => false)); 
  580. //print("Found start of object at {$c}/n"); 
  581. } elseif (($chrs{$c} == '}') && ($top['what'] == SERVICES_JSON_IN_OBJ)) { 
  582. // found a right-brace, and we're in an object 
  583. array_pop($stk); 
  584. //print("Found end of object at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."/n"); 
  585. } elseif (($substr_chrs_c_2 == '/*') && 
  586. in_array($top['what'], array(SERVICES_JSON_SLICE, SERVICES_JSON_IN_ARR, SERVICES_JSON_IN_OBJ))) { 
  587. // found a comment start, and we are in an array, object, or slice 
  588. array_push($stk, array('what' => SERVICES_JSON_IN_CMT, 'where' => $c, 'delim' => false)); 
  589. $c++; 
  590. //print("Found start of comment at {$c}/n"); 
  591. } elseif (($substr_chrs_c_2 == '*/') && ($top['what'] == SERVICES_JSON_IN_CMT)) { 
  592. // found a comment end, and we're in one now 
  593. array_pop($stk); 
  594. $c++; 
  595. for ($i = $top['where']; $i <= $c; ++$i) 
  596. $chrs = substr_replace($chrs, ' ', $i, 1); 
  597. //print("Found end of comment at {$c}: ".substr($chrs, $top['where'], (1 + $c - $top['where']))."/n"); 
  598. if (reset($stk) == SERVICES_JSON_IN_ARR) { 
  599. return $arr; 
  600. } elseif (reset($stk) == SERVICES_JSON_IN_OBJ) { 
  601. return $obj; 
  602. /** 
  603. * @todo Ultimately, this should just call PEAR::isError() 
  604. */ 
  605. function isError($data, $code = null
  606. if (class_exists('pear')) { 
  607. return PEAR::isError($data, $code); 
  608. } elseif (is_object($data) && (get_class($data) == 'services_json_error' || is_subclass_of($data, 'services_json_error'))) { 
  609. return true
  610. return false
  611. if (class_exists('PEAR_Error')) { 
  612. class Services_JSON_Error extends PEAR_Error 
  613. function Services_JSON_Error($message = 'unknown error', $code = null, $mode = null, $options = null, $userinfo = null
  614. parent::PEAR_Error($message, $code, $mode, $options, $userinfo); 
  615. else { 
  616. /** 
  617. * @todo Ultimately, this class shall be descended from PEAR_Error 
  618. */ 
  619. class Services_JSON_Error 
  620. function Services_JSON_Error($message = 'unknown error', $code = null, $mode = null, $options = null, $userinfo = null
  621. ?> 

希望本文所述对大家的php程序设计有所帮助。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表