版本信息那些事

本文介绍如何在Java项目中管理和使用版本信息,包括通过Manifest文件、Ant脚本以及使用注解的方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在做产品时,我们经常会遇到这样的需求:

  • 打Jar包时,需要压入产品的版本信息,甚至还有版权信息。
  • 在运行时,程序需要读取版本信息来做显示或者做分支处理。
向下兼容的一种实现方式是读取版本信息来做分支处理...Lucene是这种方式的代表之一。
版本信息的运行时获取的实现手段非常有限,大多使用 java.lang.Package来实现。

 

一、最简单的实现方式 - manifest

public class VersionInfo {

	public static String getVersion() {
		Package pkg = VersionInfo.class.getPackage();
		return (pkg != null ? pkg.getImplementationVersion() : null);
	}

}

这也是Spring的实现方法,但关键是版本信息的元数据来自哪里?这些信息存于Jar包内的MANIFEST.MF文件。该文件用于描述jar的情况,具体请见:http://baike.baidu.com/view/1857179.htm

 

这些属性中,和版本信息相关的属性是:

Implementation-Version: xxxx

 

 

 

二、ANT + manifest

思路是在build.xml里建立一个存放版本号的属性,并在打Jar包时,将版本号压入MANIFEST文件。程序在执行的时候,读取Jar包的MANIFEST来获取版本号。获取MENIFEST文件内版本号的方法没有任何改变,参考上一节。

 

build.xml

使用Eclipse自动生成build.xml后,在build.xml内增加这么几行:

	<property name="binary.version" value="1.0-SNAPSHOT" />

	<target name="package" description="package a jar" depends="cleanall, build-project">
		<jar jarfile="target/binary_version.jar" basedir="eclipse-build">
			<manifest>
				<attribute name="Implementation-Version" value="${binary.version}" />
				<attribute name="Main-Class" value="com.joshua.version.YourService" />
			</manifest>
		</jar>
	</target>

 

第一行是存放版本号的字段。名为package的target则负责把版本号压入manifest并制作jar包。

*注意:我存放class文件的目录是eclipse-build,大家需要把这个路径改成自己的目录(一般都是bin)

 

使用ant package命令打包,Jar包会生成在工程根目录下的target文件夹下。使用命令:java -jar binary_version.jar执行。就可以看到控制台上打印出来的版本信息了。

 

*注意:上述方式有个问题,就是不打包的情况下(比如在IDE里执行),是获取不到版本号的。

  

带SVN版本号的版本信息

当然现实的情况不可能这么简单,版本信息一般会带上Revision。比如,SVN的版本号。针对SVN,我们可以在build.xml里加个target:

  <target name="svnCheck" description="Use SVN to get the current revision" unless="disable-svnCheck">
    <exec executable="svn" logerror="true" outputproperty="svn.exec.result" failonerror="true" failifexecutionfails="true">
      <arg line="info"/>
    </exec>
    <loadresource property="svn.revision">
      <string value="${svn.exec.result}"/>
      <filterchain>
        <linecontains>
          <contains value="Last Changed Rev: "/>
        </linecontains>
        <tokenfilter>
            <!-- Remove all but the revision number -->
            <replaceregex pattern=".*: " replace=""/>
        </tokenfilter>
        <striplinebreaks/>
      </filterchain>
    </loadresource>
    <echo level="info" message="svn.revision=${svn.revision}"/>
  </target>

 

三、使用Annotation来实现

这种方式,我在Hadoop的源码中看到过,虽然繁琐,不过也挺有意思。优点是在非打包情况下也能工作。思路是定义1个Package级的annotation,annotation里定义了版本等属性。然后使用脚本自动获取版本信息并生成package-info.java,通过package-info传递Annotation的所有参数(包括版本信息),并且指定包名。在代码中使用反射获取被指定包的这个annotation,来获取版本信息。关于package-info的介绍,我这里就不做介绍了,google一下你就知道。

 

 

Annotation

 

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.PACKAGE)
public @interface HadoopVersionAnnotation {
 
  /**
   * Get the Hadoop version
   * @return the version string "0.6.3-dev"
   */
  String version();
  
  ...
}

 

 生成package-info的脚本我就不给了,因为不同项目实现也不同。下面给出被生成的package-info.java:

@HadoopVersionAnnotation(version="1.0.3"...【省略其他属性】)
package org.apache.hadoop;

 

接着,在代码里可以使用如下方式获取版本信息:

    myPackage = HadoopVersionAnnotation.class.getPackage();
    version = myPackage.getAnnotation(HadoopVersionAnnotation.class);

 

 按上述方式,如果使用shell来生成package-info,那么我仍需要获取到软件的版本信息。版本可能源于SVN,或者GIT等等。下面提供些shell脚本供大家参考。

 

获取git的revision:

revision=`git log -1 --pretty=format:"%H"`

 

获取git的branch名

branch=`git branch | sed -n -e 's/^* //p'`

 

获取SVN的revision

revision=`svn info | sed -n -e 's/Last Changed Rev: \(.*\)/\1/p'`

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值