数据类型和Json格式.docx
《数据类型和Json格式.docx》由会员分享,可在线阅读,更多相关《数据类型和Json格式.docx(13页珍藏版)》请在冰豆网上搜索。
数据类型和Json格式
数据类型和Json格式
作者:
阮一峰
日期:
2009年5月30日
1.
前几天,我才知道有一种简化的数据交换格式,叫做yaml。
我翻了一遍它的文档,看懂的地方不多,但是有一句话令我茅塞顿开。
它说,从结构上看,所有的数据(data)最终都可以分解成三种类型:
第一种类型是标量(scalar),也就是一个单独的字符串(string)或数字(numbers),比如"北京"这个单独的词。
第二种类型是序列(sequence),也就是若干个相关的数据按照一定顺序并列在一起,又叫做数组(array)或列表(List),比如"北京,上海"。
第三种类型是映射(mapping),也就是一个名/值对(Name/value),即数据有一个名称,还有一个与之相对应的值,这又称作散列(hash)或字典(dictionary),比如"首都:
北京"。
我恍然大悟,数据构成的最小单位原来如此简单!
难怪在编程语言中,只要有了数组(array)和对象(object)就能够储存一切数据了。
2.
我马上想到了json。
21世纪初,DouglasCrockford寻找一种简便的数据交换格式,能够在服务器之间交换数据。
当时通用的数据交换语言是XML,但是DouglasCrockford觉得XML的生成和解析都太麻烦,所以他提出了一种简化格式,也就是Json。
Json的规格非常简单,只用一个页面几百个字就能说清楚,而且DouglasCrockford声称这个规格永远不必升级,因为该规定的都规定了。
1)并列的数据之间用逗号(",")分隔。
2)映射用冒号(":
")表示。
3)并列数据的集合(数组)用方括号("[]")表示。
4)映射的集合(对象)用大括号("{}")表示。
上面四条规则,就是Json格式的所有内容。
比如,下面这句话:
"北京市的面积为16800平方公里,常住人口1600万人。
上海市的面积为6400平方公里,常住人口1800万。
"
写成json格式就是这样:
[
{"城市":
"北京","面积":
16800,"人口":
1600},
{"城市":
"上海","面积":
6400,"人口":
1800}
]
如果事先知道数据的结构,上面的写法还可以进一步简化:
[
["北京",16800,1600],
["上海",6400,1800]
]
由此可以看到,json非常易学易用。
所以,在短短几年中,它就取代xml,成为了互联网上最受欢迎的数据交换格式。
我猜想,DouglasCrockford一定事先就知道,数据结构可以简化成三种形式,否则怎么可能将json定义得如此精炼呢!
3.
学习javascript的时候,我曾经一度搞不清楚"数组"(array)和"对象"(object)的根本区别在哪里,两者都可以用来表示数据的集合。
比如有一个数组a=[1,2,3,4],还有一个对象a={0:
1,1:
2,2:
3,3:
4},然后你运行alert(a[1]),两种情况下的运行结果是相同的!
这就是说,数据集合既可以用数组表示,也可以用对象表示,那么我到底该用哪一种呢?
我后来才知道,数组表示有序数据的集合,而对象表示无序数据的集合。
如果数据的顺序很重要,就用数组,否则就用对象。
4.
当然,数组和对象的另一个区别是,数组的数据没有"名称"(name),对象的数据有"名称"(name)。
但是问题是,很多编程语言中,都有一种叫做"关联数组"(associativearray)的东西。
这种数组中的数据是有名称的。
比如在javascript中,可以这样定义一个对象:
vara={"城市":
"北京","面积":
16800,"人口":
1600};
但是,也可以定义成一个关联数组:
a["城市"]="北京";
a["面积"]=16800;
a["人口"]=1600;
这起初也加剧了我对数组和对象的混淆,后来才明白,在Javascript语言中,关联数组就是对象,对象就是关联数组。
这一点与php语言完全不同,在php中,关联数组也是数组。
比如运行下面这段javascript:
vara=[1,2,3,4];
a['foo']='HelloWorld';
alert(a.length);
最后的结果是4,也就是说,数组a的元素个数是4个。
但是,运行同样内容的php代码就不一样了:
php
$a=array(1,2,3,4);
$a["foo"]="Helloworld";
echocount($a);
?
>
最后的结果是5,也就是说,数组a的元素个数是5个。
(完)
在PHP语言中使用JSON
作者:
阮一峰
日期:
2011年1月14日
目前,JSON已经成为最流行的数据交换格式之一,各大网站的API几乎都支持它。
我写过一篇《数据类型和JSON格式》,探讨它的设计思想。
今天,我想总结一下PHP语言对它的支持,这是开发互联网应用程序(特别是编写API)必须了解的知识。
从5.2版本开始,PHP原生提供)和)函数,前者用于编码,后者用于解码。
一、)
该函数主要用来将数组和对象,转换为json格式。
先看一个数组转换的例子:
$arr=array('a'=>1,'b'=>2,'c'=>3,'d'=>4,'e'=>5);
echo$arr);
结果为
{"a":
1,"b":
2,"c":
3,"d":
4,"e":
5}
再看一个对象转换的例子:
$obj->body='anotherpost';
$obj->id=21;
$obj->approved=true;
$obj->favorite_count=1;
$obj->status=NULL;
echo$obj);
结果为
{
"body":
"anotherpost",
"id":
21,
"approved":
true,
"favorite_count":
1,
"status":
null
}
由于json只接受utf-8编码的字符,所以)的参数必须是utf-8编码,否则会得到空字符或者null。
当中文使用GB2312编码,或者外文使用ISO-8859-1编码的时候,这一点要特别注意。
二、索引数组和关联数组
PHP支持两种数组,一种是只保存"值"(value)的索引数组(indexedarray),另一种是保存"名值对"(name/value)的关联数组(associativearray)。
由于javascript不支持关联数组,所以)只将索引数组(indexedarray)转为数组格式,而将关联数组(associativearray)转为对象格式。
比如,现在有一个索引数组
$arr=Array('one','two','three');
echo$arr);
结果为:
["one","two","three"]
如果将它改为关联数组:
$arr=Array('1'=>'one','2'=>'two','3'=>'three');
echo$arr);
结果就变了:
{"1":
"one","2":
"two","3":
"three"}
注意,数据格式从"[]"(数组)变成了"{}"(对象)。
如果你需要将"索引数组"强制转化成"对象",可以这样写
(object)$arr);
或者
json_encode($arr,JSON_FORCE_OBJECT);
三、类(class)的转换
下面是一个PHP的类:
classFoo{
constERROR_CODE='404';
public$public_ex='thisispublic';
private$private_ex='thisisprivate!
';
protected$protected_ex='thisshouldbeprotected';
publicfunctiongetErrorCode(){
returnself:
:
ERROR_CODE;
}
}
现在,对这个类的实例进行json转换:
$foo=newFoo;
$foo_json=$foo);
echo$foo_json;
输出结果是
{"public_ex":
"thisispublic"}
可以看到,除了公开变量(public),其他东西(常量、私有变量、方法等等)都遗失了。
四、)
该函数用于将json文本转换为相应的PHP数据结构。
下面是一个例子:
$json='{"foo":
12345}';
$obj=$json);
print$obj->{'foo'};//12345
通常情况下,)总是返回一个PHP对象,而不是数组。
比如:
$json='{"a":
1,"b":
2,"c":
3,"d":
4,"e":
5}';
var_dump($json));
结果就是生成一个PHP对象:
object(stdClass)#1(5){
["a"]=>int
(1)
["b"]=>int
(2)
["c"]=>int(3)
["d"]=>int(4)
["e"]=>int(5)
}
如果想要强制生成PHP关联数组,)需要加一个参数true:
$json='{"a":
1,"b":
2,"c":
3,"d":
4,"e":
5}';
var_dump($json,true));
结果就生成了一个关联数组:
array(5){
["a"]=>int
(1)
["b"]=>int
(2)
["c"]=>int(3)
["d"]=>int(4)
["e"]=>int(5)
}
五、)的常见错误
下面三种json写法都是错的,你能看出错在哪里吗?
$bad_json="{'bar':
'baz'}";
$bad_json='{bar:
"baz"}';
$bad_json='{"bar":
"baz",}';
对这三个字符串执行)都将返回null,并且报错。
第一个的错误是,json的分隔符(delimiter)只允许使用双引号,不能使用单引号。
第二个的错误是,json名值对的"名"(冒号左边的部分),任何情况下都必须使用双引号。
第三个的错误是,最后一个值之后不能添加逗号(trailingcomma)。
另外,json只能用来表示对象(object)和数组(array),如果对一个字符串或数值使用),将会返回null。
var_dump("HelloWorld"));//null
六、参考材料
[1]PHPManual
[2]EdFinkler,JSONisEverybody'sFriend
(完)
php和js如何通过json互相传递数据相关问题探讨
当我们在结合php和javascript实现某些功能时,经常会用到json。
json是js的一种数据格式,可以直接被js解析。
而php无法直接读取json数据,但是php提供了json_decode函数来对json数据进行转化,从而可以被php脚本访问。
同时,php也提供了json_encode函数来将数据转化成json格式。
那么,js中的原生json与php中通过json_encode函数转化后的json是否完全一样呢?
今天,站长就和大家一起来探讨这个问题。
我们通过php向javascript传递数组数据时,通常要将其转化为json格式,以便javascript来获取,那么我们就以数组为例,先来看一下两者之间的区别。
1、一维数组
考虑php数组
$array=array("1","2","3");
使用json_encode函数转化后,对应的json字符串为
复制代码代码如下:
["1","2","3"]。
细心的朋友很快就发现,转化后得到的json字符串,就是javascript中的数组形式,那么是否可以用js的数组访问方式来访问呢?
当然是可以的,但是你将这个json字符串传递给给js时,需要使用urlencode函数对其编码,如:
代码如下:
show('
phpechourlencode(array('1','2','3')));?
>')"id="aj">访问json
我们可以用下面的js代码来验证:
复制代码代码如下:
functionshow(str){
varjobj=eval_r(decodeURI(str));
alert(jobj[2]);
}
大家自己试一试就会发现,是的,可以用js中访问一维数组的方式来访问它。
eval方法将json字符串解释为json对象,因为传递过来的是字符串,不转化的话,你得到将是字符串中第三个字符的值。
我们再来对这个一维数组做一下变化,我们发现上面的一维数组没有指定索引,所以它默认为数字索引,现在我们来给它加上键名:
考虑php数组
$array=array('a'=>'1','b'=>'2','c'=>'3');
使用json_encode函数转化后,对应的json字符串为
{"a":
"1","b":
"2","c":
"3"}
。
我们很快就发现了其中的不同,最明显的就是字符串两端的[]变成了{},那么这个字符串是否也可以按上面那样处理后被js访问呢?
我们不防试一试:
show('
phpechourlencode(array('a'=>'1','b'=>'2','c'=>'3')));?
>')"id="aj">访问json
functionshow(str){
varjobj=eval_r(decodeURI(str));
alert(jobj.a);
}
大家如果动手试了就知道,点击链接后,没有出现弹窗。
为什么呢?
是PHP生成的json字符串格式不对吗?
不是的,这是我们在使用eval函数解释的时候,出错了。
把上面的函数代码换成:
复制代码代码如下:
functionshow(str){
varjobj=eval_r('('+decodeURI(str)+')');
alert(jobj.a);
}
再试试吧!
怎么样,可以访问了吧。
这告诉我们,在使用eval方法处理带有键名的json字符串时,需要在字符串两端加速括号。
至于为什么,站长也不知道,站在巨人的肩膀上而已。
这里要注意,尽管PHP生成的json字符串
{"a":
"1","b":
"2","c":
"3"}被传递给js后无法被直接解释为json格式,但是如果你在js中使用该字符串直接创建json数据,是可以的。
试试下面的代码吧:
复制代码代码如下:
varjobj={"a":
"1","b":
"2","c":
"3"};
alert(jobj.b);
2、二维数组
二维数组在PHP用的应用非常广泛,因此了解二维数组转化后的json格式非常重要。
有了上面的例子做铺垫,下面站长就直接给出示例代码:
复制代码代码如下:
show('
phpechourlencode(array(array('1','2','3'))));?
>')"id="aj">访问json
functionshow(str){
varjobj=eval_r(decodeURI(str));
alert(jobj[0][0]);
}
大家运行,会发现,这跟一维数组差不多,这是不带键名的例子,因此在show函数中,去掉字符串两端的括号也是可以的。
下面,我们对二维数组进行一下变化,在第二维中加入键名,请看示例代码:
复制代码代码如下:
show('
phpechourlencode(array(array("a"=>'1',"b"=>'2','3'))));?
>')"id="aj">访问json
functionshow(str){
varjobj=eval_r('('+decodeURI(str)+')');
alert(jobj[0].a);
}
大家运行代码后,发现,这里我们访问json数据的方式有点不一样。
上面我们用的是
alert(jobj[0][0]);
而这里我们用的是
alert(jobj[0].a);不要问我为什么,就是这样。
这就是json的访问方式。
上面的例子,我们对二维数组的第二维添加了键名,下面我们对第一维添加键名,看看访问方式又有什么不同:
复制代码代码如下:
show('
phpechourlencode(array('k'=>array('1','2','3'))));?
>')"id="aj">访问json
functionshow(str){
varjobj=eval_r('('+decodeURI(str)+')');
alert(jobj.k[1]);
}
这里我们使用的是
jobj.k[1]这样的方式,大家一定已经发现了,只要数组中含有键名,当数组被转化为json格式后,就要使用
json对象.键名
这样的方式来访问该键下的元素,上面的例子中,k键下的数组元素是数字索引,所以在json中使用k[1]这样的方式来访问。
下面,我们对数组的第一维和第二维都添加键名:
复制代码代码如下:
show('
phpechourlencode(array('k'=>array("a"=>'1','2','3'))));?
>')"id="aj">访问json
functionshow(str){
varjobj=eval_r('('+decodeURI(str)+')');
alert(jobj.k.a);
}
正如上面所提到的,只要含有键名,就必须以
json对象.键名
的方式来访问,如果有多个键就要用
json对象.键名.键名...
,不要问我为什么,这就是json的访问方式,只有javascript的发明者能向你解释,他为什么要这样规定。
结论:
1、将php中的数组转化为json字符串传递给js时。
如果数组没有指定键名,那么可以直接使用js的eval方法将其转化为json格式供js处理;如果数组中含有键名,那么在使用eval方法处理时,需要使用()将json字符串括起来。
2、如果数组中含有键名,转化为json字符串后,在js中要用
json对象.键名.键名...
的方式来访问,如果是数字索引则用
json对象[1]
或者
json对象.键名[1]
这样的方式。
上面,我们主要讨论了,在PHP向js传递json字符串时,需要注意的事项。
下面我们再来讨论,用js向php传递json字符串时需要如何处理。
聪明的你肯定已经知道了,只要将json数据用引号引起来作为字符串传递给PHP【通常用ajax进行】就可以用json_decode函数解码了。
没错!
就是这样!
但是在构造json字符串的时候一定要仔细,如果你不经常构造json字符串,那么不妨用
echoarray('k'=>array("a"=>'1','2','3')))
这样的方式,查看你需要构造的目标字符串的json格式。
这样你就可以在js中根据你想要的结果来构造了!
好了,今天对php和js之间如何使用json数据进行通信就讨论到这里,大家可以自己再试试将php的对象类型进行json编码后如何传递给js。