項(xiàng)目處理問題遇到如下情況,引用第三方的jar包報(bào)出ClassDefNotFoundException錯(cuò)誤導(dǎo)致app崩潰,雙方溝通確認(rèn)該類存在且排除工程未clean,代碼混淆導(dǎo)致等原因。
注意到編譯app的時(shí)候提示了很多如下與正常情況下不一樣的信息:
trouble processing:
bad class file magic (cafebabe) or version (0033.0000)
...while parsing com/baidu/mapapi/SDKInitializer.class
...while processing com/baidu/mapapi/SDKInitializer.class
279 warnings
SDKInitializer正是ClassDefNotFoundException報(bào)錯(cuò)的類。
百度搜索出現(xiàn)這個(gè)編譯提示找到了問題出現(xiàn)的原因:第三方公司打包這個(gè)jar包使用的jdk1.7版本(大于1.6),而我們編譯app使用的是1.6版本,由于編譯出的字節(jié)碼版本不一致,導(dǎo)致無法轉(zhuǎn)換成android虛擬機(jī)刻度的字節(jié)碼,因此引用的jar包實(shí)際并未編譯到當(dāng)前apk中,程序運(yùn)行的時(shí)候當(dāng)然就找不到相應(yīng)的類了。
公司編譯服務(wù)器實(shí)際上配置了多個(gè)jdk版本,而且android編譯是在設(shè)置環(huán)境變量的時(shí)候設(shè)置的jdk版本。編譯android4.4使用的是jdk1.6,而編譯android5.0則使用的是jdk1.7,這就涉及到了如何隨時(shí)切換jdk版本。
先來看看如何設(shè)置jdk環(huán)境變量,百度搜索得到的答案多是修改~/.bashrc或修改/etc/profile文件一次性完成配置。我用vi打開這兩個(gè)文件并未找到相應(yīng)的jdk環(huán)境變量,但是使用java -version確實(shí)能看到默認(rèn)的jdk版本。這就有了疑問,當(dāng)前默認(rèn)jdk軟件的環(huán)境變量是在哪里配置的?
使用 which java 命令找到當(dāng)前java可執(zhí)行程序的位置
fordreamxin@compiler207:~$ which java
/usr/bin/java
/usr/bin/目錄下存放的多是用戶安裝的軟件
fordreamxin@compiler207:~$ ll /usr/bin/java
lrwxrwxrwx 1 root root 22 Nov 13 06:18 /usr/bin/java -> /etc/alternatives/java*
alternatives是一個(gè)linux下的多版本管理軟件,利用它就可以實(shí)現(xiàn)jdk的版本切換,這點(diǎn)稍后再描述。先來alternatives目錄地下看看有些什么東西。
fordreamxin@compiler207:/etc/alternatives$ ls java*
java java.1.gz javac javac.1.gz javadoc javadoc.1.gz javah javah.1.gz javap javap.1.gz java_vm javaws javaws.1.gz
fordreamxin@compiler207:/etc/alternatives$ ll java
lrwxrwxrwx 1 root root 36 Nov 13 06:17 java -> /usr/lib/jvm/java-6-sun/jre/bin/java*
原來該目錄下放置了很多軟件的版本引用,從這里可以找到當(dāng)前軟件版本(jdk)的實(shí)際位置
fordreamxin@compiler207:/etc/alternatives$ ls /usr/lib/jvm/
java-1.7.0-openjdk-amd64 java-6-sun java-6-sun-1.6.0.26 java-7-openjdk-amd64
在/usr/lib/jvm文件夾下時(shí)間存儲(chǔ)了各個(gè)jdk版本的軟件,而alternatives正是通過修改它的java引用達(dá)到版本切換的目的。
使用alternatives實(shí)現(xiàn)版本切換的方法如下:
1. 查看相應(yīng)的jdk是否在 ubuntu的jdk菜單里,查看:
update-alternatives --config java
update-alternatives --config javac
2.如果沒有在菜單里可以如下方式添加:
update-alternatives --install /usr/bin/java java /usr/lib/jvm/java/jdk1.6.0_12/bin/java 300
update-alternatives --install /usr/bin/javac javac /usr/lib/jvm/java/jdk1.6.0_12/bin/javac 300
注意:jdk1.6.0_12 版本不同會(huì)有變動(dòng)
3.sudo update-alternatives --config java
sudo update-alternatives --config javac
選擇序號(hào),回車即可;
4.然后java -version,javac -version查看當(dāng)前jdk版本
但是使用alternatives需要sudo超級(jí)權(quán)限,也就是說這種切換是一種全局切換,同修改~/.bashrc或/etc/profile一樣,一次操作,全局有效。如果沒有超級(jí)權(quán)限怎么辦?
設(shè)置環(huán)境變量linux提供了export命令,該命令的修改只對(duì)當(dāng)前終端有效。
fordreamxin@compiler207:/$ echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games
上面的命令顯示了當(dāng)前的環(huán)境變量,也就是~/.bashrc或/etc/profile文件的內(nèi)容,它表示了當(dāng)前所有能夠全局使用的所有軟件或命令的環(huán)境變量,沒有這些變量,我們不能在任意目錄下使用很多命令或軟件,這也是環(huán)境變量的意義所在。
通過如下命令可以在$PATH的開始或末尾添加jdk的環(huán)境變量(bin目錄下保存了java, javac, javah等可執(zhí)行命令)
export PATH="/usr/lib/jvm/java-7-openjdk-amd64/bin":$PATH
export PATH="$PATH:/usr/lib/jvm/java-7-openjdk-amd64/bin/"
由于執(zhí)行的時(shí)候總是從$PATH的開始搜索可執(zhí)行文件的位置,所以如果環(huán)境變量中已經(jīng)設(shè)置了一個(gè)jdk,那么把新的jdk設(shè)置在PATH的開始才會(huì)有用。
export的具體使用可參考它的使用手冊(cè)。