[聚合文章] 关于通过webclient和JSON格式报文与服务器之间通讯的解决方法和遇到的难题

c# 2017-03-22 4 阅读

话说做这个很久了,不过我比较磨洋工。所以托了很久,今天才真正跑通了整个流程。

一开始老师要我做的时候,我有点没明白意思,后来看了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-...(最后有提到)

但是坑爹的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");,不然无法上传。

注:本文内容来自互联网,旨在为开发者提供分享、交流的平台。如有涉及文章版权等事宜,请你联系站长进行处理。