-
- static PHP_FUNCTION(json_encode)
- {
- zval *parameter;
- smart_str buf = {0};
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z|l", ¶meter, &options) == FAILURE) {
- return;
- }
- php_json_encode(&buf, parameter, options TSRMLS_CC);
-
- ZVAL_STRINGL(return_value, buf.c, buf.len, 1);
- 🎜> }
-
-
- 複製程式碼
-
-
- JSON_G(error_code) = PHP_JSON_ERROR_NONE;
是定義的json錯誤,該錯誤可以透過json_last_error函數獲取,你用過嗎?反正我沒用過。
php_json_encode是主要的操作
PHP_JSON_API void php_json_encode(smart_str *buf, zval *val, int options TSRMLS_DC) /* {str* Z_TYPE_P(val)) { case IS_NULL: smart_str_appendl(buf, "null", 4); //輸出NULL break: -
- case IS_BOOL:
- if (Z_BVAL_P(val)) {
- smart_str_appendl(buf, "true", 4);//輸出true
- } else {
- smart_str_appendl(buf, "false", 5);
- }
- break;
-
- }
- break;
-
- case IS_LONG:
- smart_str_append_long(buf, Z_LVAL_P(val));//輸出長整形的值
- break;
-
-
- >; IS_DOUBLE:
- {
- char *d = NULL;
- int len;
- double dbl = Z_DVAL_P(val);
-
- if (!zend_isinf(val);
-
- if (!zend_isinf(dbl) && ! )) {//非無窮無盡
- len = spprintf(&d, 0, "%.*k", (int) EG(precision), dbl);
- smart_str_appendl(buf, d, len);
- efree(d);
- } else {
- php_error_docref(NULL TSRMLS_CC, E_WARNING, "double %.9g does not conform to the JSON spec, encoded as 0", dblm }
- }
- break;
-
- case IS_STRING://字串
- json_escape_string(buf, Z_STRVAL_P(val), Z_STRLEN_P(val), opbuf TSRMLS_CC);
- break;
-
- case IS_ARRAY://陣列與物件
- case IS_OBJECT:
- json_encode_array(buf, &val, options TSRMLS_CC) default:
- php_error_docref(NULL TSRMLS_CC, E_WARNING, "type is unsupported, encoded as null");
- smart_str_appendl(buf, "nullull");
- smart_str_appendl(buf, "nullull"); >
- return;
- }
-
-
- 複製程式碼
-
-
很明顯,根據不同的類型,會有對應的case。
最複雜的是 字串 、陣列 、物件這三種類型,陣列和物件是同一種操作。
先看看字串吧,很長,註解直接寫在程式碼裡。
-
- //options應該是5.3版本之後才支援的,由以下常數組成的二進位遮罩: JSON_HEX_QUOT, JSON_HEX_TAG, JSON_HEX_AMP, JSLSON_MER_AMP ASHES, JSON_FORCE_OBJECT, JSON_UNESCAPED_UNICODE.雖然我沒用過。 。 。
- static void json_escape_string(smart_str *buf, char *s, int len, int options TSRMLS_DC) /* {{{ */
- {
- int pos = 0; > unsigned short *utf16;
-
- if (len == 0) {//如果長度為0,則直接回傳雙引號""
- smart_str_appendl(buf, """", 2);
- return;
- }
-
- if (options & PHP_JSON_NUMERIC_CHECK) {//偵測是否為0-9的數字,如果是數字,那麼就會直接把資料傳回為long或double型別。
- double d;
- int type;
- long p;
-
- if ((type = is_numeric_string(s, len, &p, &d, 0)) != 0) {
- if (type == IS_LONG) {
- smart_str_append_long(buf, p);
- } else if (type == IS_DOUBLE) {
- if (!zend_isinf(d) && !zend_isnan(d)) { 🎜> char *tmp;
- int l = spprintf(&tmp, 0, "%.*k", (int) EG(precision), d);
- smart_str_appendl(buf, tmp, l);
- efree(tmp);
- } else {
- php_error_docref(NULL TSRMLS_CC, E_WARNING, "double %.9g does not conform to the JSON spec, encoded as 0", d3); '0');
- }
- }
- return;
- }
-
- }
-
- utf16 = (unsigned short *) safe_emalloc(lenlen, sort ), 0);
- len = utf8_to_utf16(utf16, s, len); //這裡會對你輸入的值一次處理轉成對應的Dec碼,例如1是49,a是97這樣的,保存到utf16中。
- if (len if (utf16) {
- efree(utf16);
- }
- if (len JSON_G(error_code) = PHP_JorsSON_ERROR_UTF8
- ifors_code); ) {
- php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid UTF-8 sequence in argument");
- }
- smart_str_appendl(buf, "null", 4); smart_str_appendl(buf, """", 2);
- }
- return;
- }
-
- smart_str_appendc(buf, '"'); //輸入"
-
- //下面這段程式碼就是將一些特殊字元轉義如雙引號,反斜線等等
- while (pos {
- us = utf16[pos++];
-
- switch (us)
- {
- case '"':
- if (options & PHP_JSON_HEX_QUOT) {
- smart_str_appendl(buf, "\u0022", 6);
- } > smart_str_appendl(buf, "\"", 2);
- }
- break;
-
- case '\':
- smart_str_appendl(buf, "\\", 2); > break;
- case '/':
- smart_str_appendl(buf, "\/", 2);
- break;
-
- case 'b':
- smart_str_str_str \b", 2);
- break;
-
- case 'f':
- smart_str_appendl(buf, "\f", 2);
- break;
-
- case 'n':
- smart_str_appendl(buf, "\n", 2);
- break;
-
- case 'r':
- smart_str_appendl(buf, "\r", 2);
- break;
-
- case 't':
- smart_str_appendl(buf, "\t", 2);
- break;
-
- case ' if (options & PHP_JSON_HEX_TAG) {
- smart_str_appendl(buf, "\u003C", 6);
- } else {
- smart_str_appendc(buf, '
- case '>':
- if (options & PHP_JSON_HEX_TAG) {
- smart_str_appendl(buf, "\u003E", 6);
- } ');
- }
- break;
-
- case '&':
- if (options & PHP_JSON_HEX_AMP) {
- smart_str_appendl(buf, "\u0026", 66); > } else {
- smart_str_appendc(buf, '&');
- }
- break;
-
- case ''':
- if (options & PHP_JSON_HEXL_APOS (buf, "\u0027", 6);
- } else {
- smart_str_appendc(buf, ''');
- }
- break;
-
- default: //一直到這裡,沒有特殊字元就會把值append到buf
- if (us >= ' ' && (us & 127) == us) {
- smart_str_appendc(buf, (unsigned char) us);
- }else {
- smart_str_appendl(buf, "\u", 2);
- 我們= REVERSE16(我們);
-
- smart_str_appendc(buf,digits[us & ((1 我們>>= 4;
- smart_str_appendc(buf, 數字[us & ((1 我們>>= 4;
- smart_str_appendcc (buf, 數字[us & ((1 我們>>= 4;
- smart_str_appendc(buf, 數字[us & ((1 }
- 休息;
- }
- }
- smart_str_appendc(buf, '"'); //結束雙引號。
- efree(utf16);
- }
複製程式碼
再來看看陣列和對象,也很簡單,
-
- static void json_encode_array(smart_str *buf, zval **val, int options TSRMLS_DC) /* {{{ */
- { 雜湊表*myht;
-
- if (Z_TYPE_PP(val) == IS_ARRAY) {
- myht = HASH_OF(*val);
- r = (選項& PHP_JSON_FORCE_OBJ) : json_define_array_type(val TSRMLS_CC);
- } else {
- myht = Z_OBJPROP_PP(val);
- r = PHP_JSON_OUTPUT_OBJECT;
- }
- php_error_docref(NULL TSRMLS_CC, E_WARNING, "偵測到遞迴");
- smart_str_appendl(buf, "null", 4);
- 回傳;
- }
- } if (r == PHP_JSON_OUTPUT_ARRAY) {
- smart_str_appendc(buf, '[');
- } else {
- smart_str_appendc(buf, '{');我=我的? zend_hash_num_elements(myht) : 0;
-
- if (i > 0)
- {
- char *key;
- zval **數據;
- ulong索引;
- uint key_len;
- HashPosition pos;
- 雜湊表*tmp_ht;
- int need_comma = 0;
-
- zend_hash_internal_pointer_reset_ex(myht, &pos);
- for運輸; zend_hash_move_forward_ex(myht, &pos)) {
- i = zend_hash_get_current_key_ex(myht, &key, &key_len, &index, 0, &pos);
- if (i == HASH_len, &index, 0, &pos);
- if (i == HASH_KEY_NON_ASH_ANT; if (zend_hash_get_current_data_ex(myht, (void **) &data, &pos) == SUCCESS) {
- tmp_ht = HASH_OF(*data);
- if (tmp_ht) {
- tmp_ht-> }
-
- if (r == PHP_JSON_OUTPUT_ARRAY) {
- if (r == PHP_JSON_OUTPUT_ARRAY) {
- if (need_comma) {
- smart_str_appendc(buf, ',');
- } else { <.com> }
- //將值追加到buf 中
- php_json_encode(buf, *data, options TSRMLS_CC);
- } else if (r == PHP_JSON_OUTPUT_OBJECT) {
- _liter {
- if (key[0] == ' ' && Z_TYPE_PP(val) == IS_OBJECT) {
- /* 跳過受保護和私有成員。 */
- if (tmp_ht) {
- tmp_ht->nApplyCount--;
- }
- 繼續;
- }
-
- if (need_comma) {
- smart_str , ',');
- } else {
- need_comma = 1;
- }
-
- json_escape_string(buf, key, key_len - 1, options & ~PHP_JSON_NUMERy_PCK; (buf, ':');
-
- php_json_encode(buf, *data, options TSRMLS_CC);
- } else {
- if (need_comma) {
- smart_str. ;
- } else {
- need_comma = 1;
- }
-
- smart_str_appendc(buf, '"');
- smart_str_append_long(buf, (long) 索引);
- smart_str_appendc(buf, '"');
- smart_strm' );
-
- php_json_encode(buf, *data, options TSRMLS_CC);
- }
- }
-
- if (tmp_ht) {
- }
-
- if (tmp_ht) {
- > }
- }
- }
- }
- // 結束標籤
- if (r == PHP_JSON_OUTPUT_ARRAY) {
- smart_str_appendc(buf, ']');
- }
- smart_str_appendc(buf, '}');
} }
複製程式碼
透過簡單的分析,說明了一個問題,相當於上面用sprintf的方法其實是一樣的,都是拼接字串,
而且為了效能,更應該鼓勵用sprintf來拼接json格式,
| 因為 json_encode 會進行很多循環操作,而且所消耗的效能是線性的 O(n)。