草庐IT

json传参到java接口部分参数接收不到

归去来 兮 2024-02-12 原文

json传参到java接口部分参数接收不到-问题深究

一、问题描述:

使用postman传参时,接口参数中有部分参数被传递进了接口内部,还有部分参数没有接收到,如下图:第一张图是postman接口参数,第二张图是接收到的参数
图一:

图二:

二、问题解决:

这个问题其实解决很简单,就是json数据转化为实体的过程,找寻get、set方法失败,导致数据无法正常从json映射到实体,从而出现的问题。解决起来两个方法,第一种就是改变量名,这种不建议使用,改动量较大,需要将DTO、VO等都需要更改,有的甚至还需要动sql,第二种就是加个注解即可,告诉实体把某个json值就给当前的属性就行,比如上面例子我们可以给pOrgCode这个属性加如下的注解:

    @JsonProperty(value = "pOrgCode")
    String pOrgCode;

这里的注解是com.fasterxml.jackson.annotation.JsonProperty这个包下面的注解。

三、问题原因

变量名的前两个字母出现了大写
下面一起探讨下,这个过程是如何失败的,失败的具体点又是什么
其实根本原因还是在get、set方法和属性的命名上,下面pojo的代码

@RequiredArgsConstructor
@Data
class DataDTO{

    String pOrgCode;
    String name;
    String uName;
    String isTrue;
    String Lname;
    String NAme;

}

可以看到代码没有什么特别的,上面的@Data是lombok的注解,可以省去我们写get、set、toString等方法。
下面再看下真正编译后的文件是什么样吧,如下:

class DataDTO {
    String pOrgCode;
    String name;
    String uName;
    String isTrue;
    String Lname;
    String HOme;

    public DataDTO() {
    }

    public String getPOrgCode() {
        return this.pOrgCode;
    }

    public String getName() {
        return this.name;
    }

    public String getUName() {
        return this.uName;
    }

    public String getIsTrue() {
        return this.isTrue;
    }

    public String getLname() {
        return this.Lname;
    }

    public String getHOme() {
        return this.HOme;
    }

    public void setPOrgCode(String pOrgCode) {
        this.pOrgCode = pOrgCode;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setUName(String uName) {
        this.uName = uName;
    }

    public void setIsTrue(String isTrue) {
        this.isTrue = isTrue;
    }

    public void setLname(String Lname) {
        this.Lname = Lname;
    }

    public void setHOme(String HOme) {
        this.HOme = HOme;
    }

    //此处省略无关方法若干
}

从上面的编辑结果我们可以看到两种情况:

1.首字母小写就是将首字母进行大写其然后前面拼接get、set
2.首字母大写则保持不变前面拼接get、set

下面看下这种场景下后台接收到的参数展示:

通过上面图片可以看出,只要前两个字母出现了大写字母,那么lombok生产的get、set方法是找寻不到真正的属性的,所以json转化实体就出了过程。
下面笔者又尝试了使用idea自动生成get、set方法,经实现,产生的get、set等如下:

class DataDTO {
    String pOrgCode;
    String name;
    String uName;
    String isTrue;
    String Lname;
    String HOme;

    public String getpOrgCode() {
        return this.pOrgCode;
    }

    public void setpOrgCode(String pOrgCode) {
        this.pOrgCode = pOrgCode;
    }

    public String getName() {
        return this.name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getuName() {
        return this.uName;
    }

    public void setuName(String uName) {
        this.uName = uName;
    }

    public String getIsTrue() {
        return this.isTrue;
    }

    public void setIsTrue(String isTrue) {
        this.isTrue = isTrue;
    }

    public String getLname() {
        return this.Lname;
    }

    public void setLname(String lname) {
        this.Lname = lname;
    }

    public String getHOme() {
        return this.HOme;
    }

    public void setHOme(String HOme) {
        this.HOme = HOme;
    }

    public DataDTO() {
    }
    //此处省略无关方法
}

从生产的get、set方法上看,这两种生产略有差别,但是值得注意的是使用idea生成的get、set一样也接收不到数据,如下图

比对发现,其实并不是lombok多不好用,这种场景下我们使用idea其实也是一样的结果。

四、使用postman请求接口会出这种问题,那接口和接口之前的调用会不会有这种问题?

上面这个问题是需要分场景的

1.http调用场景

答案是:部分是部分否,也就是说http调用就会有上述我们碰到的问题,但是只要一部分字段有这总问题,postman其实就是模仿http客户端发出请求调用接口的。

@RestController
@RequestMapping("/org")
public class TestController {
    @PostMapping("/test2")
    public void testJson(@RequestBody DataDTO dataDTO){
        System.out.println(dataDTO.toString());
        System.out.println(dataDTO.toString());

    }

    @PostMapping("/test")
    public void testJson2(@RequestBody DataDTO dataDTO){
        HttpClientUtil httpClientUtil = new HttpClientUtil();
        dataDTO = new DataDTO();
        dataDTO.setpOrgCode("666");
        dataDTO.setHOme("555");
        dataDTO.setIsTrue("444");
        dataDTO.setLname("333");
        dataDTO.setName("222");
        dataDTO.setuName("111");

        Map<String,String> headMap = new HashMap<>();
        headMap.put("Content-type","application/json;charset=UTF-8");
        String s = httpClientUtil.doPost("http://localhost:8888/org/test2", headMap, JSONObject.toJSON(dataDTO).toString());
        System.out.println("调用结束");

    }
}

如上代码,从新包了一个接口用于调用原接口,接口调用使用http方式来调用,这样原接口的输出结果如下所示:

然后我们发现除了前两个字母都是大写的场景下会出问题,其他都是ok的,所以这种问题其实也算是工具的问题。

2.RPC调用场景

若是RPC调用传递还是JSON结论则和HttpClient调用没啥区别了,若是传递实体则不用有这种问题了。

五、总结

出现这个问题的原因是使用postman调用接口传递json解析失败造成的,后面使用httpclient验证,只有在前两个字母均是大写的场景下,使用httpclient才会出问题,正常情况下postman会出问题的场景,httpclient并没有,所以平时代码还是放心写就是,当然为了以防万一我们一定不要写这种代码(属性前两个字母出现大写),若是非要这么写也要加个注释:JsonPropertity(value=“filedName”)。

有关json传参到java接口部分参数接收不到的更多相关文章

  1. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  2. ruby - RSpec - 使用测试替身作为 block 参数 - 2

    我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere

  3. ruby-on-rails - Rails HTML 请求渲染 JSON - 2

    在我的Controller中,我通过以下方式在我的index方法中支持HTML和JSON:respond_todo|format|format.htmlformat.json{renderjson:@user}end在浏览器中拉起它时,它会自然地以HTML呈现。但是,当我对/user资源进行内容类型为application/json的curl调用时(因为它是索引方法),我仍然将HTML作为响应。如何获取JSON作为响应?我还需要说明什么? 最佳答案 您应该将.json附加到请求的url,提供的格式在routes.rb的路径中定义。这

  4. ruby - 如何在 Ruby 中拆分参数字符串 Bash 样式? - 2

    我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"

  5. ruby - 检查方法参数的类型 - 2

    我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)

  6. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  7. ruby-on-rails - 在默认方法参数中使用 .reverse_merge 或 .merge - 2

    两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option

  8. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  9. ruby-on-rails - RSpec:避免使用允许接收的任何实例 - 2

    我正在处理旧代码的一部分。beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)endRubocop错误如下:Avoidstubbingusing'allow_any_instance_of'我读到了RuboCop::RSpec:AnyInstance我试着像下面那样改变它。由此beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)end对此:let(:sport_

  10. ruby - rails 3 redirect_to 将参数传递给命名路由 - 2

    我没有找到太多关于如何执行此操作的信息,尽管有很多关于如何使用像这样的redirect_to将参数传递给重定向的建议:action=>'something',:controller=>'something'在我的应用程序中,我在路由文件中有以下内容match'profile'=>'User#show'我的表演Action是这样的defshow@user=User.find(params[:user])@title=@user.first_nameend重定向发生在同一个用户Controller中,就像这样defregister@title="Registration"@user=Use

随机推荐