目前, JSON 已经成为最流行的数据交换格式之一,各大网站的 API 几乎都支持它。
今天,我想总结一下 PHP 语言对它的支持,这是开发互联网应用程序(特别是编写 API )必须了解的知识。
从 5.2 版本开始, PHP 原生提供 json_encode() 和 json_decode() 函数,前者用于编码,后者用于解码。
一、 json_encode()
该函数主要用来将数组和对象,转换为 json 格式。先看一个数组转换的例子:
$arr = array ('a'=>1,'b'=>2,'c'=>3,'d'=>4,'e'=>5); echo json_encode($arr);
结果为
{"a":1,"b":2,"c":3,"d":4,"e":5}
再看一个对象转换的例子:
$obj->body = 'another post'; $obj->id = 21; $obj->approved = true; $obj->favorite_count = 1; $obj->status = NULL; echo json_encode($obj);
结果为
{ "body":"another post", "id":21, "approved":true, "favorite_count":1, "status":null }
由于 json 只接受 utf-8 编码的字符,所以 json_encode() 的参数必须是 utf-8 编码,否则会得到空字符或者 null 。当中文使用 GB2312 编码,或者外文使用 ISO-8859-1 编码的时候,这一点要特别注意。
二、索引数组和关联数组
PHP 支持两种数组,一种是只保存 ” 值 ” ( value )的索引数组( indexed array ),另一种是保存 ” 名值对 ” ( name/value )的关联数组( associative array )。
由于 javascript 不支持关联数组,所以 json_encode() 只将索引数组( indexed array )转为数组格式,而将关联数组( associative array )转为对象格式。
比如,现在有一个索引数组
$arr = Array('one', 'two', 'three'); echo json_encode($arr);
结果为:
["one","two","three"]
如果将它改为关联数组:
$arr = Array('1'=>'one', '2'=>'two', '3'=>'three'); echo json_encode($arr);
结果就变了:
{"1":"one","2":"two","3":"three"}
注意,数据格式从 “[]” (数组)变成了 “{}” (对象)。
如果你需要将 ” 索引数组 ” 强制转化成 ” 对象 ” ,可以这样写
json_encode( (object)$arr );
或者
json_encode ( $arr, JSON_FORCE_OBJECT );
三、类( class )的转换
下面是一个 PHP 的类:
class Foo { const ERROR_CODE = '404'; public $public_ex = 'this is public'; private $private_ex = 'this is private!'; protected $protected_ex = 'this should be protected'; public function getErrorCode() { return self::ERROR_CODE; } }
现在,对这个类的实例进行 json 转换:
$foo = new Foo; $foo_json = json_encode($foo); echo $foo_json;
输出结果是
{"public_ex":"this is public"}
可以看到,除了公开变量( public ),其他东西(常量、私有变量、方法等等)都遗失了。
四、 json_decode()
该函数用于将 json 文本转换为相应的 PHP 数据结构。下面是一个例子:
$json = '{"foo": 12345}'; $obj = json_decode($json); print $obj->{'foo'}; // 12345
通常情况下, json_decode() 总是返回一个 PHP 对象,而不是数组。比如:
$json = '{"a":1,"b":2,"c":3,"d":4,"e":5}'; var_dump(json_decode($json));
结果就是生成一个 PHP 对象:
object(stdClass)#1 (5) { ["a"] => int(1) ["b"] => int(2) ["c"] => int(3) ["d"] => int(4) ["e"] => int(5) }
如果想要强制生成 PHP 关联数组, json_decode() 需要加一个参数 true :
$json = '{"a":1,"b":2,"c":3,"d":4,"e":5}'; var_dump(json_decode($json,true));
结果就生成了一个关联数组:
array(5) { ["a"] => int(1) ["b"] => int(2) ["c"] => int(3) ["d"] => int(4) ["e"] => int(5) }
五、 json_decode() 的常见错误
下面三种 json 写法都是错的,你能看出错在哪里吗?
$bad_json = "{ 'bar': 'baz' }"; $bad_json = '{ bar: "baz" }'; $bad_json = '{ "bar": "baz", }';
对这三个字符串执行 json_decode() 都将返回 null ,并且报错。
第一个的错误是, json 的分隔符( delimiter )只允许使用双引号,不能使用单引号。第二个的错误是, json 名值对的 ” 名 ” (冒号左边的部分),任何情况下都必须使用双引号。第三个的错误是,最后一个值之后不能添加逗号( trailing comma )。
另外, json 只能用来表示对象( object )和数组( array ),如果对一个字符串或数值使用 json_decode() ,将会返回 null 。
var_dump(json_decode("Hello World")); //null
六、参考材料
[1] PHP Manual
[2] Ed Finkler, JSON is Everybody’s Friend