项目处理问题遇到如下情况,引用第三方的jar包报出ClassDefNotFoundException错误导致app崩溃,双方沟通确认该类存在且排除工程未clean,代码混淆导致等原因。
注意到编译app的时候提示了很多如下与正常情况下不一样的信息:
troubleprocessing:
badclassfilemagic(cafebabe)orversion(0033.0000)
...whileparsingcom/baidu/mapapi/SDKInitializer.class
...whileprocessingcom/baidu/mapapi/SDKInitializer.class
279warnings
SDKInitializer正是ClassDefNotFoundException报错的类。
百度搜索出现这个编译提示找到了问题出现的原因:第三方公司打包这个jar包使用的jdk1.7版本(大于1.6),而我们编译app使用的是1.6版本,由于编译出的字节码版本不一致,导致无法转换成android虚拟机刻度的字节码,因此引用的jar包实际并未编译到当前apk中,程序运行的时候当然就找不到相应的类了。
公司编译服务器实际上配置了多个jdk版本,而且android编译是在设置环境变量的时候设置的jdk版本。编译android4.4使用的是jdk1.6,而编译android5.0则使用的是jdk1.7,这就涉及到了如何随时切换jdk版本。
先来看看如何设置jdk环境变量,百度搜索得到的答案多是修改~/.bashrc或修改/etc/profile文件一次性完成配置。我用vi打开这两个文件并未找到相应的jdk环境变量,但是使用java-version确实能看到默认的jdk版本。这就有了疑问,当前默认jdk软件的环境变量是在哪里配置的?
使用whichjava命令找到当前java可执行程序的位置
fordreamxin@compiler207:~$whichjava
/usr/bin/java
/usr/bin/目录下存放的多是用户安装的软件
fordreamxin@compiler207:~$ll/usr/bin/java
lrwxrwxrwx1rootroot22Nov1306:18/usr/bin/java->/etc/ alternatives/java*
alternatives是一个linux下的多版本管理软件,利用它就可以实现jdk的版本切换,这点稍后再描述。先来 alternatives目录地下看看有些什么东西。
fordreamxin@compiler207:/etc/ alternatives$lsjava*
javajava.1.gzjavacjavac.1.gzjavadocjavadoc.1.gzjavahjavah.1.gzjavapjavap.1.gzjava_vmjavawsjavaws.1.gz
fordreamxin@compiler207:/etc/ alternatives$lljava
lrwxrwxrwx1rootroot36Nov1306:17java->/usr/lib/jvm/java-6-sun/jre/bin/java*
原来该目录下放置了很多软件的版本引用,从这里可以找到当前软件版本(jdk)的实际位置
fordreamxin@compiler207:/etc/ alternatives$ls/usr/lib/jvm/
java-1.7.0-openjdk-amd64java-6-sunjava-6-sun-1.6.0.26java-7-openjdk-amd64
在/usr/lib/jvm文件夹下时间存储了各个jdk版本的软件,而 alternatives正是通过修改它的java引用达到版本切换的目的。
使用 alternatives实现版本切换的方法如下:
1.查看相应的jdk是否在ubuntu的jdk菜单里,查看:
update- alternatives--configjava
update- alternatives--configjavac
2.如果没有在菜单里可以如下方式添加:
update- alternatives--install/usr/bin/javajava/usr/lib/jvm/java/jdk1.6.0_12/bin/java300
update- alternatives--install/usr/bin/javacjavac/usr/lib/jvm/java/jdk1.6.0_12/bin/javac300
注意:jdk1.6.0_12版本不同会有变动
3.sudoupdate- alternatives--configjava
sudoupdate- alternatives--configjavac
选择序号,回车即可;
4.然后java-version,javac-version查看当前jdk版本
但是使用 alternatives需要sudo超级权限,也就是说这种切换是一种全局切换,同修改~/.bashrc或/etc/profile一样,一次操作,全局有效。如果没有超级权限怎么办?
设置环境变量linux提供了export命令,该命令的修改只对当前终端有效。
fordreamxin@compiler207:/$echo$PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games
上面的命令显示了当前的环境变量,也就是~/.bashrc或/etc/profile文件的内容,它表示了当前所有能够全局使用的所有软件或命令的环境变量,没有这些变量,我们不能在任意目录下使用很多命令或软件,这也是环境变量的意义所在。
通过如下命令可以在$PATH的开始或末尾添加jdk的环境变量(bin目录下保存了java,javac,javah等可执行命令)
exportPATH="/usr/lib/jvm/java-7-openjdk-amd64/bin":$PATH
exportPATH="$PATH:/usr/lib/jvm/java-7-openjdk-amd64/bin/"
由于执行的时候总是从$PATH的开始搜索可执行文件的位置,所以如果环境变量中已经设置了一个jdk,那么把新的jdk设置在PATH的开始才会有用。
export的具体使用可参考它的使用手册。