Sky's blog

我和我追逐的梦

常用链接

统计

其他链接

友情链接

最新评论

linux/unix + RESIN 验证码无法显示的问题

最近公司项目开发中遇到的一个问题,整理一下,和大家分享。

验证码无法显示的问题,验证码的代码就是google上查找到的最常见的代码,服务器采用resin部署于linux或unix。不是常见的out.clear()问题,这次的问题发现在一个我压根就没有想到的地方,profile DISPLAY 环境变量。

1) 问题描述:
登录页面等有验证玛显示的页面,通常可以正确显示验证码图片,但是在某些情况下发现验证码图片无法显示,并且目前只发生在linux/unix平台,windows下正常.而且和resin/jdk版本无关.

bug的直接表现是表现为ie下是红叉,firefox下无实现.将验证码图片的地址在ie输入框中输入,则页面报错:

代码
  1. 500 Servlet Exception  
  2. java.lang.NoClassDefFoundError  
  3.     at java.lang.Class.forName0(Native Method)  
  4.     at java.lang.Class.forName(Class.java:164)  
  5.     at java.awt.GraphicsEnvironment.getLocalGraphicsEnvironment(GraphicsEnvironment.java:68)  
  6.     at java.awt.image.BufferedImage.createGraphics(BufferedImage.java:1141)  
  7.         at com.asiainfo.aimc.wmail.action.CreateImageServlet.doGet(CreateImageServlet.java:104)  

这里的java.lang.NoClassDefFoundError 极其误导人,一直以为是CLASSPATH或者jar包的问题,所以反复检查resin和jdk版本。
始终无法找到问题,只好尝试追查jdk源码,看到底发生了什么。

2) jdk源码追查

调用的servlet:
BufferedImage bi = new BufferedImage(...)
Graphics2D g = bi.createGraphics();

查jdk: BufferedImage.createGraphics():
GraphicsEnvironment env =
GraphicsEnvironment.getLocalGraphicsEnvironment();

再查GraphicsEnvironment.getLocalGraphicsEnvironment:
String nm = (String) java.security.AccessController.doPrivileged
(new sun.security.action.GetPropertyAction
("java.awt.graphicsenv", null));
......
localEnv = GraphicsEnvironment) Class.forName(nm).newInstance();
......
问题应该和nm有关,这里明显是一个类似工厂模式的设计,"java.awt.graphicsenv"到nm 然后Class.forName() 生成GraphicsEnvironment对象。
由于代码在jdk中,不方便修改,因此单独将这些代码提出来到简单的测试类 Test.java:

3) 测试代码分析

代码
  1. public class Test {  
  2.     public static void main(String[] args) {  
  3.         String nm = (String) java.security.AccessController.doPrivileged  
  4.         (new sun.security.action.GetPropertyAction  
  5.          ("java.awt.graphicsenv"null));  
  6.           
  7.         System.out.println(nm);  
  8.           
  9.         try {  
  10.             Class.forName(nm).newInstance();  
  11.         } catch (Throwable e) {  
  12.             System.out.println("error=" + e.getClass().getName());  
  13.               
  14.             e.printStackTrace();  
  15.         }   
  16.     }  
  17. }  

在windows平台下运行,结果正常,打印:
sun.awt.Win32GraphicsEnvironment

将代码放到出问题的resin安装所在的linux平台,手工编译运行:
javac Test.java
java -cp . Test

报错,打印为:

代码
  1. sun.awt.X11GraphicsEnvironment  
  2. Throwable=java.lang.InternalError  
  3. java.lang.InternalError: Can't connect to X11 window server using '10.3.18.16' as the value of the DISPLAY variable.  
  4.         at sun.awt.X11GraphicsEnvironment.initDisplay(Native Method)  
  5.         at sun.awt.X11GraphicsEnvironment.access$000(X11GraphicsEnvironment.java:53)  
  6.         at sun.awt.X11GraphicsEnvironment$1.run(X11GraphicsEnvironment.java:142)  
  7.         at java.security.AccessController.doPrivileged(Native Method)  
  8.         at sun.awt.X11GraphicsEnvironment.<clinit>(X11GraphicsEnvironment.java:131)  
  9.         at java.lang.Class.forName0(Native Method)  
  10.         at java.lang.Class.forName(Class.java:164)  
  11.         at Test.main(Test.java:13)  

从错误信息" Can't connect to X11 window server using '10.3.18.16' as the value of the DISPLAY variable."来看,和DISPLAY环境变量有关
执行unset再运行可以发现问题消失:
$> unset DISPLAY
$> java -cp . Test
sun.awt.X11GraphicsEnvironment
$>

在此情况下(unset DISPLAY )下重新启动resin,发现验证码可以正常显示。

4) 解决的方法:
必须保证resin运行时DISPLAY 环境变量没有设置,如果resin运行的环境有其他要求必须使用DISPLAY,则可以在运行resin前使用unset清除. 建议的简单而有效的方法是直接修改resin/bin/httpd.sh文件,在第二行(具体行数无所谓,但必须在最后一行前)插入:
#! /bin/sh
unset DISPLAY
#....

5)疑惑
1. Can't connect to X11 window server using '10.3.18.16' as the value of the DISPLAY variable
为什么要去连X11 window server ?不懂

2. 从Test.java运行看抛出的是Error : java.lang.InternalError,但是页面上显示的是java.lang.NoClassDefFoundError,看了看源代码也没 有发现先catch 后throws的错误处理,不清楚这里的具体处

posted on 2007-12-05 15:33 sky ao 阅读(2133) 评论(2)  编辑  收藏 所属分类: web

评论

# re: linux/unix + RESIN 验证码无法显示的问题[未登录] 2007-12-05 17:40 sky

在java运行参数上加-Djava.awt.headless=true
即可

后来听说java5以上版本没有问题,没有试过  回复  更多评论   

# re: linux/unix + RESIN 验证码无法显示的问题[未登录] 2007-12-05 21:17 飘然

哦,明天试一下,这样也可以解决啊。

我用的是java5,有这个问题,java6没有试过。  回复  更多评论   


只有注册用户登录后才能发表评论。


网站导航: