有些时候,需要通过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项目(或者运行项目)的更多相关文章

  1. 用UBOOT自带loadb命令加载应用程序到SDRAM中运行的方法

    S3C44B0开发板中,用UBOOT自带loadb命令加载应用程序到SDRAM中运行的方法    1.开发板说明:  开发板上已有移植好的UBOOT运行.   2.交叉编译工具链为arm-linu-g ...

  2. VC中加载LIB库文件的三种方法

    VC中加载LIB库文件的三种方法 在VC中加载LIB文件的三种方法如下: 方法1:LIB文件直接加入到工程文件列表中   在VC中打开File View一页,选中工程名,单击鼠标右键,然后选中&quo ...

  3. 使用composer命令加载vendor中的第三方类库

    1.首先下载需要的第三方类库,放在TP框架下的vendor文件夹下 2.给第三方类库SDK写上命名空间,命名空间为该类库的文件夹对应的名字 3.在composer.json文件中添加需要加载的第三方类 ...

  4. SuperSocket命令加载器 (Command Loader)

    在某些情况下,你可能希望通过直接的方式来加载命令,而不是通过自动的反射. 如果是这样,你可以实现你自己的命令加载器 (Command Loader): public interface IComman ...

  5. SuperSocket命令和命令加载器

    关键字: 命令, 命令加载器, 多命令程序集 命令 (Command) SuperSocket 中的命令设计出来是为了处理来自客户端的请求的, 它在业务逻辑处理之中起到了很重要的作用. 命令类必须实现 ...

  6. 【C++】C++程序加载lib静态库

    使用Visual Studio 编写C++程序有几种配置lib的方法,以下是在代码中加载lib文件的方法: 在项目所在目录下创建文件夹lib,将lib文件此路径下,包括Debug和Release两种模 ...

  7. VC6.0加载lib文件的三种方法

    MFC编写程序,都要用到动态链接库,MFC相关的动态库有MFCD42和MFC42等,MFC框架程序已经自动加载,那么如何引入第三方的动态链接库到工程中呢? 静态链接库是要先把程序中所需要使用的函数编译 ...

  8. 脚本命令加载外部配置文件夹conf

    加载log4j配置文件 Log4iConfigurer类 public class Log4iConfigurer { private static boolean INITIALIZED = fal ...

  9. maven打包时无法加载lib下的jar

    © 版权声明:本文为博主原创文章,转载请注明出处 问题描述: 项目在本地部署没有问题,但是使用maven打包时报错: ***(引用jar中某个类的的路径) 不存在 ***(某个java类中的某行某列) ...

随机推荐

  1. Linux kernel develop -- Hello World

    hello.c: #include <linux/init.h> #include <linux/module.h> #include <linux/kernel.h&g ...

  2. IIS7.5 发布程序后cookie丢失问题

    给客户部署网站程序时,发现网站的用户登录不了系统,就是跳转不到另外一个页面,一直保留在当前的页面.在本地的VS里面调试的时候是正常的,算在本地的IIS上发布了一下,测试也是这个问题. -------- ...

  3. 狗日的js的闭包

    一.变量的作用域 要懂得闭包,起首必须懂得Javascript特别的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript说话的特别之处,就在于函数内部可以直接读取全局变量 ...

  4. paip.取当天记录的方法sql跟hql hibernate

    paip.取当天记录的方法sql跟hql hibernate #------两个方法...函数法和日期计算法.. 函数法: DATEDIFF(d,createTime,GETDATE())=0   / ...

  5. SQL语句的基础

    注释语法:#注释语 一.T-SQL语句注意:1.语句写完后用"分号:"代表这一句结束2.列结束用逗号,最后一列写完不用写逗号3.符号一定是英文的 关键字:主键:primary ke ...

  6. el表达式的function标签

    使用el调用Java方法 1:EL表达式语法允许开发人员开发自定义函数,以调用java类的方法. ~示例:${el:method(params)} ~在EL表达式中调用的只能是java类的静态方法. ...

  7. 我和Java的故事-------第1弹

    前言:兴趣是学习任何新知识最好的老师 --------------------------------------------------------------------------------- ...

  8. Leetcode 70 Climbing Stairs 递推

    其实就是斐波那契数列 参考dp[n] = dp[n-1] +dp[n-2]; class Solution { public: int climbStairs(int n) { ; ; ; ; i & ...

  9. Java 线程 — JMM Java内存模型

    JMM Java Memory Model,Java内存模型,属于语言级的内存模型 并发编程中存在的问题: 如何通信:用于线程之间交换信息.两种方式:共享内存,消息传递 如何同步:用于控制不同线程间操 ...

  10. ubuntu安装gitlab

    #安装所有的依赖包 sudo apt-get install -y build-essential git-core sudo apt-get install -y zlib1g-dev libyam ...