在开始讨论编码解码之前,首先来明确一下问题。

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

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

表单里提交时也是如此,当包含非西欧字符的字符串时,系统也会将这些字符转换成application/x-www-form-urlencoded字符串。

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

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

Browser端

表单的ENCTYPE属性值为multipart/form-data,它告诉我们传输的数据要用到多媒体传输协议,由于多媒体传输的都是大量的数据,所以规    定上传文件必须是post方法,的type属性必须是file。

form中的其他enctype请查看这里。跑题了我。

我们经常会在浏览器的地址栏里看到这样的字符串%E6%96%87%E6%A1%A3

这就是被编码后的字符串,下面我们就讨论一下java 的url编码与解码问题

代码

java.net.URLDecoder.decode(String s,String enc);

将application/x-www-form-urlencoded字符串转换成普通字符串。

java.net.URLEncoder.decode(String s,String enc);

将普通字符串转换成application/x-www-form-urlencoded字符串

代码

importjava.net.URLDecoder;importjava.net.URLEncoder;publicclassURLDecoderTest {publicstaticvoidmain(String[] args)throwsException {//将application/x-www-form-urlencoded字符串//转换成普通字符串//必须强调的是编码方式必须正确,如baidu的是gb2312,而google的是UTF-8String keyWord=URLDecoder.decode("%E6%96%87%E6%A1%A3","gb2312");

System.out.println(keyWord);//将普通字符串转换成//application/x-www-form-urlencoded字符串//必须强调的是编码方式必须正确,如baidu的是gb2312,而google的是UTF-8String urlStr=URLEncoder.encode("文档","gb2312");

System.out.println(urlStr);

}

}

下面转载一片文章

/*

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

1.字符"a"-"z","A"-"Z","0"-"9",".","-","*",和"_" 都不会被编码;

2.将空格转换为加号 (+) ;

3.将非文本内容转换成"%xy"的形式,xy是两位16进制的数值;

4.在每个 name=value 对之间放置 & 符号。

*/

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

web 设计者面临的众多难题之一便是怎样处理不同操作系统间的差异性。这些差异性能引起URL方面的问题:例如,一些操作系统允许文件名中含有空格符,有些又不允许。大多数操作系统不会认为文件名中含有符号“#”会有什么特殊含义;但是在一个URL中,符号“#”表示该文件名已经结束,后面会紧跟一个 fragment(部分)标识符。其他的特殊字符,非字母数字字符集,它们在URL或另一个操作系统上都有其特殊的含义,表述着相似的问题。为了解决这些问题,我们在URL中使用的字符就必须是一个ASCII字符集的固定字集中的元素,具体如下:1.大写字母A-Z2.小写字母a-z3.数字 0-94.标点符 - _ . ! ~ * ' (和 ,)诸如字符: / & ? @ # ; $ + = 和 %也可以被使用,但是它们各有其特殊的用途,如果一个文件名包括了这些字符( / & ? @ # ; $ + = %),这些字符和所有其他字符就应该被编码。编码过程非常简单,任何字符只要不是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对象的字符串对象负责,确保所有字符都会被恰当地编码。

幸运的是,java提供了一个类URLEncoder把string编码成这种形式。Java1.2增加了一个类URLDecoder它能以这种形式解码string。这两个类都不用初始化:

public class URLDecoder extends Object

public class URLEncoder extends Object

一、URLEncoder

在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

两种关于编码的方法,都把任何非字母数字字符转换成%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 stringsimport 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("This*string*has*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("This~string~has~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

This*string*has*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

package com.macfaq.net;

import java.net.URLEncoder;

import java.io.UnsupportedEncodingException;

public class QueryString {

private StringBuffer query = new StringBuffer();

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);

}

url编码 java_java中的url 编码与解码相关推荐

  1. java url类_Java中的URL类和示例 - Break易站

    URL类是Internet上任何可用资源的网关.类URL表示统一资源定位符,它是指向万维网上"资源"的指针.资源可以指向简单的文件或目录,也可以引用更复杂的对象,例如对数据库或搜索 ...

  2. gbk转unicode java_Java中的Unicode编码转换问题

    1)不可避免的编码转换 计算机只能理解和存储二进制字节,而文字实际上是一种图形,所以要使用一种编码方法将图形转换为对应的二进制字节. -Java中这种编码为Unicode. -中文Window系统中默 ...

  3. html截取url字段,Html中截取url参数 实现HTML间的url传值

    大家好: 今天遇到一个问题,页面全是html,url传值,竟然获取不到参数值: A.html //登录按钮 jQuery(function($) { $("#login").cli ...

  4. 韩文编码python_python中的字符串编码问题——4.unicode编解码(以实际工作中遇到的韩文编码为例)...

    韩文unicode编解码 问题是这样,工作中遇到有韩文数据出现乱码,说是unicode码. 类似这样: id name323 52186863 149 63637538 314 65516863 32 ...

  5. nginx php跳转url参数,NGINX 中把url中的内容当初参数处理

    作用:把url为:127.0.0.1/index.php/a/b/c 这样处理 把 post参数: /a/b/c到127.0.0.1/index.php nginx 配置如下: server { li ...

  6. python中对字符串进行编码_Python 中的字符串编码

    对Python字符编码一直没搞明白,今天看<Python参考手册>再次遇到这个问题,重新整理下 Python中字符串字面量用于指定一个字符序列,其定义方法是把文本放入单引号('),双引号( ...

  7. python url解析_Python中实现URL的解析

    在Python中的urlparse模块主要是用于解析url中的参数  对url按照一定格式进行 拆分或拼接 1.urlparse.urlparse 将url分为6个部分,返回一个包含6个字符串项目的元 ...

  8. python字典编码_python中包含UTF-8编码中文的列表或字典的输出

    >>> dict = {"asdf": "我们的python学习"} >>> print dict {'asdf': '\x ...

  9. php html url编码,html中url编码是什么?有什么用?

    本篇文章给大家带来的内容是介绍HTML中的URL编码是什么,有什么用.有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所帮助. 我们在介绍URL编码之前,首先来了解一下URL是什么,URL的相 ...

  10. Python 中的url,Base64和MD5编码解码的使用

    1.encode()和decode() 我们所说的加密方式,都是对二进制编码的格式进行加密的,对应到Python中,则是我们的Bytes(字节). 所以当我们在Python中进行加密操作的时候,要确保 ...

最新文章

  1. redis命令_Redis 命令执行过程(下)
  2. Android Thread第二次Thread.start()报错的疑问
  3. info replication
  4. java如何创造一个整数的类_【技术干货】Java 面试宝典:Java 基础部分(1)
  5. 《Android 应用案例开发大全(第二版)》——导读
  6. javaWeb三大框架总结
  7. 最好的虚拟服务器,最好虚拟主机推荐给大家
  8. 你们要的二维码测试点来了,扫它!
  9. java做万年历,Java做的万年历
  10. Linux netstat 命令安装
  11. 微信群发红包原理 计算机,微信红包实现原理探讨
  12. java文字竖排_Java输出竖排文字
  13. 吴恩达深度学习总结(15)
  14. android项目版本,怎查看Android项目的Android版本
  15. 最高奖励68万!多邻国英语测试设立研究基金
  16. 什么是SEO,新手SEO须知!
  17. linux设备模型 —— sysfs
  18. 数学建模写作时相关的套路
  19. 【screen】解决screen中连不上Attached窗口的问题
  20. 让你读过的书都变成钱!

热门文章

  1. linux - android安卓 - stagefright
  2. laravel 语言包
  3. android如何局域网通信,android局域网怎么通信
  4. 限流算法: 漏桶算法和令牌桶算法
  5. XP系统优化超简单实用版
  6. Ubuntu下安装Master PDF Editor
  7. N81新手入门全攻略——N81常规设置问答篇
  8. 大气压力换算公式_压力公式换算
  9. 334个地级市名单_中国各级行政区划数量,统计到乡镇一级
  10. cef 前进后台 实现_使用CefSharp前端后台交换