前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >URL解码之URLEncoder

URL解码之URLEncoder

作者头像
全栈程序员站长
发布于 2022-09-07 02:21:10
发布于 2022-09-07 02:21:10
2.2K00
代码可运行
举报
运行总次数:0
代码可运行

大家好,又见面了,我是你们的朋友全栈君。

关于URL解码看到了一篇易懂文章

什么是application/x-www-form-urlencoded字符串?

答:它是一种编码类型。当URL地址里包含非西欧字符的字符串时,系统会将这些字符转换成application/x-www-form-urlencoded字符串。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  表单里提交时也是如此,当包含非西欧字符的字符串时,系统也会将这些字符转换成application/x-www-form-urlencoded字符串。

  然而,在向服务器发送大量的文本、包含非ASCII字符的文本或二进制数据时这种编码方式效率很低。这个时候我们就要使用另一种

  编码类型“multipart/form-data”,比如在我们在做上传的时候,表单的enctype属性一般会设置成“multipart/form-data”。

网页中的表单使用POST方法提交时,数据内容的类型是 application/x-www-form-urlencoded,这种类型会:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1.字符"a"-"z""A"-"Z""0"-"9"".""-""*",和"_" 都不会被编码;
2.将空格转换为加号 (+) ;
3.将非文本内容转换成"%xy"的形式,xy是两位16进制的数值;
4.在每个 name=value 对之间放置 & 符号。

URLEncoder类包含将字符串转换为application/x-www-form-urlencoded MIME 格式的静态方法。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   web 设计者面临的众多难题之一便是怎样处理不同操作系统间的差异性。这些差异性能引起URL方面的问题:例如,一些操作系统允许文件名中含有空格符,有些又不允许。大多数操作系统不会认为文件名中含有符号“#”会有什么特殊含义;但是在一个URL中,符号“#”表示该文件名已经结束,后面会紧跟一个 fragment(部分)标识符。其他的特殊字符,非字母数字字符集,它们在URL或另一个操作系统上都有其特殊的含义,表述着相似的问题。为了解决这些问题,我们在URL中使用的字符就必须是一个ASCII字符集的固定字集中的元素,具体如下:


1.大写字母A-Z
2.小写字母a-z
3.数字 0-9
4.标点符 - _ . ! ~ * ' (,)


  诸如字符: / & ? @ # ; $ + =%也可以被使用,但是它们各有其特殊的用途,如果一个文件名包括了这些字符( / & ? @ # ; $ + = %),这些字符和所有其他字符就应该被编码。

  编码过程非常简单,任何字符只要不是ASCII码数字,字母,或者前面提到的标点符,它们都将被转换成字节形式,每个字节都写成这种形式:一个“%”后面跟着两位16进制的数值。空格是一个特殊情况,因为它们太平常了。它除了被编码成“%20”以外,还能编码为一个“+”。加号(+)本身被编码为%2B。当/ # = &?作为名字的一部分来使用时,而不是作为URL部分之间的分隔符来使用时,它们都应该被编码。

 WARNING这种策略在存在大量字符集的异构环境中效果不甚理想。例如:在U.S. Windows 系统中, é 被编码为 %E9.U.S. Mac中被编码为%8E。这种不确定性的存在是现存的URI的一个明显的不足。所以在将来URI的规范当中应该通过国际资源标识符(IRIs)进行改善。

类 URL并不自动执行编码或解码工作。你能生成一个URL对象,它可以包括非法的ASCII和非ASCII字符和/或%xx。当用方法getPath() 和toExternalForm( ) 作为输出方法时,这种字符和转移符不会自动编码或解码。你应对被用来生成一个URL对象的字符串对象负责,确保所有字符都会被恰当地编码。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
幸运的是,java提供了一个类URLEncoder把string编码成这种形式。Java1.2增加了一个类URLDecoder它能以这种形式解码string。这两个类都不用初始化:

public class URLDecoder extends Object public class URLEncoder extends Object

一、URLEncoder

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
在java1.3和早期版本中,类java.net.URLEncoder包括一个简单的静态方法encode( ), 它对string以如下规则进行编码:
  public static String encode(String s)

这个方法总是用它所在平台的默认编码形式,所以在不同系统上,它就会产生不同的结果。结果java1.4中,这个方法被另一种方法取代了。该方法要求你自己指定编码形式:

public static String encode(String s, String encoding) throws UnsupportedEncodingException

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 两种关于编码的方法,都把任何非字母数字字符转换成%xx(除了空格,下划线(_),连字符(?,句号(。),和星号(*))。两者也都编码所以的非ASCII字符。空格被转换成一个加号。这些方法有一点过分累赘了;它们也把“~”,“‘”,“()”转换成%xx,即使它们完全用不着这样做。尽管这样,但是这种转换并没被URL规范所禁止。所以web浏览器会自然地处理这些被过分编码后的URL。

 两中关于编码的方法都返回一个新的被编码后的string,java1.3的方法encode( ) 使用了平台的默认编码形式,得到%xx。这些编码形式典型的有:在 U.S. Unix 系统上的ISO-8859-1,U.S. Windows 系统上的Cp1252,U.S. Macs上的MacRoman,和其他本地字符集等。因为编码解码过程都是与本地操作平台相关的,所以这些方法是令人不爽的,不能跨平台的。
 这就明确地回答了为什么在java1.4中这种方法被抛弃了,转而投向了要求以自己指定编码形式的这种方法。尽管如此,如果你执意要使用所在平台的默认编码形式,你的程序将会像在java1.3中的程序一样,是本地平台相关的。在另一种编码的方法中,你应该总是用UTF-8,而不是其他什么。 UTF-8比起你选的其他的编码形式来说,它能与新的web浏览器和更多的其他软件相兼容。

 例子7-8是使用URLEncoder.encode( ) 来打印输出各种被编码后的string。它需要在java1.4或更新的版本中编译和运行。

Example 7-8. x-www-form-urlencoded strings

import java.net.URLEncoder; import java.net.URLDecoder; import java.io.UnsupportedEncodingException; public class EncoderTest { public static void main(String[] args) { try { System.out.println(URLEncoder.encode(“This string has spaces”,“UTF-8”)); System.out.println(URLEncoder.encode(“Thisstringhas*asterisks”,“UTF-8”)); System.out.println(URLEncoder.encode(“This%string%has%percent%signs”, “UTF-8”)); System.out.println(URLEncoder.encode(“This+string+has+pluses”,“UTF-8”)); System.out.println(URLEncoder.encode(“This/string/has/slashes”,“UTF-8”)); System.out.println(URLEncoder.encode(“This”string”has”quote”marks”, “UTF-8”)); System.out.println(URLEncoder.encode(“This:string:has:colons”,“UTF-8”)); System.out.println(URLEncoder.encode(“Thisstringhas~tildes”,“UTF-8”)); System.out.println(URLEncoder.encode(“This(string)has(parentheses)”, “UTF-8”)); System.out.println(URLEncoder.encode(“This.string.has.periods”,“UTF-8”)); System.out.println(URLEncoder.encode(“This=string=has=equals=signs”, “UTF-8”)); System.out.println(URLEncoder.encode(“This&string&has&ersands”,“UTF-8”)); System.out.println(URLEncoder.encode(“Thiséstringéhasé non-ASCII characters”,“UTF-8”)); // System.out.println(URLEncoder.encode(“this中华人民共和国”,“UTF-8”)); } catch (UnsupportedEncodingException ex) { throw new RuntimeException(“Broken VM does not support UTF-8”); } } } 下面就是它的输出。需要注意的是这些代码应该以其他编码形式被保存而不是以ASCII码的形式,还有就是你选择的编码形式应该作为一个参数传给编译器,让编译器能据此对源代码中的非ASCII字符作出正确的解释。

% javac -encoding UTF8 EncoderTest %

java EncoderTest This+string+has+spaces Thisstringhas*asterisks This%25string%25has%25percent%25signs This%2Bstring%2Bhas%2Bpluses This%2Fstring%2Fhas%2Fslashes This%22string%22has%22quote%22marks This%3Astring%3Ahas%3Acolons This%7Estring%7Ehas%7Etildes This%28string%29has%28parentheses%29 This.string.has.periods This%3Dstring%3Dhas%3Dequals%3Dsigns This%26string%26has%26ampersands This%C3%A9string%C3%A9has%C3%A9non-ASCII+characters

特别需要注意的是这个方法编码了符号,“\” ,&,=,和:。它不会尝试着去规定在一个URL中这些字符怎样被使用。由此,所以你不得不分块编码你的URL,而不是把整个URL一次传给这个方法。这是很重要的,因为对类URLEncoder最通常的用法就是查询string,为了和服务器端使用GET方法的程序进行交互。例如,假设你想编码这个查询sting,它用来搜索AltaVista网站: pg=q&kl=XX&stype=stext&q=+“Java+I/O”&search.x=38&search.y=3

这段代码对其进行编码: String query = URLEncoder.encode( “pg=q&kl=XX&stype=stext&q=+“Java+I/O”&search.x=38&search.y=3”);System.out.println(query);

不幸的是,得到的输出是: pg%3Dq%26kl%3DXX%26stype%3Dstext%26q%3D%2B%22Java%2BI%2FO%22%26search.x%3D38%26search.y%3D3

出现这个问题就是方法URLEncoder.encode( ) 在进行盲目地编码。它不能区分在URL或者查询string中被用到的特殊字符(象前面string中的“=”,和“&”)和确实需要被编码的字符。由此,所以URL需要像下面这样一次只编码一块:

String query = URLEncoder.encode(“pg”); query += “=”; query += URLEncoder.encode(“q”); query += “&”; query += URLEncoder.encode(“kl”); query += “=”; query += URLEncoder.encode(“XX”); query += “&”; query += URLEncoder.encode(“stype”); query += “=”; query += URLEncoder.encode(“stext”); query += “&”; query += URLEncoder.encode(“q”); query += “=”; query += URLEncoder.encode(““Java I/O””); query += “&”; query += URLEncoder.encode(“search.x”); query += “=”; query += URLEncoder.encode(“38”); query += “&”; query += URLEncoder.encode(“search.y”); query += “=”; query += URLEncoder.encode(“3”); System.out.println(query);

这才是你真正想得到的输出: pg=q&kl=XX&stype=stext&q=%2B%22Java+I%2FO%22&search.x=38&search.y=3

例子7-9是一个QueryString类。在一个java对象中,它使用了类URLEncoder来编码连续的属性名和属性值对,这个java对象被用来发送数据到服务器端的程序。

当你在创建一个QueryString对象时,你可以把查询string中的第一个属性对传递给类QueryString的构造函数,得到初始 string。如果要继续加入后面的属性对,就应调用方法add(),它也能接受两个string作为参数,能对它们进行编码。方法getQuery( )返回一个属性对被逐个编码后得到的整个string。

Example 7-9. -The QueryString class

import java.net.URLEncoder; import java.io.UnsupportedEncodingException;

public class QueryString { private StringBuffer query = new StringBuffer();

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public QueryString(String name, String value) { 
    encode(name, value);
}

public synchronized void add(String name, String value) {
    query.append('&');
    encode(name, value);
}

private synchronized void encode(String name, String value) {
  try {
    query.append(URLEncoder.encode(name, "UTF-8"));
    query.append('=');
    query.append(URLEncoder.encode(value, "UTF-8"));
    } catch (UnsupportedEncodingException ex) {
         throw new RuntimeException("Broken VM does not support UTF-8");
    }
}

public String getQuery() {
    return query.toString();
}

public String toString() {
    return getQuery();
}

}

利用这个类,现在我们就能对前面那个例子中的string进行编码了:

QueryString qs = new QueryString(“pg”, “q”); qs.add(“kl”, “XX”); qs.add(“stype”, “stext”); qs.add(“q”, “+“Java I/O””); qs.add(“search.x”, “38”); qs.add(“search.y”, “3”); String url = “http://www.altavista.com/cgi-bin/query?” + qs; System.out.println(url);

二、URLDecoder 与URLEncoder 类相对应的URLDecoder 类有两种静态方法。它们解码以x-www-form-url-encoded这种形式编码的string。也就是说,它们把所有的加号(+)转换成空格符,把所有的%xx分别转换成与之相对应的字符: public static String decode(String s) throws Exception public static String decode(String s, String encoding) // Java 1.4 throws UnsupportedEncodingException

第一种解码方法在java1.3和java1.2中使用。第二种解码方法在java1.4和更新的版本中使用。如果你拿不定主意用哪种编码方式,那就选择UTF-8吧。它比其他任何的编码形式更有可能得到正确的结果。

如果string包含了一个“%”,但紧跟其后的不是两位16进制的数或者被解码成非法序列,该方法就会抛出 IllegalArgumentException 异常。当下次再出现这种情况时,它可能就不会被抛出了。这是与运行环境相关的,当检查到有非法序列时,抛不抛出 IllegalArgumentException 异常,这时到底会发生什么是不确定的。在Sun’s JDK 1.4中,不会抛出什么异常,它会把一些莫名其妙的字节加进不能被顺利编码的string中。这的确令人头疼,可能就是一个安全漏洞。

由于这个方法没有触及到非转义字符,所以你可以把整个URL作为参数传给该方法,不用像之前那样分块进行。例如:

String input = “http://www.altavista.com/cgi-bin/” + “query?pg=q&kl=XX&stype=stext&q=%2B%22Java+I%2FO%22&search.x=38&search.y=3”; try { String output = URLDecoder.decode(input, “UTF-8”); System.out.println(output); }


作者:qq_39361915 来源:CSDN 原文:https://blog.csdn.net/qq_39361915/article/details/98037782

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/147732.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
des加密,url编码,url解码,des解密 DES加解密及Wrong key size错误处理
des加密,url编码,url解码,des解密 DES加解密及Wrong key size错误处理
oktokeep
2024/10/09
1750
传参base64时的+号变空格问题
上上周,看到一位老哥找我们组同事联调接口,不知道是什么问题,两人坐一起搞了快1个小时,看起来好像有点复杂。
科技新语
2023/03/28
1.2K0
百度GPSutil「建议收藏」
=================================================
全栈程序员站长
2022/07/07
2660
解码和转码 原
  import Java.NET.URLDecoder;   import java.Net.URLEncoder;
wuweixiang
2018/08/14
3.3K0
SOP页面跳转设计 RAS AES加密算法应用跨服务免登陆接口设计
SOP,是 Standard Operating Procedure三个单词中首字母的大写 ,即标准作业程序,指将某一事件的标准操作步骤和要求以统一的格式描述出来,用于指导和规范日常的工作。
oktokeep
2024/10/09
730
SOP页面跳转设计 RAS AES加密算法应用跨服务免登陆接口设计
九、状态管理
将浏览器与web服务器之间多次交互当作一个整体来处理,并且将多次交互所涉及的数据(即状态)保存下来。
Qwe7
2022/07/24
3910
url编码和解码分析URLEncoder.encode和URLDecoder.decode
1.Get请求会将参数做默认的url解码操作,接口接收到的值是Get解码后的值。 2.可以将Get操作修改成Post操作,这样不会url解码。可以在接口中做url解码。 3.在多次传递参数的过程中,无需反复的编码(或者加了空格,加了换行),否则会将整个字符串错乱了。 (/ %2F %252F) (+ %2B %252B)
oktokeep
2024/10/09
1900
url编码和解码分析URLEncoder.encode和URLDecoder.decode
Web阶段:第十五章:文件上传&下载
文件的上传 1、首先需要一个form表单 2、然后在表单内有input type=”file” 3、提交的方式必须是method=”POST” 4、enctype="multipart/form-data"编码类型必须是多段的表单数据
马克社区
2023/09/08
1850
dk1.8和jdk17底层对url参数的编码不一样吗
当涉及到处理 URL 参数时,JDK 1.8和 JDK 17 在底层的编码方式上有一些差异。在本文中,我们将通过一个具体的案例来说明这些差异,并提供相应的代码示例。
GeekLiHua
2025/01/21
780
判断URL是否编码,编码后的sign对签名和验签都有影响,导致验签不通过
判断URL是否编码,编码后的sign对签名和验签都有影响,导致验签不通过 如果含有 + %符号无法判断, 否则判断不准或报错 Exception in thread "main" java.lang.IllegalArgumentException: URLDecoder: Illegal hex characters in escape (%) pattern - For input string: "#&"
oktokeep
2024/10/09
1010
MalformedURLException:无效的URL格式完美解决方法
大家好,我是默语!在这篇文章中,我们将深入探讨Java中的MalformedURLException,它是一个常见的异常,通常在处理URL时出现。作为一名初学者,理解这个异常的原因、常见场景以及解决方案非常重要。本文将提供详细的解释和示例,帮助你有效避免和解决这个问题,让你在编程的道路上更加顺利!🚀
默 语
2024/11/22
3010
Base64和URL编解码操作
在下载文件的时候有时候文件名称中含有中文名,下载下来后会乱码,所以就对文件名称进行一些编解码操作,来解决乱码。 BASE64编解码(解决火狐浏览器乱码): new BASE64Encoder().encode(需要编码的字节数组) —> 编码 new BASE64Decoder().decodeBuffer(解码内容) —> 解码 URL编解码: URLEncoder.encode(需要编码的内容, “UTF-8”); —> 编码 URLDecoder.decode(需要解码的内容,“UTF
阿年、嗯啊
2021/04/27
1.8K0
Web开发须知:URL编码与解码
通常如果一样东西需要编码,说明这样东西并不适合传输。原因多种多样,如Size过大,包含隐私数据,对于Url来说,之所以要进行编码,是因为Url中有些字符会引起歧义。   例如,Url参数字符串中使用k
李海彬
2018/03/27
2.6K0
Web开发须知:URL编码与解码
java urlencoder,java中的URLEncoder和URLDecoder类「建议收藏」
URLEncoder类包含将字符串转换为application/x-www-form-urlencoded MIME 格式的静态方法。
全栈程序员站长
2022/09/14
7780
WEB:字符集、编码、乱码 —— 看这篇就够了
ASCII(美国信息交换标准代码)是基于拉丁字母的一套电脑编码系统。它主要用于显示现代英语,是现今最通用的单字节编码系统。
WEBJ2EE
2019/07/19
4.2K0
WEB:字符集、编码、乱码 —— 看这篇就够了
php des加密用java解析不了 改个模式 加个IV php密钥/IV要求都是8位
php des加密用java解析不了 改个模式 加个IV  php密钥/IV要求都是8位
oktokeep
2024/10/09
930
Java中文异常——全解
如果在处理中文字符时出现乱码或无法正确显示的情况,很可能是由于字符集不匹配导致的。Java中常用的字符集是UTF-8和GBK,确保输入、输出和存储的字符集一致。
红目香薰
2023/10/11
2050
Java中文异常——全解
浅谈URLEncoder编码算法
一、为什么要用URLEncoder 客户端在进行网页请求的时候,网址中可能会包含非ASCII码形式的内容,比如中文。 而直接把中文放到网址中请求是不允许的,所以需要用URLEncoder编码地址, 将网址中的非ASCII码内容转换成可以传输的字符 不会被编码的内容 1.大写字母A-Z 2.小写字母a-z 3.数字 0-9 4.标点符 - _ . ! ~ * ' (和 ,) 二、编码原理 1、将需要转换的内容(ASCII码形式之外的内容),用十六进制表示法转换出来,并在之前加上%开头 eg:  0x9c  U
听着music睡
2018/05/18
1.7K0
Web阶段:第十五章:文件上传&下载
文件的上传 1、首先需要一个form表单 2、然后在表单内有input type=”file” 3、提交的方式必须是method=”POST” 4、enctype="multipart/form-data"编码类型必须是多段的表单数据
Java廖志伟
2022/09/28
3350
Web阶段:第十五章:文件上传&下载
【Java】22 网络连接
java.net.InetAddress此类表示互联网协议 (IP) 地址。IP 地址是 IP 使用的 32 位或 128 位无符号数字,它是一种低级协议,UDP 和 TCP 协议都是在它的基础上构建的。InetAddress 的实例包含 IP 地址,还可能包含相应的主机名(取决于它是否用主机名构造或者是否已执行反向主机名解析)。
Demo_Null
2020/09/28
1.4K0
【Java】22 网络连接
推荐阅读
相关推荐
des加密,url编码,url解码,des解密 DES加解密及Wrong key size错误处理
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验