SpringMVC无法获取请求中的参数的问题的调查与解决(1)

*无论@RequestBody还是@RequestParam注解一样,都会使用全局的Encoding进行解码,会导致特殊编码的参数值丢失。

只要抛弃掉注解,就完全可以在Controller层得到请求的Raw数据!

-----

使用框架可以节约开发时间,但有时由于隐藏了一些实现细节,导致对底层的原理知之不详,碰到问题时不知道该从哪一个层面入手解决。因此我特意记录了下面这个典型问题的调查和解决过程供参考。

 

事情是这样的,我们原来有一个移动端调用的发表评论的API,是几年前在NET平台上开发的,移植到JAVA后,发现安卓版APP无法正常发表汉字评论。

基于SpringMVC创建的JAVA版API接口大致如下,经调查发现,关键的content参数,在Controller层检查结果为空。

 

	@RequestMapping(value = "/Test.api")
	public Object test(
			HttpServletRequest request,
			HttpServletResponse response,
    		        @RequestParam(value = "content", required = false, defaultValue="") String content) {

                // 在这里,content的值为空   

        }

用Charles抓包检查Post的Form数据,确实有字段content,且有汉字值。但检查其Raw数据居然为这样的形式:content=%u611f%u53d7%u4e00%u4e0b%u8d85%u4eba%u7684%u808c%u8089%uff0c

 

我们知道,目前java常用的URLEncoder类,一般将汉字转换成"%xy"的形式,xy是两位16进制的数值,不会出现%u后面跟4个字符这种情况。

%u开头代表这是一种Unicode编码格式,后面的四个字符是二字节unicode的四位16进制码。在Charles软件上,支持这种解码,所以可以正常看到抓包数据中的汉字。

 

但是我们从SpringMVC框架层面统一指定了encoding为UTF-8,根据@RequestParam注解,使用UTF-8进行content参数的解码时,必然异常,由此导致了Post过来的content字段丢失。

 

和安卓团队确认,发现过去他们确实采用了自己独有的Encode方法对Post数据进行编码:

    public static String UrlEncodeUnicode(final String s)
    {
        if (s == null)
        {
            return null;
        }
        final int length = s.length();
        final StringBuilder builder = new StringBuilder(length); // buffer
        for (int i = 0; i < length; i++)
        {
            final char ch = s.charAt(i);
            if ((ch & 0xff80) == 0)
            {
                if (Utils.IsSafe(ch))
                {
                    builder.append(ch);
                }
                else if (ch == ' ')
                {
                    builder.append('+');
                }
                else
                {
                    builder.append("%");
                    builder.append(Utils.IntToHex((ch >> 4) & 15));
                    builder.append(Utils.IntToHex(ch & 15));
                }
            }
            else
            {
                builder.append("%u");
                builder.append(Utils.IntToHex((ch >> 12) & 15));
                builder.append(Utils.IntToHex((ch >> 8) & 15));
                builder.append(Utils.IntToHex((ch >> 4) & 15));
                builder.append(Utils.IntToHex(ch & 15));
            }
        }
        return builder.toString();
    }

采用这种方式的原因已经不可考证,并且安卓团队已经决定将在未来版本中放弃该编码方式,采用JAVA常用的Encoder类进行UTF-8的编码。问题定位后,决定新版API中必须要兼容新旧两种编码方式。

但是目前SpringMVC的@RequestParam注解负责了请求数据的解码,我们从哪一层切入,截获请求数据,判断其编码方式,并动态选用不同的解码方式来处理呢?

经过DEBUG,觉得下面的方式是可行的。

 

解决问题的步骤:

修改API的接口形式,放弃@RequestParam注解,放弃@RequestBody注解,在Controller层直接从request的stream中得到参数的raw数据并自己解析

    @RequestMapping(value = "/Test.api")
    public Object test(
            HttpServletRequest request,
            HttpServletResponse response) {

            String line = "";
            StringBuilder body = new StringBuilder();
            int counter = 0;
            
            InputStream stream;
            stream = request.getInputStream();
            
            //读取POST提交的数据内容
            BufferedReader reader = new BufferedReader(new InputStreamReader(stream));
            while ((line = reader.readLine()) != null) {
                if(counter > 0){
                    body.append("\r\n");
                }
                body.append(line);
                counter++;
            }
            
            //POST请求的raw数据可以获得
            System.out.println(body.toString());

                // 使用自定义的解析工具类对body的内容进行解码
                RequestParamMap map = new RequestParamMap(body.toString());
                ...
       }

 

只要在进入controller层之前,确保没有别的Filter之类调用了request.getInputStream()就没有问题,因为request.getInputStream()只能被碰一次,之后就再无法获取原始的请求数据了。

 

然而,接着就发现,确实有一个自定义的拦截器,需要获取POST的InputStream数据...

http://www.cnblogs.com/csliwei/p/5557353.html

-----

 

转载于:https://www.cnblogs.com/csliwei/p/5526281.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: SpringMVC获取请求参数的方式有多种,最常用的是通过注解@RequestParam获取请求参数。例如,@RequestParam("username") String username表示获取名为username的请求参数,并将其赋值给String类型的变量username。另外,还可以通过HttpServletRequest对象获取请求参数,例如HttpServletRequest.getParameter("username")。此外,还可以使用@PathVariable注解获取URL路径参数,例如@RequestMapping("/user/{id}"),其{id}表示获取路径的id参数。 ### 回答2: SpringMVC是一种基于Java的web框架,用于开发web应用程序。通过SpringMVC框架,可以很方便地获取请求参数。 在SpringMVC获取请求参数有多种方式。其,最常用的方式是通过方法的参数获取请求参数。在控制器方法定义参数参数名与请求参数名一致,SpringMVC会自动将请求参数的值绑定到对应的参数上。例如,如果请求有一个参数名为"username"的参数,可以在控制器方法定义一个同名的String类型的参数来接收该参数的值。 除了通过方法参数获取请求参数,还可以使用@RequestParam注解。该注解可以用于方法参数上,指定请求参数的名称,以及是否为必需的参数。例如,@RequestParam("id") int userId,表示要获取名为"id"的请求参数,并将其转换为int类型的userId参数。可以通过设置required属性来指定该参数是否必需,默认为true,如果请求没有该参数,会抛出异常。 另外,还可以使用@PathVariable注解来获取路径参数。路径参数是在URL定义的参数,可以通过在控制器方法的路径使用占位符来表示。例如,@RequestMapping("/user/{id}"),可以通过@PathVariable("id")来获取路径参数值。 除了以上方式,还可以通过HttpServletRequest对象来获取请求参数。在控制器方法,可以将HttpServletRequest对象作为方法参数传入,然后通过该对象的getParameter方法来获取请求参数的值。 总之,SpringMVC提供了多种简便的方式来获取请求参数。无论是通过方法参数、@RequestParam注解、@PathVariable注解,还是使用HttpServletRequest对象,都可以方便地获取请求参数的值,便于进行后续的业务逻辑处理。 ### 回答3: SpringMVC是基于Java的一种MVC(Model-View-Controller)框架,用于构建Web应用程序。在SpringMVC获取请求参数是非常常见的任务并且非常容易。 首先,要使用SpringMVC获取请求参数,你需要定义一个处理请求的控制器类,并在类的方法上添加@RequestMapping注解来指定请求的URL映射。 在控制器方法,你可以使用@RequestParam注解来获取请求参数。该注解有几个属性可以使用,最常用的是value属性,用于指定请求参数的名称。例如,你可以使用@RequestParam("name")来获取名为name的请求参数。如果请求参数的名称与方法参数的名称相同,你也可以省略value属性,只使用@RequestParam注解。 @RequestParam注解也支持一些其他属性,例如required属性用于指示请求参数是否是必需的,默认为true,即必须提供该参数。你还可以使用defaultValue属性来指定请求参数的默认值。 除了@RequestParam注解,你还可以使用@PathVariable注解来获取URL路径参数。该注解将URL的占位符与方法参数进行映射。例如,如果URL有一部分为"/users/{id}",你可以在方法参数使用@PathVariable("id")来获取该路径的id参数。 另外,SpringMVC还支持通过@RequestParamMap注解来获取所有的请求参数,并以Map的形式进行处理。这对于处理不确定数量的请求参数非常有用。 总而言之,SpringMVC提供了多种方法来获取请求参数。你可以使用@RequestParam注解来获取单个请求参数,使用@PathVariable注解来获取URL路径参数,或者使用@RequestParamMap注解来获取所有的请求参数。这些功能使得处理请求参数变得非常方便和灵活。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值