在打增量包每次都需要将class文件、jsp文件等拷贝到增量包中比较麻烦。所以就写了一个增量打包工具。
工作原理:根据文件的最后修改时间来打增量。
1、查找Java类增量:根据eclipse工程下的.classpath文件中配置的javasrc目录,来查找修改的java文件,然后将其class文件拷贝到增量目录下。
2、查找jsp文件、配置文件,可以自定义配置。
下面为代码:
XmlReadUtil
package com.aspire.bdc.common.utils; import java.io.File; import java.util.ArrayList; import java.util.List; import java.util.Properties; import org.apache.commons.configuration.ConfigurationException; import org.apache.commons.configuration.XMLConfiguration; import org.apache.log4j.Logger; /** * This class is used to parse an xml configuration file and return specified value. */ public final class XmlReadUtil { private static final String KEY_CONNECTOR = "."; private static final String DATA_FILE_NAME = System.getProperty("user.dir") + "/.classpath"; private static final Logger LOGGER = Logger.getLogger(XmlReadUtil.class); private static XmlReadUtil instance; private XMLConfiguration xmlConfig; private XmlReadUtil() { try { xmlConfig = new XMLConfiguration(DATA_FILE_NAME); } catch (ConfigurationException e) { LOGGER.error(e); throw new RuntimeException(e); } } public static XmlReadUtil getInstance() { if (instance == null) { instance = new XmlReadUtil(); } return instance; } @SuppressWarnings("rawtypes") public List<String> getClasspathEntry() { List lstSrc = xmlConfig.getList("classpathentry[@kind]"); List listPath = xmlConfig.getList("classpathentry[@path]"); List<String> listResult = new ArrayList<String>(); if (listPath != null && !listPath.isEmpty()) { for (int i = 0; i < listPath.size(); i++) { if (lstSrc.get(i).equals("src")) { listResult.add(System.getProperty("user.dir") + File.separator + listPath.get(i)); } } } return listResult; } }
IncremenPublish.java
import java.io.File; import java.text.SimpleDateFormat; import java.util.Date; import java.util.List; import org.apache.commons.io.FileUtils; import org.apache.commons.lang.StringUtils; import org.apache.log4j.Logger; /** * 增量打包工具类 * * @author lipeng * @since 1.0 * @version 2014-8-19 lipeng */ public class IncremenPublish { private static final Logger logger=Logger.getLogger(IncremenPublish.class); private List<String> javaPath; private Date lastDate; private String classPath; private String webPath; private String configPath; private String dbScriptPath; private static final String tempFileDir = "D:\\patch"; private static final SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd"); /** * 构造函数 * * @param lastDate * @param webDirName 存放web应用的目录名称,比如说webapps、webContent等 */ public IncremenPublish(String hour, String webPath, String configPath, String dbScriptPath) { this.javaPath = XmlReadUtil.getInstance().getClasspathEntry(); this.lastDate = parseDate(hour); classPath = IncremenPublish.class.getClass().getResource("/").getPath(); classPath = classPath.substring(1, classPath.length() - 1); if (StringUtils.isNotBlank(webPath)) { this.webPath = webPath; } else { this.webPath = classPath.replace("WEB-INF/classes", ""); } if (StringUtils.isNotBlank(configPath)) { this.configPath = configPath; } else { this.configPath = System.getProperty("user.dir") + File.separator + "config"; } if (StringUtils.isNotBlank(dbScriptPath)) { this.dbScriptPath = dbScriptPath; } else { this.configPath = System.getProperty("user.dir") + File.separator + "db_script"; } } /** * 获取增量文件 * * @Date 2013-11-9 * @author lipeng */ public void getIncremenPublishClassFile() { System.out.println("######################## patch start#####################"); try { File tempFile = new File(tempFileDir); FileUtils.deleteDirectory(tempFile); } catch (Exception e) { e.printStackTrace(); } File file = new File(tempFileDir); if (!file.exists()) { file.mkdirs(); } // 获取class增量 System.out.println("**********start class increment**********"); for (String path : javaPath) { moveIncremenFile(path, "classes", new String(path)); } System.out.println("**********start jsp increment**********"); moveIncremenFile(webPath, null, webPath); System.out.println("**********start config increment*********"); moveIncremenFile(configPath, null, configPath); System.out.println("**********start jsp increment**********"); moveIncremenFile(dbScriptPath, null, dbScriptPath); System.out.println("######################## patch end #####################"); } /** * 获取增量文件 */ public boolean moveIncremenFile(String javaPath, String dirName, String srcPath) { try { File file = new File(javaPath); if(!file.exists()) return false; if (!file.isDirectory()&&!file.getAbsolutePath().contains("vssver2")) { Date fileDate = new Date(file.lastModified()); if (fileDate.getTime() > lastDate.getTime()) { copyFile(srcPath, file, dirName); } } else if (file.isDirectory() && !file.getAbsolutePath().contains("svn") && !file.getAbsolutePath().endsWith("WEB-INF")) { String[] filelist = file.list(); for (int i = 0; i < filelist.length; i++) { File readfile = new File(javaPath + File.separator + filelist[i]); if (!readfile.isDirectory()&&!file.getAbsolutePath().contains("vssver2")) { Date fileDate = new Date(readfile.lastModified()); if (fileDate.getTime() > lastDate.getTime()) { copyFile(srcPath, readfile, dirName); } } else if (readfile.isDirectory()) { moveIncremenFile(javaPath + File.separator + filelist[i], dirName, srcPath); } } } } catch (Exception e) { System.out.println("获取增量文件 Exception:" + e.getMessage()); } return true; } public static void main(String[] args) { String hour = null; String webpath = null; String configPath = null; String dbScriptPath = null; if (args.length > 0) { hour = args[0]; System.out.println("hout:" + hour + "小时"); } if (args.length > 1) { webpath = args[1]; System.out.println("webPath:" + webpath); } if (args.length > 2) { configPath = args[2]; System.out.println("configPath:" + configPath); } if (args.length > 1) { dbScriptPath = args[3]; System.out.println("dbScriptPath:" + dbScriptPath); } IncremenPublish publish = new IncremenPublish(hour, webpath, configPath, dbScriptPath); publish.getIncremenPublishClassFile(); } /** * parseDate * * @Date 2013-11-12 * @author lipeng * @param strDate * @return */ public static Date parseDate(String hour) { Date date = null; if (StringUtils.isBlank(hour) || !hour.matches("[1-9]*")) { try { date = format.parse(format.format(new Date())); } catch (Exception e) { e.printStackTrace(); } } else { date = new Date(new Date().getTime() - 3600 * Integer.parseInt(hour)); } return date; } /** * 移动增量文件 * * @Date 2013-11-12 * @author lipeng * @param file * @param dirName */ private void copyFile(String srcPath, File file, String dirName) { if (dirName == null) { copyJspFile(file); } else { copyClassFile(srcPath, file, dirName); } } /** * 迁移class文件 * * @Date 2013-11-12 * @author lipeng * @param file * @param dirName */ private void copyClassFile(String srcPath, File file, String dirName) { File tempJava = new File(srcPath); String path1 = file.getPath().replace(tempJava.getAbsolutePath(), classPath).replace("java", "class"); File tempFile = new File(path1); String path2 = path1.replace(classPath, tempFileDir + File.separator + dirName); File tempFile1 = new File(path2); tempFile1.getParentFile().mkdirs(); try { FileUtils.copyFile(tempFile, tempFile1); } catch (Exception e) { System.out.println("拷贝class文件出错"); } logger.info("path=" + path2); System.out.println("path=" + path2); } /** * 迁移jsp文件 * * @Date 2013-11-12 * @author lipeng * @param file */ private void copyJspFile(File file) { String path = file.getPath().replace(System.getProperty("user.dir"), tempFileDir); File tempFile = new File(path); tempFile.getParentFile().mkdirs(); try { FileUtils.copyFile(file, tempFile); } catch (Exception e) { System.out.println("拷贝jsp文件出错"); } System.out.println("path=" + path); } }
ant脚本
<?xml version="1.0" encoding="UTF-8"?> <project name="anttest" basedir="." default="run.test"> <!-- 定义一个属性 classes --> <property name="classes" value="func_unit_web/uspc/WEB-INF/classes"> </property> <property name="lib" value="func_unit_web/uspc/WEB-INF/lib"> </property> <target name="init"> <path id="ant.run.lib.path"> <pathelement path="${classes}" /> <fileset dir="${lib}"> <include name="**/*.jar" /> </fileset> </path> </target> <target name="run.test" id="run"> <!--指明要调用的java类的名称 --> <java classname="com.aspire.bdc.common.utils.IncremenPublish" fork="true" failonerror="true"> <!--指明要调用的java类的class路径 --> <classpath refid="ant.run.lib.path"></classpath> <!--时间间隔,如果是3小时,则扫描时只扫描3小时内修改的文件,不配置则扫描当天修改的文件 --> <arg value="" /> <!--webpath --> <arg value=""/> <!--configpath --> <arg value=""/> <!--dbscriptpath --> <arg value=""/> </java> </target> </project>
相关推荐
"JAVA增量包打包工具"就是针对这一需求而设计的专业桌面应用程序,旨在帮助开发者实现自动化、高效的Web项目增量打包和发布。这款工具支持多种版本控制系统,包括Gitee(国内的Git托管平台)、Git和SVN,为多模块...
【svn增量打包小工具】是一种基于Java开发的实用程序,专为IT专业人士设计,用于高效地管理和打包在Subversion(svn)版本控制系统中的代码更改。这个小工具的主要目的是简化版本控制过程,特别是在处理大型项目时,...
本"java增量升级程序.zip"文件主要是基于Subversion(SVN)版本控制系统来实现的增量打包。SVN是一个开源的版本控制系统,用于跟踪项目文件的历史变更,便于团队协作和代码管理。在这个过程中,我们首先需要确保本地...
3. **打包插件**:使用 Eclipse 的打包工具将插件及其依赖打包成一个文件。 4. **安装插件**:将打包好的插件文件通过 Eclipse 的 “Help” 菜单下的 “Install New Software” 功能安装到目标 Eclipse IDE 中。 ##...
5. **本地应用更新**:客户端接收到增量包后,利用这个差异数据来更新本地的APK文件。这个过程可能涉及到合并差异数据到旧APK,或者使用特定的安装器来应用这些差异。 6. **验证更新**:更新完成后,需要验证新版本...
在Java项目中,Maven负责编译源码、运行测试、打包应用,并管理项目的依赖关系。 下面,我们将详细解析整个过程: 1. **创建Shell脚本**:首先,你需要创建一个.sh文件(例如:deploy.sh),并确保其可执行权限(`...
【Ant脚本自动构建SVN增量】是一种高效且节省资源的软件开发实践,它使得开发和测试团队能够自动化处理从代码提交到构建、打包、再到部署的整个过程。使用Ant,一个基于XML的构建工具,可以创建可重复执行的脚本来...
PG数据同步工具(Java实现) 隧道是一个将postgresql的实时数据同步到es或kafka的服务 版本支持 Postgresql 9.4或更高版本 Kafka 0.8或更高版本 ElasticSearch 5.x 架构图 原理 tunnel利用pg内部的逻辑复制功能,...
这个工具的核心功能是帮助开发者进行增量打包,这意味着它只包含自上次发布以来代码库中的更改,而不是整个项目的所有源代码。这种做法大大减少了补丁包的大小,从而加快了分发和应用补丁的过程。 在软件开发中,...
2. **打包增量包**:将差异部分打包成增量更新包,这个包只包含必要的修改,体积相对较小。 3. **下载与验证**:在用户设备上,应用检测到新版本后,下载增量包,并验证其完整性。 4. **应用增量包**:在本地解压...
- 增量更新的基本思路是对比新旧两个版本的APK文件,找出差异部分,然后将这些差异打包成一个较小的文件供用户下载。下载完成后,系统会使用这个差分包对现有应用进行更新。 2. **实现方式**: - 可以使用专门的...
在 Scala 中,SBT(Simple Build Tool)是默认的项目构建工具,它使得构建、管理和打包 Scala 和 Java 项目变得简单易行。本主题将深入探讨 Scala 项目构建工具 SBT,特别是关于版本 0.13.12 的使用。 SBT 0.13.12 ...
UE4.20版本引入了对Gradle的集成,这是一款广泛用于Android应用构建的自动化工具,它使得游戏引擎可以更有效地与Android Studio和其他Android开发环境协同工作。以下是关于UE4.20中.gradle本地缓存文件及其相关知识...
IntelliJ IDEA是一款广受欢迎的Java集成开发环境(IDE),...out目录则是IDE编译生成的输出目录,包含了编译后的类文件和打包的资源。在使用插件前,确保这些文件都在正确的位置,并且配置正确,才能保证插件正常工作。
Gradle是一款强大的构建自动化工具,尤其在Java、Android开发领域广泛应用。版本4.2.1是Gradle在2017年发布的一个稳定版本,它提供了许多改进和新特性,旨在提升开发者的工作效率和构建速度。 1. **Gradle 4.2.1的...
- **Gradle概念介绍**:Gradle是一款基于Java的开源自动化构建工具,它通过一种称为Groovy的语言定义构建逻辑,提供了一种灵活的方式来管理项目的构建过程。 - **Gradle在Android中的作用**:Gradle在Android开发...
Docker允许我们将应用及其依赖打包成独立的容器,这样就可以在任何兼容Docker的环境中一致地运行,无论是在本地开发机还是远程服务器上。通过Docker,FlowCI可以轻松地创建和管理构建环境,确保每次构建都在干净、...
4. **同步策略**:根据需求设置同步规则,例如增量同步(仅更新变化的部分)、全量同步(定期刷新整个目录)或实时同步(任何更改立即反映到LDAP)。 5. **测试与监控**:在实际部署前进行测试,确保数据正确无误地...
- **Maven**: Maven是一个项目管理和自动化构建工具,它使用一个叫做POM(项目对象模型)的XML文件来配置项目的构建环境和构建指令。 - **groupId**: 表示组织或者项目的唯一标识符,通常是公司或者组织的域名反写。...