java -uf_Java如何快速修改Jar包里的文件内容

需求背景:写了一个实时读取日志文件以及监控的小程序,打包成了Jar包可执行文件,通过我们的web主系统上传到各个服务器,然后调用ssh命令执行。每次上传前都要通过解压缩软件修改或者替换里面的配置文件,这样感觉有点麻烦,就想办法能不能通过程序动态生成配置文件,然后修改或者替换掉Jar包里的配置文件,最后再上传到各个服务器去执行。

实现历程:刚开始看了大量文章,整理出来了一个操作Jar包的工具类,用工具类里面的方法去修改一个30M左右的Jar包文件时,发现耗时竟然要7秒,而且修改Jar文件的方法确实有点复杂(这个可能需要开发JDK的专业人士提供简单、高效的方法了),果断忍受不了,然后想能不能通过其他的方法来实现。想到了一个方案,就是不通过java去修改Jar里的文件,而是用linux的Jar命令(当然也是通过java去调用linux命令),先解压缩jar包,然后将动态生成的配置文件替换解压缩包里的配置文件,最后再将目录压缩成Jar文件。经过测试,这样大概需要4秒的时间,确实快了不少。想了很久也没有想到其他更好的办法,正打算实施的时候,看到了这条命令:jar uf test.jar manifest.mf  -u 更新已存在的 JAR 文件包 (添加文件到 JAR 文件包中)  -f 指定 JAR 文件名;瞬间来了灵感,直接调用这条命令不就OK了!通过这条命令几秒钟就轻轻松松搞定了,简直是山穷水尽疑无路,柳暗花明又一村。

下面是我整理的操作Jar包的工具类(已经通过测试,可以拿去直接用),包括读取Jar包的文件内容,遍历 Jar包,修改 Jar包文件内容等操作,供大家参考。

8f900a89c6347c561fdf2122f13be562.png

961ddebeb323a10fe0623af514929fc1.png

packagecom.agent.util;importjava.io.BufferedReader;importjava.io.ByteArrayOutputStream;importjava.io.FileNotFoundException;importjava.io.FileOutputStream;importjava.io.IOException;importjava.io.InputStream;importjava.io.InputStreamReader;importjava.util.Enumeration;importjava.util.Iterator;importjava.util.Map;importjava.util.Set;importjava.util.TreeMap;importjava.util.jar.JarEntry;importjava.util.jar.JarFile;importjava.util.jar.JarOutputStream;public classJarUtil {/*** 读取jar包所有的文件内容,显示JAR文件内容列表

*@paramjarFileName

*@throwsIOException*/

public static void readJARList(String jarFilePath) throwsIOException {//创建JAR文件对象

JarFile jarFile = newJarFile(jarFilePath);//枚举获得JAR文件内的实体,即相对路径

Enumeration en =jarFile.entries();

System.out.println("文件名\t文件大小\t压缩后的大小");//遍历显示JAR文件中的内容信息

while(en.hasMoreElements()) {//调用方法显示内容

process(en.nextElement());

}

}//显示对象信息

private static voidprocess(Object obj) {//对象转化成Jar对象

JarEntry entry =(JarEntry) obj;//文件名称

String name =entry.getName();//文件大小

long size =entry.getSize();//压缩后的大小

long compressedSize =entry.getCompressedSize();

System.out.println(name+ "\t" + size + "\t" +compressedSize);

}/*** 读取jar包里面指定文件的内容

*@paramjarFileName jar包文件路径

*@paramfileName 文件名

*@throwsIOException*/

public static void readJarFile(String jarFilePath,String fileName) throwsIOException{

JarFile jarFile= newJarFile(jarFilePath);

JarEntry entry=jarFile.getJarEntry(fileName);

InputStream input=jarFile.getInputStream(entry);

readFile(input);

jarFile.close();

}public static void readFile(InputStream input) throwsIOException{

InputStreamReader in= newInputStreamReader(input);

BufferedReader reader= newBufferedReader(in);

String line ;while((line = reader.readLine())!=null){

System.out.println(line);

}

reader.close();

}/*** 读取流

*

*@paraminStream

*@return字节数组

*@throwsException*/

public static byte[] readStream(InputStream inStream) throwsException {

ByteArrayOutputStream outSteam= newByteArrayOutputStream();byte[] buffer = new byte[1024];int len = -1;while ((len = inStream.read(buffer)) != -1) {

outSteam.write(buffer,0, len);

}

outSteam.close();

inStream.close();returnoutSteam.toByteArray();

}/*** 修改Jar包里的文件或者添加文件

*@paramjarFile jar包路径

*@paramentryName 要写的文件名

*@paramdata 文件内容

*@throwsException*/

public static void writeJarFile(String jarFilePath,String entryName,byte[] data) throwsException{//1、首先将原Jar包里的所有内容读取到内存里,用TreeMap保存

JarFile jarFile = newJarFile(jarFilePath);//可以保持排列的顺序,所以用TreeMap 而不用HashMap

TreeMap tm = newTreeMap();

Enumeration es=jarFile.entries();while(es.hasMoreElements()){

JarEntry je=(JarEntry)es.nextElement();byte[] b =readStream(jarFile.getInputStream(je));

tm.put(je.getName(),b);

}

JarOutputStream jos= new JarOutputStream(newFileOutputStream(jarFilePath));

Iterator it=tm.entrySet().iterator();boolean has = false;//2、将TreeMap重新写到原jar里,如果TreeMap里已经有entryName文件那么覆盖,否则在最后添加

while(it.hasNext()){

Map.Entry item=(Map.Entry) it.next();

String name=(String)item.getKey();

JarEntry entry= newJarEntry(name);

jos.putNextEntry(entry);byte[] temp ;if(name.equals(entryName)){//覆盖

temp =data;

has= true;

}else{

temp= (byte[])item.getValue();

}

jos.write(temp,0, temp.length);

}if(!has){//最后添加

JarEntry newEntry = newJarEntry(entryName);

jos.putNextEntry(newEntry);

jos.write(data,0, data.length);

}

jos.finish();

jos.close();

}/*** 测试案例

*@paramargs

*@throwsException*/

public static void main(String args[]) throwsException{// readJarFile("D:\\esjavaclient-0.0.1-SNAPSHOT.jar","es-info.properties");

String data= "helloBabydsafsadfasdfsdafsdgasdgweqtqwegtqwfwefasdfasfadfasf";long start =System.currentTimeMillis();

writeJarFile("D:\\esjavaclient-0.0.1-SNAPSHOT.jar","es-info.properties",data.getBytes());long end =System.currentTimeMillis();

System.out.println(end-start);

readJarFile("D:\\esjavaclient-0.0.1-SNAPSHOT.jar","es-info.properties");

}

}

View Code

上面有个测试案例,测试读取其中一个配置文件,然后修改这个文件,最后再读取出来。 一个30M左右的Jar包文件耗时7秒多,实在忍受不了。网上找了很多文章,同时看了Zip相关的源码,也木有找到可以直接修改Jar包文件内容的方法。 最后只能另辟蹊径,找到了一个执行效率非常快的方法。

关于工具类没有删除文件的方法,大家根据修改的方法稍微改造一下即可。

工具类参考的博客:

如何快速修改Jar包里的文件内容:

贴出来我巧妙实现的代码:(如果不知道Java 如何调用linux命令的同学,请先补习一下这方面的知识)

String cmd = "jar uf esjavaclient-0.0.1-SNAPSHOT.jar config.properties";

String[] cmds = {"/bin/sh","-c",cmd};

Process pro;

try{

pro =Runtime.getRuntime().exec(cmds);

pro.waitFor();

InputStream in =pro.getInputStream();

BufferedReader read = new BufferedReader(newInputStreamReader(in));

String line = null;

while((line = read.readLine())!=null){

System.out.println(line);

}

} catch(Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

通过执行这条条命令,就可以很快将我们生成的配置文件config.properties覆盖掉Jar里的文件,从而达到修改的目的。

总结:如果Jar包里的文件不大的话,完全可以用工具类提供的方法去操作Jar包。比较大的话,修改Jar包的方法,推荐用我那个巧妙的方法。

最后:给开发JDK的专业人士提点建议,关于操作压缩包这方面的API还不够强大,希望后续可以完善一下

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/505582.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java .vm h2_java-H2服务器在调试时挂起

由于正在创建内存数据库,因此启动tcp服务器将无济于事.我建议改为在线程中启动控制台,并在同一段代码(例如,使用jdbc)中打开与此数据库的连接,但不要关闭/释放它.使用此代码段执行此操作:请根据H2文档添加其他选项,例如允许其他人使用(我建议暂时将其保留)org.h2.to…

java 静态变量 new_java中静态对象和普通变量在初始化静态变量的时候有什么区别??高手!!...

下面有一个例子,将语句(6)直接改为一个新的对象后,结果会不同,解释的清楚一些吗??豁出去了,家当10分publicclassStaticVariableTest{privatestaticStaticVariableTestsvtnewS...下面有一个例子,…

java子类怎么编译_java – 无法编译从基类实现抽象方法的子类

编译我已经定义的基类的子类有一个问题,它有一个单独的方法,而每个子类都实现了抽象基类方法,但是javac说他们甚至没有在子类中明确定义它们.DbModel.java(基类)package com.manodestra.db;import java.sql.ResultSet;import java.sql.SQLException;public abstract class DbMo…

java循环遍历类属性_java循环遍历类属性 get 和set值方法

//遍历sqspb类 成员为String类型 属性为空的全部替换为“/”Field[] fields sqspb.getClass().getDeclaredFields();for (int i 0; i < fields.length; i) {// 获取属性的名字String name fields[i].getName();// 将属性的首字符大写&#xff0c;方便构造get&#xff0c;…

java 浏览器 爬虫_java 网络编程-爬虫+模拟浏览器

网络爬虫模拟浏览器(获取有权限网站资源)&#xff1a;获取URL下载资源分析处理public class http {public static void main(String[]args) throws Exception{//https更安全//URL.openStream()打开于URL的连接&#xff0c;并返回一个InputStream用于从连接中读取数据//获取URLU…

java 序列化实例_Java中的序列化与反序列化实例

创建的字节流与平台无关。因此&#xff0c;在一个平台上序列化的对象可以在另一个平台上反序列化。为了使Java对象可序列化&#xff0c;我们实现java.io.Serializable可序列化接口。ObjectOutputStream类包含writeObject()序列化对象的方法。public final voidwriteObject(Obje…

java map for循环遍历_java中Map遍历的四种方式

java中Map遍历的四种方式在java中所有的map都实现了Map接口&#xff0c;因此所有的Map(如HashMap, TreeMap, LinkedHashMap, Hashtable等)都可以用以下的方式去遍历。方法一&#xff1a;在for循环中使用entries实现Map的遍历&#xff1a;/*** 最常见也是大多数情况下用的最多的…

java 代码 _程序员用1.5小时写出的Java代码,让同事瞠目结舌!直呼优秀

1.曾经不止一次在生产中见过类似这样的代码&#xff1a;这有很多变种&#xff0c;例如用 Integer.valueOf(1)、 (Integer)1 之类的&#xff0c;那些细节都不重要。重要的是&#xff1a;凭空用一个 Integer 对象作为锁对象。2.AbstractComponentBuilderTemplateFactory3.HelloWo…

java保存文件到linux指定目录_怎么使用java编程实现linux下全部文件目录的遍历

为了避免目录列举消耗时间过长,请指定一个目录来模拟,命令行参数:代表路径的字符串.如果认可代码,请加分50,谢谢----import javax.swing.*;import javax.swing.tree.*;import java.awt.*;import java.io.*;final public class FileTree extends JFrame {public FileTree(File d…

avro 序列化java_Avro 对象序列化与反序列化,及转Json对象序列化处理

Avro 工具类 序列化与反序列化public class AvroHelper {public byte[]serializeAvroToByteArray(List dcs) {try {ByteArrayOutputStream baos new ByteArrayOutputStream();DatumWriter pictureDatumWriter new SpecificDatumWriter();DataFileWriter dataFileWriter new Da…

java监听剪贴板_在java中实现windows剪贴板监视

在google上搜了一把&#xff0c;结果搜到了自己原创的文章&#xff0c;不胜窃喜。于是重新格式化了一下&#xff0c;重发之。 package com.ibm.bloghelper; import java.awt.datatransfer.Clipboard; import java.awt.datatransfer.DataFlavor; import java.awt.datatransfer.F…

mysql workbench 从model建库_使用MySQL Workbench进行数据库设计——MySQL Workbench用法总结...

转载请注明出处&#xff1a;http://blog.csdn.net/dongdong9223/article/details/48318877 本文出自【我是干勾鱼的博客】 1 简单介绍MySQL Workbench是一款专为MySQL设计的ER/数据库建模工具。它是著名的数据库设计工具DBDesigner4的继任者。你能够用MySQL Workbench设计和创…

mysql5.7 hibenate5.1_5.7 Spring与Hibernate整合应用

下面以一个简单的实例说明Spring与Hibernate的整合策略&#xff0c;步骤如下。1 在SQL Server 2005中创建数据库表数据库名为XSCJ&#xff0c;表见附录A的登录表。2 创建Web项目命名为“Hibernate_Spring”3 添加Spring的开发能力右击项目名&#xff0c;选择【MyEclipse】→【A…

java 多进程多线程_Java并发编程原理与实战三:多线程与多进程的联系以及上下文切换所导致资源浪费问题...

一、进程考虑一个场景&#xff1a;浏览器&#xff0c;网易云音乐以及notepad 三个软件只能顺序执行是怎样一种场景呢&#xff1f;另外&#xff0c;假如有两个程序A和B&#xff0c;程序A在执行到一半的过程中&#xff0c;需要读取大量的数据输入(I/O操作)&#xff0c;而此时CPU只…

mysql适合什么阵列_如何选择最合适的RAID级别

众所周知&#xff0c;最常用的RAID配置等级分别是RAID-0、RAID-1、RAID-5。这三种RAID等级针对数据的传输速度和保护程度都有所不同:RAID-0(数据 条带化存储阵列)旨在提供速度&#xff0c;在所有RAID中速度最快&#xff0c;但是提供的保护最少;RAID-1(透明或条带化存储镜像)这种…

python写if语句_python if语句

## Python条件和If语句Python支持数学中通常的逻辑条件&#xff1a;* 等于&#xff1a;a b* 不等于&#xff1a;a&#xff01; b* 小于&#xff1a;a * 小于或等于&#xff1a;a < b* 大于&#xff1a;a> b* 大于或等于&#xff1a;a> b这些条件可以几种方式使用&am…

python用法查询笔记_Python爬虫学习笔记(三)

handler处理器自定义 - Cookies && URLError && json简单使用Cookies&#xff1a;以抓取https://www.yaozh.com/为例Test1(不使用cookies)&#xff1a;代码&#xff1a;import urllib.request# 1.添加URLurl "https://www.yaozh.com/"# 2.添加请求头…

java 线程 交给spring_浅谈Java中spring 线程异步执行

多线程并发处理起来通常比较麻烦&#xff0c;如果你使用spring容器来管理业务bean&#xff0c;事情就好办了多了。spring封装了Java的多线程的实现&#xff0c;你只需要关注于并发事物的流程以及一些并发负载量等特性&#xff0c;具体来说如何使用spring来处理并发事务&#xf…

java编程实现素数环_结对编程(JAVA实现)

项目成员&#xff1a;黄思扬(3117004657)、刘嘉媚(3217004685)二、PSP表格PSPPersonal Software Process Stages预估耗时(分钟)实际耗时(分钟)Planning计划6040 Estimate 估计这个任务需要多少时间6040Development开发14401505 Analysis 需求分析3015 Design Spec 生成设计文档…

自学java编译老是出错_为什么按照书上的代码,编译老是出错?

老是又小伙伴在群里说我的代码是按照书上的代码敲的&#xff0c;就是编译不过&#xff1f;想不通呀&#xff01;目前市面上的一些书都是十来年以前编写的了&#xff0c;你不幸看的还是这些书&#xff0c;错误原因就在这里。比如谭浩强的C语言程序设计这个本书都出了N版了&#…