`
zhangziyangup
  • 浏览: 1078605 次
文章分类
社区版块
存档分类
最新评论

Java SE 6 新特性: HTTP 增强

 
阅读更多

  概述

  Java 语言从诞生的那天起,就非常注重网络编程方面的应用。随着互联网应用的飞速发展,Java 的基础类库也不断地对网络相关的 API 进行加强和扩展。在 Java SE 6 当中,围绕着 HTTP 协议出现了很多实用的新特性:NTLM 认证提供了一种 Window 平台下较为安全的认证机制;JDK 当中提供了一个轻量级的 HTTP 服务器;提供了较为完善的 HTTP Cookie 管理功能;更为实用的 NetworkInterface;DNS 域名的国际化支持等等。

  NTLM 认证

  不可避免,网络中有很多资源是被安全域保护起来的。访问这些资源需要对用户的身份进行认证。下面是一个简单的例子:

  

import java.net.*;

import java.io.*;

public class Test {

public static void main(String[] args) throws Exception {

URL url = new URL("http://PROTECTED.com");

URLConnection connection = url.openConnection();

InputStream in = connection.getInputStream();

byte[] data = new byte[1024];

while(in.read(data)>0)

{

//do something for data

}

in.close();

}

}



  当 Java 程序试图从一个要求认证的网站读取信息的时候,也就是说,从联系于 http://Protected.com 这个 URLConnection 的 InputStream 中 read 数据时,会引发 FileNotFoundException。尽管笔者认为,这个 Exception 的类型与实际错误发生的原因实在是相去甚远;但这个错误确实是由网络认证失败所导致的。

  要解决这个问题,有两种方法:

  其一,是给 URLConnection 设定一个“Authentication”属性:

  

String credit = USERNAME + ":" + PASSWORD;

String encoding = new sun.misc.BASE64Encoder().encode (credit.getBytes());

connection.setRequestProperty ("Authorization", "Basic " + encoding);



  这里假设 http://PROTECTED.COM 使用了基本(Basic)认证类型。

  从上面的例子,我们可以看出,设定 Authentication 属性还是比较复杂的:用户必须了解认证方式的细节,才能将用户名/密码以一定的规范给出,然后用特定的编码方式加以编码。Java 类库有没有提供一个封装了认证细节,只需要给出用户名/密码的工具呢?

  这就是我们要介绍的另一种方法,使用 java.net.Authentication 类。

  每当遇到网站需要认证的时候,HttpURLConnection 都会向 Authentication 类询问用户名和密码。

  Authentication 类不会知道究竟用户应该使用哪个 username/password 那么用户如何向 Authentication 类提供自己的用户名和密码呢?

  提供一个继承于 Authentication 的类,实现 getPasswordAuthentication 方法,在 PasswordAuthentication 中给出用户名和密码:

  

class DefaultAuthenticator extends Authenticator {

public PasswordAuthentication getPasswordAuthentication () {

return new PasswordAuthentication ("USER", "PASSWORD".toCharArray());

}

}



  然后,将它设为默认的(全局)Authentication:

  

Authenticator.setDefault (new DefaultAuthenticator());



  那么,不同的网站需要不同的用户名/密码又怎么办呢?

  Authentication 提供了关于认证发起者的足够多的信息,让继承类根据这些信息进行判断,在 getPasswordAuthentication 方法中给出了不同的认证信息:

  getRequestingHost()

  getRequestingPort()

  getRequestingPrompt()

  getRequestingProtocol()

  getRequestingScheme()

  getRequestingURL()

  getRequestingSite()

  getRequestorType()

  另一件关于 Authentication 的重要问题是认证类型。不同的认证类型需要 Authentication 执行不同的协议。至 Java SE 6.0 为止,Authentication 支持的认证方式有:

  HTTP Basic authentication

  HTTP Digest authentication

  NTLM

  Http SPNEGO Negotiate

  Kerberos

  NTLM

  这里我们着重介绍 NTLM。

  NTLM 是 NT LAN Manager 的缩写。早期的 SMB 协议在网络上明文传输口令,这是很不安全的。微软随后提出了 WindowsNT 挑战/响应验证机制,即 NTLM。

  NTLM 协议是这样的:

  客户端首先将用户的密码加密成为密码散列;

  客户端向服务器发送自己的用户名,这个用户名是用明文直接传输的;

  服务器产生一个 16 位的随机数字发送给客户端,作为一个 challenge(挑战) ;

  客户端用步骤1得到的密码散列来加密这个 challenge ,然后把这个返回给服务器;

  服务器把用户名、给客户端的 challenge 、客户端返回的 response 这三个东西,发送域控制器 ;

  域控制器用这个用户名在 SAM 密码管理库中找到这个用户的密码散列,然后使用这个密码散列来加密 challenge;

  域控制器比较两次加密的 challenge ,如果一样,那么认证成功;

  Java 6 以前的版本,是不支持 NTLM 认证的。用户若想使用 HttpConnection 连接到一个使用有 Windows 域保护的网站时,是无法通过 NTLM 认证的。另一种方法,是用户自己用 Socket 这样的底层单元实现整个协议过程,这无疑是十分复杂的。

  终于,Java 6 的 Authentication 类提供了对 NTLM 的支持。使用十分方便,就像其他的认证协议一样:

  

class DefaultAuthenticator extends Authenticator {

private static String username = "username ";

private static String domain = "domain ";

private static String password = "password ";



public PasswordAuthentication getPasswordAuthentication() {

String usernamewithdomain = domain + "/ "+username;

return (new PasswordAuthentication(usernamewithdomain, password.toCharArray()));

}

}



  这里,根据 Windows 域账户的命名规范,账户名为域名+”/”+域用户名。如果不想每生成 PasswordAuthentication 时,每次添加域名,可以设定一个系统变量名“http.auth.ntlm.domain“。

  Java 6 中 Authentication 的另一个特性是认证协商。目前的服务器一般同时提供几种认证协议,根据客户端的不同能力,协商出一种认证方式。比如,IIS 服务器会同时提供 NTLM with kerberos 和 NTLM 两种认证方式,当客户端不支持 NTLM with kerberos 时,执行 NTLM 认证。

  目前,Authentication 的默认协商次序是:

  GSS/SPNEGO -> Digest -> NTLM -> Basic



  那么 kerberos 的位置究竟在哪里呢?

  事实上,GSS/SPNEGO 以 JAAS 为基石,而后者实际上就是使用 kerberos 的。

  轻量级 HTTP 服务器

  Java 6 还提供了一个轻量级的纯 Java Http 服务器的实现。下面是一个简单的例子:

  

public static void main(String[] args) throws Exception{

HttpServerProvider httpServerProvider = HttpServerProvider.provider();

InetSocketAddress addr = new InetSocketAddress(7778);

HttpServer httpServer = httpServerProvider.createHttpServer(addr, 1);

httpServer.createContext("/myapp/", new MyHttpHandler());

httpServer.setExecutor(null);

httpServer.start();

System.out.println("started");

}

static class MyHttpHandler implements HttpHandler{

public void handle(HttpExchange httpExchange) throws IOException {

String response = "Hello world!";

httpExchange.sendResponseHeaders(200, response.length());

OutputStream out = httpExchange.getResponseBody();

out.write(response.getBytes());

out.close();

}

}



  然后,在浏览器中访问 http://localhost:7778/myapp/,我们得到:

  图一 浏览器显示

  

  

  首先,HttpServer 是从 HttpProvider 处得到的,这里我们使用了 JDK 6 提供的实现。用户也可以自行实现一个 HttpProvider 和相应的 HttpServer 实现。

  其次,HttpServer 是有上下文(context)的概念的。比如,http://localhost:7778/myapp/ 中“/myapp/”就是相对于 HttpServer Root 的上下文。对于每个上下文,都有一个 HttpHandler 来接收 http 请求并给出回答。

  最后,在 HttpHandler 给出具体回答之前,一般先要返回一个 Http head。这里使用 HttpExchange.sendResponseHeaders(int code, int length)。其中 code 是 Http 响应的返回值,比如那个著名的 404。length 指的是 response 的长度,以字节为单位。

  Cookie 管理特性

  Cookie 是 Web 应用当中非常常用的一种技术, 用于储存某些特定的用户信息。虽然,我们不能把一些特别敏感的信息存放在 Cookie 里面,但是,Cookie 依然可以帮助我们储存一些琐碎的信息,帮助 Web 用户在访问网页时获得更好的体验,例如个人的搜索参数,颜色偏好以及上次的访问时间等等。网络程序开发者可以利用 Cookie 来创建有状态的网络会话(Stateful Session)。 Cookie 的应用越来越普遍。在 Windows 里面,我们可以在“Documents And Settings”文件夹里面找到IE使用的 Cookie,假设用户名为 admin,那么在 admin 文件夹的 Cookies 文件夹里面,我们可以看到名为“admin@(domain)”的一些文件,其中的 domain 就是表示创建这些 Cookie 文件的网络域, 文件里面就储存着用户的一些信息。

  JavaScript 等脚本语言对 Cookie 有着很不错的支持。 .NET 里面也有相关的类来支持开发者对 Cookie 的管理。 不过,在 Java SE 6 之前, Java一直都没有提供 Cookie 管理的功能。在 Java SE 5 里面, java.net 包里面有一个 CookieHandler 抽象类,不过并没有提供其他具体的实现。到了 Java SE 6, Cookie 相关的管理类在 Java 类库里面才得到了实现。有了这些 Cookie 相关支持的类,Java 开发者可以在服务器端编程中很好的操作 Cookie, 更好的支持 HTTP 相关应用,创建有状态的 HTTP 会话。

  用 HttpCookie 代表 Cookie

  java.net.HttpCookie 类是 Java SE 6 新增的一个表示 HTTP Cookie 的新类, 其对象可以表示 Cookie 的内容, 可以支持所有三种 Cookie 规范:

  Netscape 草案

  RFC 2109 - http://www.ietf.org/rfc/rfc2109.txt

  RFC 2965 - http://www.ietf.org/rfc/rfc2965.txt

  这个类储存了 Cookie 的名称,路径,值,协议版本号,是否过期,网络域,最大生命期等等信息。

  用 CookiePolicy 规定 Cookie 接受策略

  java.net.CookiePolicy 接口可以规定 Cookie 的接受策略。 其中唯一的方法用来判断某一特定的 Cookie 是否能被某一特定的地址所接受。 这个类内置了 3 个实现的子类。一个类接受所有的 Cookie,另一个则拒绝所有,还有一个类则接受所有来自原地址的 Cookie。

  用CookieStore 储存 Cookie

  java.net.CookieStore 接口负责储存和取出 Cookie。 当有 HTTP 请求的时候,它便储存那些被接受的 Cookie; 当有 HTTP 回应的时候,它便取出相应的 Cookie。 另外,当一个 Cookie 过期的时候,它还负责自动删去这个 Cookie。

  用 CookieManger/CookieHandler 管理 Cookie

  java.net.CookieManager 是整个 Cookie 管理机制的核心,它是 CookieHandler 的默认实现子类。下图显示了整个 HTTP Cookie 管理机制的结构:

  图 2. Cookie 管理类的关系

  

  

  一个 CookieManager 里面有一个 CookieStore 和一个 CookiePolicy,分别负责储存 Cookie 和规定策略。用户可以指定两者,也可以使用系统默认的 CookieManger。

  例子

  下面这个简单的例子说明了 Cookie 相关的管理功能:

  

// 创建一个默认的 CookieManager

CookieManager manager = new CookieManager();

// 将规则改掉,接受所有的 Cookie

manager.setCookiePolicy(CookiePolicy.ACCEPT_ALL);

// 保存这个定制的 CookieManager

CookieHandler.setDefault(manager);



// 接受 HTTP 请求的时候,得到和保存新的 Cookie

HttpCookie cookie = new HttpCookie("...(name)...","...(value)...");

manager.getCookieStore().add(uri, cookie);



// 使用 Cookie 的时候:

// 取出 CookieStore

CookieStore store = manager.getCookieStore();

// 得到所有的 URI

List<uri></uri> uris = store.getURIs();

for (URI uri : uris) {

// 筛选需要的 URI

// 得到属于这个 URI 的所有 Cookie

List<httpcookie></httpcookie> cookies = store.get(uri);

for (HttpCookie cookie : cookies) {

// 取出了 Cookie

}

}



// 或者,取出这个 CookieStore 里面的全部 Cookie

// 过期的 Cookie 将会被自动删除

List<httpcookie></httpcookie> cookies = store.getCookies();

for (HttpCookie cookie : cookies) {

// 取出了 Cookie

}



  其他新特性

  NetworkInterface 的增强

  从 Java SE 1.4 开始,JDK 当中出现了一个网络工具类 java.net.NetworkInterface,提供了一些网络的实用功能。 在 Java SE 6 当中,这个工具类得到了很大的加强,新增了很多实用的方法。例如:

  public boolean isUp()

  用来判断网络接口是否启动并运行

  public boolean isLoopback()

  用来判断网络接口是否是环回接口(loopback)

  public boolean isPointToPoint()

  用来判断网络接口是否是点对点(P2P)网络

  public boolean supportsMulticast()

  用来判断网络接口是否支持多播

  public byte[] getHardwareAddress()

  用来得到硬件地址(MAC)

  public int getMTU()

  用来得到最大传输单位(MTU,Maximum Transmission Unit)

  public boolean isVirtual()

  用来判断网络接口是否是虚拟接口

  关于此工具类的具体信息,请参考 Java SE 6 相应文档(见 参考资源)。

  域名的国际化

  在最近的一些 RFC 文档当中,规定 DNS 服务器可以解析除开 ASCII 以外的编码字符。有一个算法可以在这种情况下做 Unicode 与 ASCII 码之间的转换,实现域名的国际化。java.net.IDN 就是实现这个国际化域名转换的新类,IDN 是“国际化域名”的缩写(internationalized domain names)。这个类很简单,主要包括 4 个静态函数,做字符的转换。

  结语

  Java SE 6 有着很多 HTTP 相关的新特性,使得 Java SE 平台本身对网络编程,尤其是基于 HTTP 协议的因特网编程,有了更加强大的支持。

  参考资料

  阅读 Java SE 6 新特性系列文章的完整列表,了解 Java SE 6 其它重要的增强。

  Java SE 6 文档:Java SE 6 的规范文档,可以找到绝大部分新特性的官方说明。

  NTLM 身份验证微软 TechNet 相关介绍 NTLM 的文章

  RFC 2109、RFC 2965:关于 Cookie 的两个 RFC

  
分享到:
评论

相关推荐

    Java SE 6 新特性 HTTP 增强

    Java SE 6 新特性 HTTP 增强.htm

    Java SE 6 新特性 HTTP 增强.doc

    Java SE 6 新特性 HTTP 增强.doc

    java8新特性

    java SE8 新特性,每一位java开发工程师的必备宝典,其中包括lambda表达式以及lambda编程、stream API处理、新的日期与时间API、JavaScript引擎、并发增强处理、杂项处理等等。

    Java SE Runtime Environment (JRE) 6.0 Update 24

    运行JAVA程序不可缺少的环境。一旦拥有Java软件,您即会进入一个全新的互动世界。  Java语言是稳居网络应用程序语言的...不过,如今的Java不仅在执行速度上有大幅度的改革,而且在内容上也有做了不少些修改以及增强。

    Java SE Runtime Environment 9.0u165 官方安装版(JRE含中文)

    Java SE Runtime Environment 9.0u165 官方安装版(JRE含中文) 运行JAVA程序不可缺少的环境Java SE Runtime。一旦拥有 Java 软件,您即会进入一个全新的jre8互动世界。  Java语言恐怕是稳居网路应用程序语言的...

    JDK-API-1-6-zzh-CN帮助文档中文版

    Java SE 6包括许多新的特性和改进,包括JDBC 4.0 API,Java Compiler API,Pluggable Annotation Processing API和新的XML和Web服务API等。此外,Java SE 6还包括对桌面应用程序的增强支持,如Swing,Java2D和Java ...

    java7u21最新版本

    这一升级版对Java Plug-in进行了功能增强,提供了对Netscape 6 Open JVM整合支持等等。由于JRE新增的功能以及程序修正之处相当多,如果需要详尽资料的话不妨可以参考Sun的官方网页。 java se runtime environment...

    Java SE Development Kit(JDK) v8.0.1110.14.zip

    Java SE Development Kit 8.0官方下载,JDK是Java开发环境。JDK(Java Development Kit)是Sun Microsystems针对Java开发员的产品。自从Java推出以来,JDK已经成为使用最广泛的Java SDK。JDK 是整个Java的核心,包括...

    Java (java运行环境)7 Update v4.3 绿色便携版.zip

    这一升级版对Java Plug-in进行了功能增强,提供了对Netscape 6 Open JVM整合支持等等。由于JRE新增的功能以及程序修正之处相当多,如果需要详尽资料的话不妨可以参考Sun的官方网页。 java se runtime environment...

    Java 14 新特性

    此版本所包含的JEP(Java增强提案,Java Enhancement Propostal),共16个新特性。其中包含两个孵化器模块(Incubator)、三个预览特性(Preview)、两个弃用功能(Deprecate)和两个删除功能(Remove)。 2. 模块...

    写给大忙人看的JAVA SE 8

    第9章你可能错过的Java 7特性 188 9.1 异常处理改进 190 9.1.1 try-with-resources语句 190 9.1.2 忽略异常 191 9.1.3 捕获多个异常 192 9.1.4 更简单地处理反射方法的异常 193 9.2 使用文件 193 9.2.1 Path 194 ...

    java8_64bit.zip

    加强了Lambda表达式,使成为java拥有最好编写的处理器代码,新的Nashorn引擎也能和JavaScript相互操作。 java8,64位特色说明: 带来一种新语言能将参数代码变为数据。 增加和修改了Java特性优化默认方法。 se平台...

    JavaEE 6 Servlet 3.0 中的新特性

    JavaEE 6Servlet 3.0 中的新特性 • 易于开发和部署 • 模块化web.xml • 动态配置 • 异步Servlet • Servlet 3.0 — 易于开发 – 主要关注点 • 增强了API 以便使用SE 5 中新的语言特性 – 例如:批注、泛型...

    jdk1.8 64位.zip

    • 显著增加和修改了Java语言特性,增添了标准库,包括默认方法,新的java.util.stream包及 Date-Time API。 • Compact Profiles包含 Java SE平台预定义子集,使应用程序员无需部署在整个平台,可运行在小型设备上...

    mysql-connector-java-5.1.25.zip

    此版本增强了对MySQL 5.6的支持,并提供了一些新的特性,如多语句执行和可插拔式身份验证。此外,MySQL Connector/J 5.1.25还包括了许多重要的错误修复和性能优化,使得它更加可靠和高效。如果你是Java程序员,使用...

    Java7 (JRE) Update 67 X64官方版.zip

    这一升级版对Java Plug-in进行了功能增强,提供了对Netscape 6 Open JVM整合支持等等。由于JRE新增的功能以及程序修正之处相当多,如果需要详尽资料的话不妨可以参考Sun的官方网页。多国语言版,支持简体中文界面。 ...

    JDK8-Java8-JavaSE8新特性和增强功能-第一部

    熟悉JDK8新特性和库对Android开发(基于Java语言)更加得心应手。 5. 熟悉JDK8新特性能方便进行Java底层开发,比如Spring MVC. 6. IDE目前基于轻便的VSCode开发说明, 也可以自行通过IDEAJ建立项目运行。

    jdk1.8 64位官方正式版(附安装文档)

    2、显著增加和修改了Java语言特性,增添了标准库,包括默认方法,新的java.util.stream包及 Date-Time API。 3、Compact Profiles包含 Java SE平台预定义子集,使应用程序员无需部署在整个平台,可运行在小型设备上...

    jdk1.8u144 官方版本windows

    JDK8.0是Java虚拟机的最新版本,加入了很多新的特性。Oracle甲骨文公司Oracle公司如期发布了Java 8正式版!现在你就可以下载Java 8正式版了,同期发布的还有JDK 8。JDK8的特性已经基本准备就绪,虽然一些特性还不是...

    Windows JDK8 官方版本

    JDK8.0是Java虚拟机的最新版本,加入了很多新的特性。Oracle甲骨文公司Oracle公司如期发布了Java 8正式版!现在你就可以下载Java 8正式版了,同期发布的还有JDK 8。JDK8的特性已经基本准备就绪,虽然一些特性还不是...

Global site tag (gtag.js) - Google Analytics