url 传递中文参数乱码问题的终极解决方法。
估计很多人在做web开发的时候,都会碰到过url传递中文参数,有时候会出现乱码的问题,但有些项目或者环境,又不会有问题。当遇到乱码的时候,上网找了很多解决方案,比如:
- 页面设置它的编码方式,改成utf-8 或者gb2312。
-
encodeURI(url),也有人说要2个encodeURI,如:window.location.href = encodeURI('b.html?cId='+id+"&cName="+encodeURIComponent(name));
然后后台String str = java.net.URLDecoder.decode(str, "UTF-8");这样进行转码等。
解决方法就不一一列举了,大家可自行百度搜索,我也搞不清楚究竟怎么能保证百分百解决问题。
项目中碰到的情况是,开发环境经过上述的2种方法,折腾过,确实能定位某个编码转换是正常的。
但部署到测试环境调试输出后发现,无论是gbk,utf8,iso等等编码测试,都无法正常还原中文字符串。
于是想到了另外一种方法,就是把中文转换为数字或者是英文字母以及标点符号等组成的字符串传递到后台,
因此有想法是把字符串加密为base64的方式传过去后,再后台再进行解密。上网找了下,没找到现成的有效方法,于是放弃了。
再后来,想着不需要加密了,直接把字符串转换成16进制传递到后台,再把它转回来吧,于是,这个终极的解决方案就出来了。
首先是jsp页面中增加脚本,把字符串转换成16进制字符串,字符之间用,号隔开,不隔开,无法区分出哪个是半角哪个是全角(如果不用,号隔开的,麻烦请提供个好的解决方法,谢谢)
1 2 3 4 5 6 7 8 9 10 | function stringToHex(str){ var val=""; for(var i = 0; i < str.length; i++){ if(val == "") val = str.charCodeAt(i).toString(16); else val += "," + str.charCodeAt(i).toString(16); } return val; } |
页面上把字符串经过该函数转换后,传递到后台,如:
var code_value=stringToHex(PrjName);
var url= "/assets/fmProjectInfo.do?method=synCheckProjectName&prjName="+code_value+"&typeId=<bean:write name="fmProjectInfoForm" property="bo.ftId"/>"
后台的转换代码为:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 | public static String decode(String unicodeStr) { if (unicodeStr == null ) { return null ; } StringBuffer retBuf = new StringBuffer(); int maxLoop = unicodeStr.length(); for ( int i = 0 ; i < maxLoop; i++) { if (unicodeStr.charAt(i) == '\\' ) { if ((i < maxLoop - 5 ) && ((unicodeStr.charAt(i + 1 ) == 'u' ) || (unicodeStr .charAt(i + 1 ) == 'U' ))) try { retBuf.append(( char ) Integer.parseInt( unicodeStr.substring(i + 2 , i + 6 ), 16 )); i += 5 ; } catch (NumberFormatException localNumberFormatException) { retBuf.append(unicodeStr.charAt(i)); } else retBuf.append(unicodeStr.charAt(i)); } else { retBuf.append(unicodeStr.charAt(i)); } } return retBuf.toString(); } //把jsp页面传递进来的,用,号隔开的16进制字符串转换成 类似:select \u7528\u6237\u540d from \u7528\u6237 的字符串,在调用decode方法把中文转换出来。 public static String jspStrInit(String sourceStr) { String[] sourceStrArray = sourceStr.split( "," ); StringBuffer sb= new StringBuffer(); for ( int i = 0 ; i < sourceStrArray.length; i++) { if (sourceStrArray[i].length()<= 2 ) sb.append(hexStr2Str(sourceStrArray[i].toUpperCase())); else sb.append( "\\u" +sourceStrArray[i]); } return sb.toString(); } public static String jspDecode(String unicodeStr) { String mStr=jspStrInit(unicodeStr); return decode(mStr); } <br> /** <br> * 十六进制转换字符串 <br> * @param String str Byte字符串(Byte之间无分隔符 如:[616C6B]) <br> * @return String 对应的字符串 <br> */ <br> public static String hexStr2Str(String hexStr) <br> { <br> String str = "0123456789ABCDEF" ; <br> char [] hexs = hexStr.toCharArray(); <br> byte [] bytes = new byte [hexStr.length() / 2 ]; <br> int n; <br> <br> for ( int i = 0 ; i < bytes.length; i++) <br> { <br> n = str.indexOf(hexs[ 2 * i]) * 16 ; <br> n += str.indexOf(hexs[ 2 * i + 1 ]); <br> bytes[i] = ( byte ) (n & 0xff ); <br> } <br> return new String(bytes); <br> } <br> public static void main(String[] args) throws Exception { String bb= "5e02,91cd,70b9,5de5,4f5c,6d4b,8bd5" ; String cc=jspDecode(bb); System.out.println(cc); } |
jspDecode这个方法就是把jsp页面传递过来的字符串进行还原的。该终极解决方法就此结束。
最后,在另外一个项目中有客户的研发保障,说修改意见那里有时候保存不成功,有时候又没问题,经过分析,修改意见那里是使用ajax封装了json的数据传递到后台进行更新的。
由于意见里面包含了回车,标点符号等特殊符号,违反了json的原则导致的,我给他提供的解决方案,也是采用了该方案,让他把该意见转换为16进制字符串之后,传递到后台再转换回来去解决。
目前还没得到问题的解决答复,但按理,应该也可以用在该场景里面。
原创作品出自努力偷懒,转载请说明文章出处:http://blog.csdn.net/kfarvid或 http://www.cnblogs.com/kfarvid/
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· 单线程的Redis速度为什么快?
· 展开说说关于C#中ORM框架的用法!
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库