有些时候,需要通过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代码:

  1. package test.compile;
  2.  
  3. import java.io.File;
  4. import java.text.SimpleDateFormat;
  5.  
  6. /**
  7. * <p>编译abc目录下所有java文件</p>
  8. * @author Ares
  9. * @version $Id: CompileTool.java, v 0.1 2015年02月26日 上午10:41:32 Exp $
  10. */
  11. public class CompileTool {
  12.  
  13. // DOS: 第一个参数是 src、cmd的父目录,第二个参数为最终压缩包(mylib.jar,mylib-source.jar的目录)所在目录
  14.  
  15. private static String os_name = System.getProperty("os.name").toLowerCase();
  16.  
  17. public static boolean compileAndJar(final String jarRootPath) throws Exception{
  18. String app_path = jarRootPath; // app_path是指定abc的绝对路径,因为java文件执行目录可能跟编译目录不一致,所以如果需要请使用此参数(脚本中第一步需要进入此目录)
  19. if(app_path !=null && (!app_path.endsWith("/") && !app_path.endsWith(File.separator)))
  20. app_path += File.separator;
  21. String compile_datetime = new SimpleDateFormat("yyyyMMddHHmmss").format(new java.util.Date());
  22. // 无后缀的jar名称(cmd 会拼写为 jarname.jar 和 jarname-source.jar)
  23. String jar_name = "myjar_" + compile_datetime; // jar名称格式,mylib_日期时间,以参数的形式传入,下面的脚本中未使用,如若使用,请替代脚本中的mylib
  24. String cmd = "";
  25. java.io.File cmdFile = null;
  26. if(os_name.indexOf("windows") > -1){
  27. cmd = "cmd.exe /C start " + app_path + "compile.exe " + app_path + " " + jar_name;
  28. cmdFile = new java.io.File(app_path + "compile.exe");
  29. }else {
  30. cmd = app_path + "compile.sh " + app_path + " " + jar_name;
  31. cmdFile = new java.io.File(app_path + "compile.sh");
  32. }
  33. if(!cmd.equals("") && cmdFile != null && cmdFile.exists()){
  34. if(!cmdFile.canExecute()) cmdFile.setExecutable(true);
  35. return callCmd(cmd);
  36. }else {
  37. throw new Exception("can't find compile command. command:"+cmdFile.getAbsolutePath());
  38. }
  39. }
  40.  
  41. private static boolean callCmd(String cmdStr) {
  42. try {
  43. Process child = Runtime.getRuntime().exec(cmdStr);
  44. java.io.InputStream in = child.getInputStream();
  45. while (in.read() != -1) {}
  46. in.close();
  47. try {
  48. child.waitFor();
  49. return true;
  50. } catch (InterruptedException e) {
  51. e.printStackTrace();
  52. }
  53. return false;
  54. } catch (java.io.IOException e) {
  55. e.printStackTrace();
  56. }
  57. return false;
  58. }
  59. }

下面是compile.sh文件内容:

  1. #!/bin/bash
  2.  
  3. CUR=$(pwd)
  4.  
  5. ## reset dest folder
  6. OUTPUT_PATH=${CUR}/dest
  7. rm -rf $OUTPUT_PATH
  8. mkdir -p ${OUTPUT_PATH}/bin
  9. mkdir -p ${OUTPUT_PATH}/libs
  10.  
  11. ## 获取依赖库和需要编译的java文件
  12. export libs=`find libs -name "*.jar" |xargs|sed "s/ /:/g"`
  13. export jfiles=`find src -name "*.java" |xargs|sed "s/ / /g"`
  14.  
  15. ## 编译
  16. opt="-d $OUTPUT_PATH/bin -cp ${libs} -encoding utf-8"
  17. javac $opt ${jfiles}
  18.  
  19. ## 打包
  20. cd ${OUTPUT_PATH}/bin
  21. jar -cvf ${CUR}/dest/mylib.jar *
  22. cd ${CUR}/src
  23. jar -cvf ${CUR}/dest/mylib-source.jar *

下面是compile.cmd文件内容:

  1. @ECHO OFF
  2. SETLOCAL enableDelayedExpansion
  3.  
  4. REM Save Current Path(保存当前路径)
  5. SET CUR=%CD%
  6.  
  7. REM Dir Structure(设置目录结构)
  8. SET OUTPUT_PATH=dest
  9. SET SRC_PATH=%CUR%\src
  10. SET LIBS_PATH=%CUR%\libs
  11.  
  12. REM Reset Output Dir(重置清空输出目录)
  13. IF EXIST %OUTPUT_PATH% RMDIR /s /q %OUTPUT_PATH%
  14. IF NOT EXIST %OUTPUT_PATH% MKDIR %OUTPUT_PATH%
  15. CD %OUTPUT_PATH%
  16. IF NOT EXIST bin MKDIR bin
  17. IF NOT EXIST libs MKDIR libs
  18.  
  19. REM Copy Files(复制jar文件)
  20. CD %CUR%
  21. SET CP=%OUTPUT_PATH%/libs
  22. copy %LIBS_PATH%\*.jar %OUTPUT_PATH%\libs\ /y
  23.  
  24. CD %CUR%
  25. FOR %%i IN ("libs\*.jar") DO SET CP=%%i;!CP!
  26. SET OPT=-d %OUTPUT_PATH%\bin -cp %CP% -encoding utf-8
  27.  
  28. CD %SRC_PATH%
  29. REM 获取所有需要编译的java源文件,并用空格链接
  30. REM 此处需要开启变量延迟(SETLOCAL enableDelayedExpansion不可缺少)
  31.  
  32. FOR /R %%b IN ( . ) DO (
  33. IF EXIST %%b/*.java SET JFILES=!JFILES! %%b/*.java
  34. )
  35.  
  36. CD %CUR%
  37. REM 正在编译...
  38. javac %OPT% %JFILES%
  39.  
  40. IF "%errorlevel%" EQU "0" (
  41. ECHO 编译成功,正在拷贝资源文件...
  42. copy %CUR%/src/*.* %OUTPUT_PATH%\src\ /y
  43. )
  44.  
  45. REM 准备打包编译文件和JAVA源文件
  46. CD %OUTPUT_PATH%/bin
  47. jar -cvf ../mylib.jar *
  48. cd %SRC_PATH%
  49. jar -cvf %CUR%/dest/mylib-source.jar *
  50.  
  51. EndLocal
  52. 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. 曲演杂坛--EXISTS语句

    通常在我写EXISTS语句时,我会写成IF EXISTS(SELECT TOP(1) 1 FROM XXX),也没细细考究过为什么要这么写,只是隐约认为这样写没有啥问题,那今天就深究下吧! 首先准备测 ...

  2. [C++] 行程编码C++代码

    #include<iostream> #include<string.h> #include<string> #include<cstring> #in ...

  3. Linux服务器Cache占用过多内存导致系统内存不足问题的排查解决(续)

    作者: 大圆那些事 | 文章可以转载,请以超链接形式标明文章原始出处和作者信息 网址: http://www.cnblogs.com/panfeng412/archive/2013/12/17/dro ...

  4. AngularJS快速入门指南01:导言

    AngularJS使用新的attributes扩展了HTML AngularJS对单页面应用的支持非常好(SPAs) AngularJS非常容易学习 现在就开始学习AngularJS吧! 关于本指南 ...

  5. JavaScript原生DOM操作API总结

    最近面试的时候被这个问题给卡了,所以抽时间好好复习一下. 原文:http://www.cnblogs.com/liuxianan/p/javascript-dom-api.html 几种对象 Node ...

  6. 固定表头/锁定前几列的代码参考[JS篇]

    引语:做有难度的事情,才是成长最快的时候.前段时间,接了一个公司的稍微大点的项目,急着赶进度,本人又没有独立带过队,因此,把自己给搞懵逼了.总是没有多余的时间来做自己想做的事,而且,经常把工作带入生活 ...

  7. animate.css

    这是个css3动画框架,现在很流行,在小动画效果有很多,使用的方式 <div class="animated bounce"></div> /*只要修改bo ...

  8. css之属性部分

    这篇写的是今天的学习到的属性,一共20个. 属性再多,但也要会使用,会在使用时可以记起它,才能起到它为我们所需要的作用. 样式属性 1.border CSS边框属性允许你指定一个元素边框的样式和颜色. ...

  9. KS8U读卡器

    凯盛读卡器KS8U 厂家:http://www.chinaiccard.com/feijiechushiduxieqi/947.html 驱动安装目录的COM目录下有网页调用的例子. 公司网站:htt ...

  10. jmap命令(Java Memory Map)(转)

    JDK内置工具使用 一.javah命令(C Header and Stub File Generator) 二.jps命令(Java Virtual Machine Process Status To ...