一、概述
编写安全的Internet应用并不是一件轻而易举的事情:只要看看各个专业公告板就可以找到连续不断的安全漏洞报告。你如何保证自己的Internet应用不象其他人的应用那样满是漏洞?你如何保证自己的名字不会出现在令人难堪的重大安全事故报道中?如果你使用Java Servlet 、JavaServer Pages(JSP)或者EJB,许多难以解决的问题都已经事先解决。当然,漏洞仍有可能出现。下面我们就来看看这些漏洞是什么,以及为什么Java程序员不必担心部分C和Perl程序员必须面对的问题。
C程序员对安全漏洞应该已经很熟悉,但象OpenBSD之类的工程提供了处理此类问题的安全系统。Java语言处理这类问题的经验要比C少20年,但另一方面,Java作为一种客户端编程语言诞生,客户端对安全的要求比服务器端苛刻得多。它意味着Java的发展有着一个稳固的安全性基础。Java原先的定位目标是浏览器。然而,浏览器本身所带的Java虚拟机虽然很不错,但却并不完美。Sun的《Chronology of security-related bugs and issues》总结了运行时环境的漏洞发现历史。我们知道,当Java用作服务器端编程语言时,这些漏洞不可能被用作攻击手段。但即使Java作为客户端编程语言,重大安全问题的数量也从1996年的6个(其中3个是相当严重的问题)降低到2000年的1个。不过,这种安全性的相对提高并不意味着Java作为服务器端编程语言已经绝对安全,它只意味着攻击者能够使用的攻击手段越来越受到限制。那么,究竟有哪些地方容易受到攻击,其他编程语言又是如何面对类似问题的呢?
二、缓存溢出
在C程序中,缓存溢出是最常见的安全隐患。缓存溢出在用户输入超过已分配内存空间(专供用户输入使用)时出现。缓存溢出可能成为导致应用被覆盖的关键因素。C程序很容易出现缓存溢出,但Java程序几乎不可能出现缓存溢出。从输入流读取输入数据的C代码通常如下所示:
char buffer[1000];
int len = read(buffer);
由于缓存的大小在读入数据之前确定,系统要检查为输入保留的缓存是否足够是很困难的。缓存溢出使得用户能够覆盖程序数据结构的关键部分,从而带来了安全上的隐患。有经验的攻击者能够利用这一点直接把代码和数据插入到正在运行的程序。
在Java中,我们一般用字符串而不是字符数组保存用户输入。与前面C代码等价的Java代码如下所示:
String buffer = in.readLine();
在这里,“缓存”的大小总是和输入内容的大小完全一致。由于Java字符串在创建之后不能改变,缓存溢出也就不可能出现。退一步说,即使用字符数组替代字符串作为缓存,Java也不象C那样容易产生可被攻击者利用的安全漏洞。例如,下面的Java代码将产生溢出:
char[] bad = new char[6];
bad[7] = 50;
这段代码总是抛出一个java.lang.ArrayOutOfBoundsException异常,而该异常可以由程序自行捕获:
try {
har[] bad = new char[6];
bad[7] = 50;
}
catch (ArrayOutOfBoundsException ex) {
... }
这种处理过程永远不会导致不可预料的行为。无论用什么方法溢出一个数组,我们总是得到ArrayOutOfBoundsException异常,而Java运行时底层环境却能够保护自身免受任何侵害。一般而言,用Java字符串类型处理字符串时,我们无需担心字符串的ArrayOutOfBoundsExceptions异常,因此它是一种较为理想的选择。Java编程模式从根本上改变了用户输入的处理方法,避免了输入缓存溢出,从而使得Java程序员摆脱了最危险的编程漏洞。
三、竞争状态
竞争状态即Race Condition,它是第二类最常见的应用安全漏洞。在创建(更改)资源到修改资源以禁止对资源访问的临界时刻,如果某个进程被允许访问资源,此时就会出现竞争状态。这里的关键问题在于:如果一个任务由两个必不可少的步骤构成,不管你多么想要让这两个步骤一个紧接着另一个执行,操作系统并不保证这一点。例如,在数据库中,事务机制使得两个独立的事件“原子化”。换言之,一个进程创建文件,然后把这个文件的权限改成禁止常规访问;此同时,另外一个没有特权的进程可以处理该文件,欺骗有特权的进程错误地修改文件,或者在权限设置完毕之后仍继续对原文件进行访问。
一般地,在标准Unix和NT环境下,一些高优先级的进程能够把自己插入到任务的多个步骤之间,但这样的进程在Java服务器上是不存在的;同时,用纯Java编写的程序也不可能修改文件的许可权限。因此,大多数由文件访问导致的竞争状态在Java中不会出现,但这并不意味着Java完全地摆脱了这个问题,只不过是问题转到了虚拟机上。我们来看看其他各种开发平台如何处理这个问题。在Unix中,我们必须确保默认文件创建模式是安全的,比如在服务器启动之前执行“umask 200”这个命令。有关umask的更多信息,请在Unix系统的命令行上执行“man umask”查看umask的man文档。在NT环境中,我们必须操作ACL(访问控制表,Access Control List)的安全标记,保护要在它下面创建文件的目录。NT的新文件一般从它的父目录继承访问许可。请参见
NT文档了解更多信息。
Java中的竞争状态大多数时候出现在临界代码区。例如,在用户登录过程中,系统要生成一个唯一的数字作为用户会话的标识符。为此,系统先产生一个随机数字,然后在散列表之类的数据结构中检查这个数字是否已经被其他用户使用。如果这个数字没有被其他用户使用,则把它放入散列表以防止其他用户使用。代码如Listing 1所示:
(Listing 1)
// 保存已登录用户的ID
Hashtable hash;
// 随机数字生成器
Random rand;
// 生成一个随机数字
Integer id = new Integer(rand.nextInt());
while (hash.containsKey(id))
{
id = new Integer(rand.nextInt());
}
// 为当前用户保留该ID
hash.put(id, data);
Listing 1的代码可能带来一个严重的问题:如果有两个线程执行Listing 1的代码,其中一个线程在hash.put(...)这行代码之前被重新调度,此时同一个随机ID就有可能被使用两次。在Java中,我们有两种方法解决这个问题。首先,Listing 1的代码可以改写成Listing 2的形式,确保只有一个线程能够执行关键代码段,防止线程重新调度,避免竞争状态的出现。第二,如果前面的代码是EJB服务器的一部分,我们最好有一个利用EJB服务器线程控制机制的唯一ID服务。
(Listing 2)
synchronized(hash)
{
// 生成一个唯一的随机数字
Integer id =
new Integer(rand.nextInt());
while (hash.containsKey(id))
{
id = new Integer(rand.nextInt());
}
// 为当前用户保留该ID
hash.put(id, data);
}
分享到:
相关推荐
【Java服务器端编程安全】\n\n在Java服务器端编程中,安全性是至关重要的,因为一个小小的安全疏漏可能导致重大的数据泄露或系统崩溃。Java作为一种面向对象的编程语言,自诞生以来就注重安全特性,尤其在服务器端...
并发编程是现代软件开发中的重要组成部分,特别是在服务器端和分布式系统中,有效地利用多核处理器的能力,提高程序的执行效率是关键。Java语言为并发编程提供了丰富的API和工具,如线程、同步机制、并发集合以及...
在Java的`java.net`包中,提供了Socket和ServerSocket类来处理客户端和服务器端的连接。例如,ServerSocket用于监听客户端的连接请求,而Socket则用于建立实际的连接并发送/接收数据。 其次,HTTP(超文本传输协议...
Java编程语言是软件开发领域中最广泛使用的编程语言之一,尤其在企业级应用和服务器端开发中占据主导地位。"JAVA程序员必读"这个压缩包文件很可能是为了帮助Java程序员提升技能,了解最新的技术和最佳实践而准备的一...
J2EE则是企业级应用平台,包括服务器端技术,如Servlet、JSP等。 在实际编程过程中,掌握如何编译和运行Java程序至关重要。当编写好.java源文件后,使用javac命令进行编译,然后用java命令执行对应的.class文件。...
9. **网络编程**:理解Socket编程,如何创建客户端和服务器端的连接,进行数据传输。 10. **Java API**:讲解Java标准库中常用的类和接口,如Math、String、Date、Calendar等,并通过实例展示如何使用。 11. **...
16.2.1. 建立TCP服务器端 16-54 16.2.2. 建立TCP客户端 16-55 16.3. 建立URL连接 16-55 16.4. UDP socket 16-58 16.4.1. 建立UDP 发送端 16-58 16.4.2. 建立UDP 接受端 16-59 17. java5.0的新特性 17-59 17.1. 泛型 ...
11. **网络编程**:Java提供了Socket编程接口,用于创建客户端和服务器端的通信。理解TCP和UDP协议,以及ServerSocket和Socket类的使用。 12. **Java反射机制**:在运行时动态获取类的信息并操作类的对象,为程序...
Java是一种广泛应用于各种领域的面向对象的编程语言,因其跨平台特性而广受欢迎,尤其适合于开发大型企业级应用、Android应用程序以及服务器端程序等。 【压缩包子文件的文件名称列表】: 1. **jdk-8u261-windows-x...
Java提供了丰富的网络编程API,如Socket和ServerSocket类,可以用来创建客户端和服务器端的应用。 10. **Java标准库** Java标准库(Java API)包含了许多预先定义的类和接口,如String类、Date类、Math类等,涵盖...
它具有平台无关性、安全性、可移植性和高性能等优势,广泛应用于服务器端开发、移动应用、大数据处理等多个领域。 2. **Java环境搭建**:学习Java首先需要安装Java Development Kit (JDK),配置好环境变量,包括...
Java Enterprise Edition(EE)则针对企业级应用,提供了服务器端开发的框架和服务。 12. **Java开发工具**:Eclipse、IntelliJ IDEA等集成开发环境(IDE)提供了代码编辑、调试、构建等一站式功能,极大地提高了...
Java Web技术是开发基于Web应用程序的关键工具,它涵盖了从服务器端编程到客户端交互的各种技术。"零点起飞学Java Web随书源代码"是一个学习资源,旨在帮助初学者理解和实践Java Web开发。这本书可能详细讲解了Java ...
Java Servlet是一种服务器端编程技术,用于处理HTTP请求和生成动态网页内容。 现在,lets talk about js。js是一种客户端脚本语言,用于在浏览器端执行脚本,以实现动态网页效果。js技术是基于ECMAScript标准的,...
你将了解如何使用这些技术创建服务器端应用程序,处理HTTP请求,以及如何与数据库进行交互。如果时间允许,课件可能还会延伸到Spring框架,它是Java企业级应用开发的主流框架,涵盖了依赖注入、AOP(面向切面编程)...
【标签】"java2"指的是Java 2平台,即J2SE(Java 2 Standard Edition),它是Java技术的核心,提供了编写桌面应用和服务器端应用的基础。J2SE包含了Java虚拟机(JVM)、类库和开发者工具,使得开发者能够编写出跨...
11. **网络编程**:学习Socket编程,实现客户端与服务器端的通信。 12. **反射机制**:了解如何在运行时动态获取类的信息,创建并操作类的对象。 13. **Java EE基础**:对Java企业版(Java EE)有一个初步认识,...