在使用POST方式向服务提交数据时,出现HTTP400异常,以下代码描述:

    服务接口定义:

    [OperationContract]
    [WebInvoke(ResponseFormat = WebMessageFormat.Json, RequestFormat = WebMessageFormat.Json, Method = "POST", BodyStyle =             WebMessageBodyStyle.Bare)]
    Stream HelloData(MyData data);

    [OperationContract]
[WebInvoke(Method = "POST", BodyStyle = WebMessageBodyStyle.Bare)]
Stream HelloDataStr(String data);

    实现只是简单的把传入的数据打印出来,略

    客户端调用:

    方法一、 

    private static void HelloDataClient()
    {

      WebClient client = new WebClient();
      client.Headers["Content-Type"] = "application/json";
      string data = "{\"ID\":1,\"Name\":\"ss\"}";
      string str = client.UploadString("http://localhost:1122/HelloData", "POST", data);
      Console.WriteLine(str);
    }

    方法二

    private static void HelloDataStrClient()
    {
      WebClient client = new WebClient();
      string str = client.UploadString("http://localhost:1122/HelloDataStr", "POST", "ss");
      Console.WriteLine(str);
    }

    以上两个方法对用调用两个不同的服务,问题在于使用Mydata数据契约的方法一OK,方法二出现HTTP400错误。为什么自定义类型的可以,而基本数据类型的不可以?起初认为自定义数据类型自己已经做了序列化,而string不是自己做的可能跟序列化有关。那没有用UriTemplate是不是参数映射出了问题?因为方法一已经成功,说明WCF会将HTTP请求参数默认映射到仅有的一个服务接口的参数上。排除此猜测。

    方法后来做了以下迭代测试:

    将方法二改为:

    private static void HelloDataStrClient()
    {
      WebClient client = new WebClient();
      string str = client.UploadString("http://localhost:1122/HelloDataStr", "POST", String.Empty);
      Console.WriteLine(str);
    }

    发现调用成功,说明数据有问题。数据有问题就是数据的格式有问题,但是服务契约二没有声明Request的格式为JSON。那么他默认为什么呢?把方法二再改如下:

    private static void HelloDataStrClient()
    {
      WebClient client = new WebClient();
      client.Headers["Content-Type"] = "application/json";
      string str = client.UploadString("http://localhost:1122/HelloDataStr", "POST", "\"ss\"");
      Console.WriteLine(str);

    }

    调用成功。

      注意:JSON格式默认为“”引起,所以 "\"ss\""不能写为“ss”,否则还是HTTP400错误。因为未识别数据,将数据拼接到其他字节了吧。

    问题描述二、

      没有使用UriTemplate做参数的映射,那多个参数是怎么映射的?将方法二改为:

    服务契约二:

    [OperationContract]
    [WebInvoke(Method = "POST", BodyStyle = WebMessageBodyStyle.Wrapped)]
    Stream HelloDataStr(String data1,string data2);

    客户端:

    private static void HelloDataStrClient()
    {
      WebClient client = new WebClient();
      client.Headers["Content-Type"] = "application/json";
      string str = client.UploadString("http://localhost:1122/HelloDataStr", "POST", "{\"data1\":\"hh\",\"data2\":\"ss\"}");
      Console.WriteLine(str);

    }

    调用成功,并且成功打印了两个参数。注意服务契约的接口定义属性WebMessageBodyStyle.Wrapped,此属性将两个参数进行了包装,否则无法映射。

    附:JAVA调用代码:

    

public static void main(String[] args) {
// TODO Auto-generated method stub
  HelloClient();

  HelloDataClient();

  HelloDataStrClient();

}
private static void HelloClient(){
  URLClient client = new URLClient();
  String resultStr = client.getDocumentAt("http://localhost:1122/Hello");
  MyData data = assembleModel(resultStr);
  System.out.print("Hello Response content: " +resultStr);
  System.out.println("MyData Name: " +data.getName());
}

private static void HelloDataClient(){
  try {

String postUrl = "http://localhost:1122/HelloData";
String postData = "{\"data1\":1,\"data2\":\"ss\"}";
HttpClient hClient = new DefaultHttpClient();
HttpPost post = new HttpPost(postUrl);
StringEntity s = new StringEntity(postData);
s.setContentEncoding("UTF-8");
s.setContentType("application/json");
post.setEntity(s);
HttpResponse response = hClient.execute(post);
HttpEntity responseEntity = response.getEntity();

String resultStr = inputStream2String(responseEntity.getContent());
MyData data = assembleModel(resultStr);
System.out.println("HelloDataStr Response content: " 
+ resultStr); 
System.out.println("MyData Name: " +data.getName());

} catch (Exception e) {

  // TODO: handle exception

  }

}

private static void HelloDataStrClient(){
  try {

String postUrl = "http://localhost:1122/HelloDataStr";
String postData = "{\"data1\":1,\"data2\":\"ss\"}";
HttpClient hClient = new DefaultHttpClient();
HttpPost post = new HttpPost(postUrl);
StringEntity s = new StringEntity(postData);
s.setContentEncoding("UTF-8");
s.setContentType("application/json");
post.setEntity(s);
HttpResponse response = hClient.execute(post);
HttpEntity responseEntity = response.getEntity();

String resultStr = inputStream2String(responseEntity.getContent());
System.out.println("HelloDataStr Response content: " 
+ resultStr);

  } catch (Exception e) {
// TODO: handle exception
    }
}

public static String inputStream2String(InputStream is) throws IOException { 
  ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
  int i = -1; 
  while ((i = is.read()) != -1) { 
    baos.write(i); 
  } 
  return baos.toString(); 

private static MyData assembleModel(String jsonStr) {
  MyData model = null;
  try {
      JsonConvertor jsonConvertor = new JsonConvertor();
      MyData[] stus = (MyData[]) jsonConvertor
    .convertToObj(jsonStr,MyData.class);
      model = stus[0];
    } catch (Exception e) {
      e.printStackTrace();
  }
  return model;
}

WCF Rest:不使用UriTemplate使用post方式传参解决HTTP400问题以及参数映射问题的更多相关文章

  1. 二十一、springboot之定制URL匹配规则(项目中遇到的问题:get方式传参,带有小数点,被忽略)

    一.问题描述: get方式传参,在传送价格,积分时(带有小数点),debug后台微服务接受到的参数,却不带小数点,如:price是0.55,后台接受后却是0 二.解决 在WebConfiguratio ...

  2. 使用Fiddler工具发送post请求(带有json数据)以及get请求(Header方式传参)

    Fiddler工具是一个http协议调试代理工具,它可以帮助程序员测试或调试程序,辅助web开发. Fiddler工具可以发送向服务端发送特定的HTTP请求以及接受服务器回应的请求和数据,是web调试 ...

  3. 同一路由带参刷新,以及params和query两种方式传参的异同

    同一路由应该不叫跳转了吧,就先叫刷新好了. 需求及问题 今天做web课设有这样一个需求: 在导航栏中一项叫做教师队伍一级菜单下,有三个二级菜单,分别为教授.副教授.讲师.这三个二级菜单分别对应一个页面 ...

  4. vue:子组件通过调用父组件的方法的方式传参

    在本案例中,由于子组件通过调用父组件的方法的方式传参,从而实现修改父组件data中的对象,所以需要啊使用$forceUpdate()进行强制刷新 父组件: provide() { return { s ...

  5. RESTful Get方式传参json格式后端400 解决方案

    前端采用vue+axios 后端采用spring boot restful 问题: 前端get 请求需要传递array 字段值 后端由于tomcat 版本问题,不支持url接受特殊字符包括 [] {} ...

  6. python flask学习第2天 URL中两种方式传参

    新创建项目   自己写个url映射到自定义的视图函数 在url中传递参数 app.py from flask import Flask app = Flask(__name__) @app.route ...

  7. 在 WinForm 中打开页面采用POST方式传参http。可以多个参数传递,返回json字符串

    //调用方法 Dictionary<string, string> postData = new Dictionary<string, string>(); postData. ...

  8. window.open打开新窗体并用post方式传参

    function openPostWindow(url,data,name){ //url要跳转到的页面,data要传递的数据,name显示方式(可能任意命名) var tempForm = docu ...

  9. yii2.0的gradview点击按钮通过get方式传参

    1.直接看views层里的代码就可以了 , <!--?= GridView::widget([ 'dataProvider' =--> $dataProvider, 'filterMode ...

随机推荐

  1. 细讲encodeURI和encodeURIComponent以及escape的区别与应用

    首先,我们都知道这三个东西都是用来编码的 先来说encodeURI()和encodeURIComponent() 这两个是在转换url时候用来编码解码用的. 有编码就会有解码, 解码就是decodeU ...

  2. access数据库的连接字符串以及数据库操作类

    <!--access数据库连接方式--> <add name="QYTangConnectionString" connectionString="Pr ...

  3. 使用 Require.js 引用第三方框架时遇到的一些情况

    使用 Require.js 引用第三方框架时遇到的一些情况 在使用Require.js解析依赖的时候,会出现以下几种情况: 程序中的依赖关系 当前程序 依赖于 B包, B包 依赖于 A包 A包与B包两 ...

  4. 认识WebSocket理论篇

    本文转自http://www.ibm.com/developerworks/cn/java/j-lo-WebSocket/ HTML5作为下一代WEB标准,拥有许多引人注目的新特性,如Canvas.本 ...

  5. hdu1754 基础线段树

    I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

  6. C语言获取键盘按键

    在写控制台游戏的时候,发现不管用cin,scanf还是getchar,都不能实时的输入按键,必须要按回车才能读进去,而按回车的话会导致输入异常,所以要使用获取键盘按键的函数. 加入头文件#includ ...

  7. 课堂里学不到的C与C++那些事(一)

    首先,声明一下这是一个系列的文章.至于整个系列有多少篇,笔者也不知道,不知道有多少篇,也不知道多久会更新一篇.反正只有一个原则,写出来的文 章能见得人才会公布出来.另外,我不是叫你逃课,而是觉得听课只 ...

  8. C# 创建execl文件 并且填充数据

    第一步:引用文件 using NPOI.HSSF.UserModel;using System.Data;using CTUClassLibrary;using System.IO;using NPO ...

  9. [C++程序设计]用数组名作函数参数

    1. 用数组元素作函数实参 #include <iostream> using namespace std; int max_value(int x, int max) { return ...

  10. python成长之路第三篇(4)_作用域,递归,模块,内置模块(os,ConfigParser,hashlib),with文件操作

    打个广告欢迎加入linux,python资源分享群群号:478616847 目录: 1.作用域 2.递归 3.模块介绍 4.内置模块-OS 5.内置模块-ConfigParser 6.内置模块-has ...