zoukankan      html  css  js  c++  java
  • 关于通过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");,不然无法上传。

  • 相关阅读:
    解决centos7的root账户下无法通过code命令启动vscode
    centos7安装epel
    centos7用过yum安装vscode
    yum install gcc报错Error: Package: glibc-2.17-260.el7_6.6.i686 (updates) Requires: glibc-common = 2.17
    centos7通过yum从vim7升级到vim8
    解决VM虚拟机安装centos7无法联网
    centos7设置开机默认使用root账户登陆
    centos7使用sudo命令提示sudo command not found
    不同编译器下C++基本数据类型的字节长度
    C++函数模板
  • 原文地址:https://www.cnblogs.com/rarator/p/3087804.html
Copyright © 2011-2022 走看看