关于通过webclient和JSON格式报文与服务器之间通讯的解决方法和遇到的难题
话说做这个很久了,不过我比较磨洋工。所以托了很久,今天才真正跑通了整个流程。
一开始老师要我做的时候,我有点没明白意思,后来看了msdn后明白了不少。
我要做的是用一句话来概括就是,通过JSON来与后台的J2EE通讯,操作数据。
但就是这样一个简单的任务,我大概有认真做了一个礼拜吧。
下面说说主要的实现方法:
首先我们要提到webclient这个类,这是system自带的一个模仿web浏览器的类,用它来模仿POST访问(其实还能GET、PUT、DELETE)。
具体的思路是:首先实体化一个webclient,打开端口下载或者上传JSON字符串,然后通过绑定实体类的方法来序列化或者反序列化JSON报文,最后做到我们需要的数据操作。
具体代码如下:
1、首先我们需要绑定实体类,之后用微软的自带类库来解析(也可以用JSON.NET这个开源的第三方类库,不过有些方法我还不懂)
[DataContract] public class jsontext { [DataMember(Order = 0, IsRequired = true)] public int total { get; set; } [DataMember(Order = 1, IsRequired = true)] public weatherInfo[] rows { get; set; } } [DataContract] public class weatherInfo { [DataMember(Order = 0, IsRequired = true)] public int id { get; set; } [DataMember(Order = 1)] public double weatherType { get; set; } [DataMember(Order = 2)] public string weatherName { get; set; } [DataMember(Order = 3)] public double maxTemperature { get; set; } [DataMember(Order = 4)] public double minTemperature { get; set; } [DataMember(Order = 5)] public double rainFall { get; set; } [DataMember(Order = 6)] public double snowFall { get; set; } } [DataContract] public class weatherInsert { [DataMember(Order = 0, Name = "weatherInfo.iweatherType")] public double iweatherType { get; set; } [DataMember(Order = 1, Name = "weatherInfo.cweatherName")] public string cweatherName { get; set; } [DataMember(Order = 2, Name = "weatherInfo.imaxTemperature")] public double imaxTemperature { get; set; } [DataMember(Order = 3, Name = "weatherInfo.iminTemperature")] public double iminTemperature { get; set; } [DataMember(Order = 4, Name = "weatherInfo.irainFall")] public double irainFall { get; set; } [DataMember(Order = 5, Name = "weatherInfo.isnowFall")] public double isnowFall { get; set; } }
2、然后实例一个webclinet
WebClient webclient = new WebClient(); if (!webclient.IsBusy) { webclient.Encoding = System.Text.Encoding.UTF8;//防止乱码 json = webclient.DownloadString(address); }
3、序列化和反序列化,这里我就写在一起了
public static class iotmonSerial { //反序列化 public static T parse<T>(string jsonstring) { using (var ms = new MemoryStream(Encoding.UTF8.GetBytes(jsonstring))) { return (T)new DataContractJsonSerializer(typeof(T)).ReadObject(ms); } } //序列化 public static string stringify(object jsonObject) { using (var ms = new MemoryStream()) { new DataContractJsonSerializer(jsonObject.GetType()).WriteObject(ms, jsonObject); return Encoding.UTF8.GetString(ms.ToArray()); } } }
4、通过调用前面的方法以及实体类,真正的解析了JSON,当然现在只是下载了数据,之后我会说如何上传。
var ppp = iotmonSerial.parse<jsontext>(json); foreach (var item in ppp.rows) { myTextBlock.Text += item.id + "\t" + item.weatherType + "\t" + item.weatherName + "\n"; }
5、然后我们来说说怎么上传数据吧~(太艰辛了)
Uri address = new Uri("http://localhost:8080/iotMon/WeatherInfo/listWeatherInfoPaged.action"); WebClient webClient = new WebClient(); webClient.UploadStringAsync(address, "POST", testString);
看上去很简单吧,但是这才是最坑人的地方,因为弄完之后,我发现死活传不上去。我比对了很多次JSP页面上发送的字符串,都是一模一样的。
这是我苦恼了很久的问题,最后和老师调试了好久,终于找到了问题所在。
首先得定义发过去字符串的标头,详见http://msdn.microsoft.com/zh-cn/library/system.net.webclient(v=vs.95).aspx(最后有提到)
但是坑爹的MSDN也没有说解决方案,最后还是感谢强大的百度和谷歌!
标头是如下(不同的服务器可能不一样,需要调整,第一个是保证编码是UTF-8的,我们的项目UTF-8的):
webClient.Encoding = System.Text.Encoding.UTF8; webClient.Headers.Add(HttpRequestHeader.Accept, "json"); webClient.Headers.Add(HttpRequestHeader.ContentType,"application/x-www-form-urlencoded; charset=UTF-8"); webClient.Headers.Add(HttpRequestHeader.UserAgent, "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)");
然后这么折腾之后我发现还是不行,最后通过浏览器看上传过去的数据,发现了问题所在!
url发送过去的东西,会有一个url编码的转换,JSON报文里面最多的“{”和“[”都被转换成了“%7B”和“%5B”,所以我们需要Microsoft.JScript.GlobalObject.encodeURIComponent来将已经做好的需要上传的报文转换格式。(需要说明的是,我们的J2EE后台写的比较奇怪,所以上传和下载下来的格式是不一样的,这一点我一开始没发现,走了不少弯路。)
到这里,就可以实现上传和下载了!真是不容易那(大変だ)!
下面是一些需要到的库文件:
System.Runtime.Serialization; 解析JSON报文
System.Runtime.Serialization.Json;
System.ServiceModel; 绑定实体类需要
System.ServiceModel.Web;
System.IO;
System.Xml;
System.Net; 解析JSON报文
System.Net.Http;
System.Net.Http.Formatting;
System.Web.Script.Serialization;
System.Json;
最后一点忘记补充了,上传上去的时候需要通过URL编码,这个视具体项目要不要转码。
最新问题,当服务器为linux系统时,需要加上webClient.Headers.Add(HttpRequestHeader.AcceptLanguage, "zh-cn,zh;q=0.8,en-us;q=0.5,en;q=0.3");,不然无法上传。