通过DOS、SHELL批处理命令加载Lib并编译和打包Java项目(或者运行项目)
有些时候,需要通过DOS批处理来编译整个项目的JAVA文件;并且编译后还要对Class文件进行打包成jar文件。。。
这还不是最烦的,最烦的是,编译和打包的时候需要依赖其他多个jar文件,困难就这么来了。。。。
项目结构
App
|-dest
| |-libs(copy过来的依赖库)
| |-bin(生成的class文件)
|
|-libs(依赖库)
|-src(需要编译的JAVA源文件)
如果需要对项目编译,首先,需要清理dest文件夹
REM 删除dest包
IF EXIST dest RMDIR /s /q dest
REM 创建 dest/libs和dest/bin文件夹
mkdir "dest/libs"
mkdir "dest/bin"
REM copy依赖库
copy libs\*.jar dest\libs\ /y
然后,从JAVA的编译命令用法入手:
javac <选项> <源文件>
其参数中,"-d <目录>",指定存放生成的类文件的位置
"-cp <路径>",指定查找用户类文件和注释处理程序的位置(个人理解就是所需要依赖的jar包)
"-encoding <编码>", 指定源文件使用的字符编码(我们使用UTF-8)
例如: javac -d dest\bin -cp libs\commons-logging-1.1.1.jar;freemarker-2.3.19.jar; -encoding UTF-8 src/a.java src/com.mycls/b.java src/com.mycls/c.java
上面Demo是加载其依赖库并使用UTF-8编码编译指定java文件
继续,对编译好的class文件进行打成jar包
REM 需要在class的根目录打包
cd dest/bin
REM 将当前目录下所有文件打包到上级目录的myjar.jar文件中
jar -cvf ../myjar.jar *
当然了,还可以通过对MANIFEST.MF的定制来指定默认运行的类
上面的内容,对于一个JAVA开发来说都不是事儿,接下来问题来了。。。。
在编译的时候,javac加载依赖库是指定的依赖库,那如果有很多依赖库,并且在开发过成功还会发生变动呢?难道要修改cmd(bat)批处理?
这个时候,就需要指定一个libs目录,加载此目录下的所有文件即可。可是怎么拼接这个参数呢?
如下所示(如果没有依赖的jar,可将cp相关部分去除):
SETLOCAL enableDelayedExpansion
FOR %%i IN ("libs\*.jar") DO SET CP=%%i;!CP!
SET OPT=-d %OUTPUT_PATH%\bin -cp %CP% -encoding utf-8
EndLocal
这样,所有的参数就形成了,
SETLOCAL enableDelayedExpansion是扩展本地环境变量延迟,是获取CP参数的一个出错点,如果没有这个,for里面的赋值语句是无效的。
OUTPUT_PATH 变量是指定编译后的class文件的位置,
CP 变量就是所有需要加载用的lib包
同理,循环src目录下的所有文件和目录并拼接所有的java文件(配置文件请自行处理),可以获取所有的java文件:
CD %SRC_PATH%
FOR /R %%b IN ( . ) DO (
IF EXIST %%b/*.java SET JFILES=!JFILES! %%b/*.java
)
CD ..
REM 正在编译...
javac %OPT% %JFILES%
REM 编译结束
然后就是打包了,
CD %OUTPUT_PATH%\bin
REM 将所有文件打包为mylib.jar并存放到父目录
jar -cvf ../mylib.jar *
REM 打包完毕
对于SHELL命令来说,这样的批处理更为简单:
#获取所有的依赖lib
export libs=`find libs -name "*.jar" |xargs|sed "s/ /:/g"`
#获取所有需要编译的java文件
export jfiles=`find src -name "*.java" |xargs|sed "s/ / /g"`
#准备编译
opt="-d $OUTPUT_PATH/bin -cp ${libs} -encoding utf-8"
javac $opt ${jfiles}
#打包
cd ${OUTPUT_PATH}/bin
jar -cvf ${CUR}/dest/mylib.jar *
cd ${CUR}/src
jar -cvf ${CUR}/dest/mylib-source.jar *
以下图目录结构为例:
里面的compile.sh和compile.cmd文件的内容分别为:
调用脚本的JAVA代码:
- package test.compile;
- import java.io.File;
- import java.text.SimpleDateFormat;
- /**
- * <p>编译abc目录下所有java文件</p>
- * @author Ares
- * @version $Id: CompileTool.java, v 0.1 2015年02月26日 上午10:41:32 Exp $
- */
- public class CompileTool {
- // DOS: 第一个参数是 src、cmd的父目录,第二个参数为最终压缩包(mylib.jar,mylib-source.jar的目录)所在目录
- private static String os_name = System.getProperty("os.name").toLowerCase();
- public static boolean compileAndJar(final String jarRootPath) throws Exception{
- String app_path = jarRootPath; // app_path是指定abc的绝对路径,因为java文件执行目录可能跟编译目录不一致,所以如果需要请使用此参数(脚本中第一步需要进入此目录)
- if(app_path !=null && (!app_path.endsWith("/") && !app_path.endsWith(File.separator)))
- app_path += File.separator;
- String compile_datetime = new SimpleDateFormat("yyyyMMddHHmmss").format(new java.util.Date());
- // 无后缀的jar名称(cmd 会拼写为 jarname.jar 和 jarname-source.jar)
- String jar_name = "myjar_" + compile_datetime; // jar名称格式,mylib_日期时间,以参数的形式传入,下面的脚本中未使用,如若使用,请替代脚本中的mylib
- String cmd = "";
- java.io.File cmdFile = null;
- if(os_name.indexOf("windows") > -1){
- cmd = "cmd.exe /C start " + app_path + "compile.exe " + app_path + " " + jar_name;
- cmdFile = new java.io.File(app_path + "compile.exe");
- }else {
- cmd = app_path + "compile.sh " + app_path + " " + jar_name;
- cmdFile = new java.io.File(app_path + "compile.sh");
- }
- if(!cmd.equals("") && cmdFile != null && cmdFile.exists()){
- if(!cmdFile.canExecute()) cmdFile.setExecutable(true);
- return callCmd(cmd);
- }else {
- throw new Exception("can't find compile command. command:"+cmdFile.getAbsolutePath());
- }
- }
- private static boolean callCmd(String cmdStr) {
- try {
- Process child = Runtime.getRuntime().exec(cmdStr);
- java.io.InputStream in = child.getInputStream();
- while (in.read() != -1) {}
- in.close();
- try {
- child.waitFor();
- return true;
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- return false;
- } catch (java.io.IOException e) {
- e.printStackTrace();
- }
- return false;
- }
- }
下面是compile.sh文件内容:
- #!/bin/bash
- CUR=$(pwd)
- ## reset dest folder
- OUTPUT_PATH=${CUR}/dest
- rm -rf $OUTPUT_PATH
- mkdir -p ${OUTPUT_PATH}/bin
- mkdir -p ${OUTPUT_PATH}/libs
- ## 获取依赖库和需要编译的java文件
- export libs=`find libs -name "*.jar" |xargs|sed "s/ /:/g"`
- export jfiles=`find src -name "*.java" |xargs|sed "s/ / /g"`
- ## 编译
- opt="-d $OUTPUT_PATH/bin -cp ${libs} -encoding utf-8"
- javac $opt ${jfiles}
- ## 打包
- cd ${OUTPUT_PATH}/bin
- jar -cvf ${CUR}/dest/mylib.jar *
- cd ${CUR}/src
- jar -cvf ${CUR}/dest/mylib-source.jar *
下面是compile.cmd文件内容:
- @ECHO OFF
- SETLOCAL enableDelayedExpansion
- REM Save Current Path(保存当前路径)
- SET CUR=%CD%
- REM Dir Structure(设置目录结构)
- SET OUTPUT_PATH=dest
- SET SRC_PATH=%CUR%\src
- SET LIBS_PATH=%CUR%\libs
- REM Reset Output Dir(重置清空输出目录)
- IF EXIST %OUTPUT_PATH% RMDIR /s /q %OUTPUT_PATH%
- IF NOT EXIST %OUTPUT_PATH% MKDIR %OUTPUT_PATH%
- CD %OUTPUT_PATH%
- IF NOT EXIST bin MKDIR bin
- IF NOT EXIST libs MKDIR libs
- REM Copy Files(复制jar文件)
- CD %CUR%
- SET CP=%OUTPUT_PATH%/libs
- copy %LIBS_PATH%\*.jar %OUTPUT_PATH%\libs\ /y
- CD %CUR%
- FOR %%i IN ("libs\*.jar") DO SET CP=%%i;!CP!
- SET OPT=-d %OUTPUT_PATH%\bin -cp %CP% -encoding utf-8
- CD %SRC_PATH%
- REM 获取所有需要编译的java源文件,并用空格链接
- REM 此处需要开启变量延迟(SETLOCAL enableDelayedExpansion不可缺少)
- FOR /R %%b IN ( . ) DO (
- IF EXIST %%b/*.java SET JFILES=!JFILES! %%b/*.java
- )
- CD %CUR%
- REM 正在编译...
- javac %OPT% %JFILES%
- IF "%errorlevel%" EQU "0" (
- ECHO 编译成功,正在拷贝资源文件...
- copy %CUR%/src/*.* %OUTPUT_PATH%\src\ /y
- )
- REM 准备打包编译文件和JAVA源文件
- CD %OUTPUT_PATH%/bin
- jar -cvf ../mylib.jar *
- cd %SRC_PATH%
- jar -cvf %CUR%/dest/mylib-source.jar *
- EndLocal
- exit
通过DOS、SHELL批处理命令加载Lib并编译和打包Java项目(或者运行项目)的更多相关文章
- 用UBOOT自带loadb命令加载应用程序到SDRAM中运行的方法
S3C44B0开发板中,用UBOOT自带loadb命令加载应用程序到SDRAM中运行的方法 1.开发板说明: 开发板上已有移植好的UBOOT运行. 2.交叉编译工具链为arm-linu-g ...
- VC中加载LIB库文件的三种方法
VC中加载LIB库文件的三种方法 在VC中加载LIB文件的三种方法如下: 方法1:LIB文件直接加入到工程文件列表中 在VC中打开File View一页,选中工程名,单击鼠标右键,然后选中&quo ...
- 使用composer命令加载vendor中的第三方类库
1.首先下载需要的第三方类库,放在TP框架下的vendor文件夹下 2.给第三方类库SDK写上命名空间,命名空间为该类库的文件夹对应的名字 3.在composer.json文件中添加需要加载的第三方类 ...
- SuperSocket命令加载器 (Command Loader)
在某些情况下,你可能希望通过直接的方式来加载命令,而不是通过自动的反射. 如果是这样,你可以实现你自己的命令加载器 (Command Loader): public interface IComman ...
- SuperSocket命令和命令加载器
关键字: 命令, 命令加载器, 多命令程序集 命令 (Command) SuperSocket 中的命令设计出来是为了处理来自客户端的请求的, 它在业务逻辑处理之中起到了很重要的作用. 命令类必须实现 ...
- 【C++】C++程序加载lib静态库
使用Visual Studio 编写C++程序有几种配置lib的方法,以下是在代码中加载lib文件的方法: 在项目所在目录下创建文件夹lib,将lib文件此路径下,包括Debug和Release两种模 ...
- VC6.0加载lib文件的三种方法
MFC编写程序,都要用到动态链接库,MFC相关的动态库有MFCD42和MFC42等,MFC框架程序已经自动加载,那么如何引入第三方的动态链接库到工程中呢? 静态链接库是要先把程序中所需要使用的函数编译 ...
- 脚本命令加载外部配置文件夹conf
加载log4j配置文件 Log4iConfigurer类 public class Log4iConfigurer { private static boolean INITIALIZED = fal ...
- maven打包时无法加载lib下的jar
© 版权声明:本文为博主原创文章,转载请注明出处 问题描述: 项目在本地部署没有问题,但是使用maven打包时报错: ***(引用jar中某个类的的路径) 不存在 ***(某个java类中的某行某列) ...
随机推荐
- 曲演杂坛--EXISTS语句
通常在我写EXISTS语句时,我会写成IF EXISTS(SELECT TOP(1) 1 FROM XXX),也没细细考究过为什么要这么写,只是隐约认为这样写没有啥问题,那今天就深究下吧! 首先准备测 ...
- [C++] 行程编码C++代码
#include<iostream> #include<string.h> #include<string> #include<cstring> #in ...
- Linux服务器Cache占用过多内存导致系统内存不足问题的排查解决(续)
作者: 大圆那些事 | 文章可以转载,请以超链接形式标明文章原始出处和作者信息 网址: http://www.cnblogs.com/panfeng412/archive/2013/12/17/dro ...
- AngularJS快速入门指南01:导言
AngularJS使用新的attributes扩展了HTML AngularJS对单页面应用的支持非常好(SPAs) AngularJS非常容易学习 现在就开始学习AngularJS吧! 关于本指南 ...
- JavaScript原生DOM操作API总结
最近面试的时候被这个问题给卡了,所以抽时间好好复习一下. 原文:http://www.cnblogs.com/liuxianan/p/javascript-dom-api.html 几种对象 Node ...
- 固定表头/锁定前几列的代码参考[JS篇]
引语:做有难度的事情,才是成长最快的时候.前段时间,接了一个公司的稍微大点的项目,急着赶进度,本人又没有独立带过队,因此,把自己给搞懵逼了.总是没有多余的时间来做自己想做的事,而且,经常把工作带入生活 ...
- animate.css
这是个css3动画框架,现在很流行,在小动画效果有很多,使用的方式 <div class="animated bounce"></div> /*只要修改bo ...
- css之属性部分
这篇写的是今天的学习到的属性,一共20个. 属性再多,但也要会使用,会在使用时可以记起它,才能起到它为我们所需要的作用. 样式属性 1.border CSS边框属性允许你指定一个元素边框的样式和颜色. ...
- KS8U读卡器
凯盛读卡器KS8U 厂家:http://www.chinaiccard.com/feijiechushiduxieqi/947.html 驱动安装目录的COM目录下有网页调用的例子. 公司网站:htt ...
- jmap命令(Java Memory Map)(转)
JDK内置工具使用 一.javah命令(C Header and Stub File Generator) 二.jps命令(Java Virtual Machine Process Status To ...