精简过的JRE一般都不是通用的,都是针对自己的应用程序把不需要的类去掉,从而达到精简目的。所以有些人想拿别人精简过的JRE运行自己的应用程序,这样是不靠谱的。最后还是得自己动手精简,过程也挺简单。


思路:

把自己的应用程序打包成jar,然后通过jar命令运行这个jar,把jar所需的类全部打印到一个文本文件里面,把文本文件里面的类提取出来,重新打包。覆盖JRE目录中的JAR包。去掉JRE下bin目录和lib目录中不需要得。瘦身完成。

 

应用程序打包后是test.jar,jre(目录) 也和它同一文件夹,写一个test.bat脚本方便测试:

@echo off
java -jar -verbose:class test.jar >>class.txt
@pause



程序运行后,最好把所有的功能使用一遍,这样输出了一个文件class.txt,里面有所有需要的class,其格式如下:

[Opened D:\data\dict\jre\lib\rt.jar]
[Loaded java.lang.Object from D:\data\dict\jre\lib\rt.jar]
[Loaded java.io.Serializable from D:\data\dict\jre\lib\rt.jar]
[Loaded java.lang.Comparable from D:\data\dict\jre\lib\rt.jar]
[Loaded java.lang.CharSequence from D:\data\dict\jre\lib\rt.jar]
....



总共会有3个jar包用到,/jre7/lib/ext/localedata.jar,/jre7/lib/rt.jar,/jre7/lib/charsets.jar, 把

class.txt 分拆成localedata.txt,rt.txt,charsets.txt,在notepad中进行一些处理,去掉所有不是rt.jar中的class的行(搜索,标记,删除书签行),去掉from后面的( from.+?$),去掉loaded等无关项目,再把“.”替换成“/”.这个可以利用正则表达式等轻松处理。

处理完后得到的文件类似如下格式:

 rt.txt

java/lang/Object
java/io/Serializable
java/lang/Comparable
java/lang/CharSequence
java/lang/String



我们再依照这个文件来裁剪charsets.jar,localedata.jar,

然后写一个程序处理,从rt或charsets中把需要的的class拷贝到另一个对应的文件夹rt1或charsets1。在运行下面JAVA程序之前需要将JRE目录中localedata.jar,rt.jar,charsets.jar分别解压到相应目录。


所需的类都抽取到rt1目录之后,把原rt目录中的“META-INF”文件夹拷贝到rt1下面,进入rt1目录,用rar压缩工具打包成rt.zip,改名为rt.jar,然后替换jre6/lib目录下的rt.jar。

charsets1目录处理同上。

 

package test;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.LineNumberReader;

public class ReduceRt {
// 文件拷贝
public static boolean copy(String file1, String file2) {
try // must try and catch,otherwide will compile error
{
// instance the File as file_in and file_out
java.io.File file_in = new java.io.File(file1);
java.io.File file_out = new java.io.File(file2);
FileInputStream in1 = new FileInputStream(file_in);
FileOutputStream out1 = new FileOutputStream(file_out);
byte[] bytes = new byte[1024];
int c;
while ((c = in1.read(bytes)) != -1)
out1.write(bytes, 0, c);
in1.close();
out1.close();
return (true); // if success then return true
} catch (Exception e) {
e.printStackTrace();
// System.out.println("Error!");
return (false); // if fail then return false
}
}

// 读取路径,copy
public static int dealClass(String needfile, String sdir, String odir)
throws IOException {
int sn = 0; // 成功个数

File usedclass = new File(needfile);
if (usedclass.canRead()) {
String line = null;
LineNumberReader reader = new LineNumberReader(
new InputStreamReader(new FileInputStream(usedclass),
"UTF-8"));
while ((line = reader.readLine()) != null) {
line = line.trim();
int dirpos = line.lastIndexOf("/");
if (dirpos > 0) {
String dir = odir + line.substring(0, dirpos);
File fdir = new File(dir);
if (!fdir.exists())
fdir.mkdirs();
String sf = sdir + line + ".class";
String of = odir + line + ".class";
boolean copy_ok = copy(sf.trim(), of.trim());
if (copy_ok)
sn++;
else {
System.out.println(line);
}
}
}
}
return sn;

}

public static void main(String[] args) {
String needfile = "E:/fishing/3.program/server/trunk/test/src/test/localedata.txt";// 运行JAR生成的,应用程序所需类的txt文件
String sdir = "E:/fishing/3.program/server/trunk/test/res/ext/localedata/"; // rt.jar解压后的目录
String odir = "E:/fishing/3.program/server/trunk/test/res/ext/localedata1/";// 抽取的类存放目录
try {
int sn = dealClass(needfile, sdir, odir);
System.out.print(sn);

} catch (IOException e) {
e.printStackTrace();
}
}
}



所需类的精简工作已经完成,接下来精简其他的。

 

1、Jre目录下的license都删除,只留bin和lib目录

2、bin下的执行文件需要运行jar时用DLL_Killer工具查看使用到了哪些文件

3、lib下只要保留 rt,charsets库就可以了(因为应用程序只用到了这两个)。

4、除了i386和zi两个子目录外,其余的子目录都可以不要(原则上都要自己试试看删除其他目录会不会报错)。

5、Zi下只需要保留自己地区的子目录和其下的一些文件就可以(这里Zi下我只保留了America文件夹)。

6、Lib下除了库之外的属性文件都要保留。

抽取出的类重新打包成rt.jar时要注意,用JAR命令和rar工具打的jar包都不行。解决如下:
 -将原生的rt.jar用rar打开,然后进入相关目录,删除掉相关目录或者文件,再把抽取出来的类拖进来就行了。

 

优化完成!