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其实也是一样的结果。
上面这个问题是需要分场景的
答案是:部分是部分否,也就是说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的,所以这种问题其实也算是工具的问题。
若是RPC调用传递还是JSON结论则和HttpClient调用没啥区别了,若是传递实体则不用有这种问题了。
出现这个问题的原因是使用postman调用接口传递json解析失败造成的,后面使用httpclient验证,只有在前两个字母均是大写的场景下,使用httpclient才会出问题,正常情况下postman会出问题的场景,httpclient并没有,所以平时代码还是放心写就是,当然为了以防万一我们一定不要写这种代码(属性前两个字母出现大写),若是非要这么写也要加个注释:JsonPropertity(value=“filedName”)。
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我有一些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
在我的Controller中,我通过以下方式在我的index方法中支持HTML和JSON:respond_todo|format|format.htmlformat.json{renderjson:@user}end在浏览器中拉起它时,它会自然地以HTML呈现。但是,当我对/user资源进行内容类型为application/json的curl调用时(因为它是索引方法),我仍然将HTML作为响应。如何获取JSON作为响应?我还需要说明什么? 最佳答案 您应该将.json附加到请求的url,提供的格式在routes.rb的路径中定义。这
我正在为一个项目制作一个简单的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"
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option
我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano
我正在处理旧代码的一部分。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_
我没有找到太多关于如何执行此操作的信息,尽管有很多关于如何使用像这样的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