通过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类中的某行某列) ...
随机推荐
- Silverlight动态载入调用XAML资源
以多语言为例子: ResourceDictionary resourceDictionary = new ResourceDictionary(); //新建资源集合 var culture = ...
- [游戏模版18] Win32 五子棋
>_<:Learning its AI logic. >_<:resource >_<:code: #include <windows.h> // C ...
- oracle数据库中varchar2陷阱
oracle数据库相信大家都比较熟悉,数据库中有一种非常常用的数据类型:字符串型. 对应该类型,在oracle中有三种比较常用的类型:varchar2(byte).varchar2(char).nva ...
- 深入理解JavaScript 事件
本文总结自<JavaScript高级程序设计>以及自己平时的经验,针对较新浏览器以及 DOM3 级事件标准(2016年8月),对少部分内容作了更正,增加了各种例子及解析. 如无特殊说明,本 ...
- iOS----友盟分享完善版本
分享 详细集成 注意:1.线上集成文档的示例代码对应的是最新版本的SDK,如果你所用的SDK版本类名或者方法名与此文档不符合,请看随包里面的线下文档或者下载使用最新版本的SDK. 设置友盟appkey ...
- 记忆化搜索hdu1078 dfs
http://acm.hdu.edu.cn/showproblem.php?pid=1078 题意:给出n*n的格子,每个各自里面有些食物,问一只老鼠每次走最多k步所能吃到的最多的食物 #includ ...
- jsp、js分页功能的简单总结
一.概述 首先,我们要明确为何需要分页技术,主要原因有以下: 1.分页可以提高客户体验度,适当地选择合适的数据条数,让页面显得更有条理,使得用户体验感良好,避免过多数据的冗余. 2.提高性能的需要.分 ...
- A new start
学习前端已经有一年多,也自己写过不少博客.笔记,但是比较不开心的是有时候自己写的博客自己都看不懂,所以最后决定重新开一个博客,目标是写能让别人看懂也能让自己看懂的东西,那才算真成功真的懂了.最后希望的 ...
- HOWTO - 压缩封装的Setup.exe(纯MSI)安装包获取运行Log
在InstallShield中我们有时即使选择Basic MSI工程,但有时还是将MSI封装到Setup.exe中交付. 那么出现运行异常如何获取Log,可以用下面的方法: Setup.exe /v& ...
- ActiveMQ 使用
ActiveMQ资料大全 官方文档 入门:http://activemq.apache.org/getting-started.html 常见问题:http://activemq.apache.org ...