最近做个IM类型的Android 应用,由于有三种客户端(pc,ios,Android),所以底层使用的是C++与服务器通信,所以通信部分基本上有c++完成,封装好Jni即可,可以把底层c++通信看成一个httpclient,Android上面只需要关注UI显示即可。
好的,交代好项目背景,下面介绍为什么使用gson。
由于底层使用c++socket通信,通信的协议还是,最原始的那种,比如先顶一个4个字节消息长度,然后根据长度,读取这么长度的数据,然后取出通信的,消息号码,然后读出里面的数据。
o,my god,做久了java,感觉使用了这个,好像回到的原始社会了。
所以我们定义java与c++之间通信使用jni,传输的协议全部都是字符串,字符串里面使用json。
做久了J2ee,所以很讲究封装。大概才传输协议时这样。
cmd是通信的消息编码,param:表示一个随机的标示,由于通信是异步的,所以req对于的resp,需要额外的添加一个标示符,来找到对于的resp。data里面就是一个通信的的内容,resp里面添加一个字段,retcode,表示服务器的响应,0为成功,1表示网络问题,这个根据实际需要来定义
req:{"cmd":100,"param":"0cee5015-aae6-4f67-b03e-7e95168b6563","data":{xxxxxxx}}
resp{"cmd":100,"retCode":0,"param":"f56f8e16-ba82-429f-b724-8e4da6d4ac03","data":{xxxxx}}
这样的简单通信协议相信很多人都是懂的,关键是data里面的数据时不一样的,项目需要根据cmd的不同,来进行分发,然后解析。
如果使用Android自带的json解析类,那么要化很多时间进行解析,所以在这里使用gson,gson是google的开源json解析工具。官网是http://code.google.com/p/google-gson/
gson使用还是比较简单的,比如:
public class User {
public Name name;
public String passwd;
public int age;
public char sex;
public String[] telphone;
public String toString() {
return "User [name=" + name + ", passwd=" + passwd + ", age=" + age + ", sex=" + sex + ", telphone=" + Arrays.toString(telphone) + "]";
}
}
public class Name {
public String firstName;
public String lastName;
public String toString() {
return "Name [firstName=" + firstName + ", lastName=" + lastName + "]";
}
}
public class Main {
public static void main(String[] args) {
Gson gson = new Gson();
Name n = new Name();
n.firstName = "liushui";
n.lastName = "bufu";
User u = new User();
u.age = 25;
u.name = n;
u.passwd = "123456";
u.sex = 'M';
String[] phone = new String[10];
for (int i = 0; i < phone.length; i++) {
phone[i] = "1381234567" + i;
}
u.telphone = phone;
System.out.println(u);
String json = gson.toJson(u);
System.out.println(json);
User u2 = gson.fromJson(json, User.class);
System.out.println(u2);
}
}
代码运行的结果是:
User [name=Name [firstName=liushui, lastName=bufu], passwd=123456, age=25, sex=M, telphone=[13812345670, 13812345671, 13812345672, 13812345673, 13812345674, 13812345675, 13812345676, 13812345677, 13812345678, 13812345679]]
{"name":{"firstName":"liushui","lastName":"bufu"},"passwd":"123456","age":25,"sex":"M","telphone":["13812345670","13812345671","13812345672","13812345673","13812345674","13812345675","13812345676","13812345677","13812345678","13812345679"]}
User [name=Name [firstName=liushui, lastName=bufu], passwd=123456, age=25, sex=M, telphone=[13812345670, 13812345671, 13812345672, 13812345673, 13812345674, 13812345675, 13812345676, 13812345677, 13812345678, 13812345679]]
gson很好完成json的解析与生成,很好的完成了数据的序列化与反序列化。
gson默认使用的是字段的的名字,在这里我没有使用setter和getter方法,直接把字段进行的public处理,实际情况需要根据项目需要来使用,有的时候项目需要使用代码混淆的时候需要注意下,一种解决方法是不把字段进行混淆,或者是使用gson提供的标注来解决。
对上面的代码进行修改,可以得到如下结果。
public class User {
@SerializedName("_name")
public Name name;
@SerializedName("_passwd")
public String passwd;
@SerializedName("_age")
public int age;
@SerializedName("_sex")
public char sex;
@SerializedName("_telphone")
public String[] telphone;
public String toString() {
return "User [name=" + name + ", passwd=" + passwd + ", age=" + age + ", sex=" + sex + ", telphone=" + Arrays.toString(telphone) + "]";
}
}
User [name=Name [firstName=liushui, lastName=bufu], passwd=123456, age=25, sex=M, telphone=[13812345670, 13812345671, 13812345672, 13812345673, 13812345674, 13812345675, 13812345676, 13812345677, 13812345678, 13812345679]]
{"_name":{"firstName":"liushui","lastName":"bufu"},"_passwd":"123456","_age":25,"_sex":"M","_telphone":["13812345670","13812345671","13812345672","13812345673","13812345674","13812345675","13812345676","13812345677","13812345678","13812345679"]}
User [name=Name [firstName=liushui, lastName=bufu], passwd=123456, age=25, sex=M, telphone=[13812345670, 13812345671, 13812345672, 13812345673, 13812345674, 13812345675, 13812345676, 13812345677, 13812345678, 13812345679]]
生成的json字符串是带有下划线的,也就是@SerializedName里面定义的别名。
另外上面所说的通信协议中的data的结构体是不确定的,有的时候是一个字符串,或者是一个实体类或者是数组,为了满足面向对象的规则,可以使用json里面的JsonObject来表示,这样可以根据json里面的内容,自动匹配,再结合java中的泛型,既可以完美解决通信模块,程序员可以竟可能的把经历放在ui和逻辑上,通信和json的解析,可以关心的很少。