標籤:
一直感覺AndroidStudio沒有eclipse快,但是最近由於遇到一個問題不得不將工程遷移到AndroidStudio上,遷移後之前在eclipse上所做的所有批量打包又得重新在AndroidStudio上搞一遍,不得不說這個過程遠比我想象的讓人愉快,AndroidStudio的強大和好用改變了之前我對這個IDE的偏見,無疑這個IDE是開發Android的最好工具。
一、普通打包配置
defaultConfig { applicationId myAppId minSdkVersion 14 targetSdkVersion 19 versionCode vCode versionName vName testApplicationId "com.xx.xxx.test" testInstrumentationRunner "android.test.InstrumentationTestRunner" multiDexEnabled true signingConfig signingConfigs.releaseConfig manifestPlaceholders = [package_name: myAppId, app_icon: myAppIcon, app_name: myAppName, umeng_appkey: umengAppKey, main_menu_json: mainMenus, main_app_change_string:appChangedText] }
這個配置是一個預設配置項,我們可以將一些公用的東西配置在這裡,在這裡可以看到我對AppId(也就是通常說的包名)用一個變數進行配置,因為在我的工程中同一個源碼要打多個應用(AppId),像這樣的需求應該不多見,但是你總有一天會碰到
def myAppId = ‘com.xx.xxx‘
大家肯定還會注意到我最下面的配置
manifestPlaceholders = [package_name: myAppId, app_icon: myAppIcon, app_name: myAppName, umeng_appkey: umengAppKey, main_menu_json: mainMenus, main_app_change_string:appChangedText]
這些配置都是在mainfest檔案中定義的變數,在這裡做一個統一配置(因為是預設配置),如果有個別渠道需要單獨配置可以在下面的productFlavors中做具體的個人化配置。
productFlavors { forum { } }
這個就是官網包的配置(使用的是預設配置),所以不需要做個人化配置,這樣就ok了,如果要添加其他渠道,比如baidu要個人化應用程式名稱。
productFlavors { forum { } baidu{ manifestPlaceholders = [app_name: ‘百度‘] } }
這樣看似很完美是不是,可以打不通的渠道不同的AppId的包了,事實上的確如此,但是我們都知道android的應用市場好幾百,一般上個20多個應用市場(也就是打20個渠道)不算多,這個時候你可能就要哭出來了,打個包的時間你就可以去好幾次洗手間,甚至可以去吃個晚飯了。AndroidStudio的build時間是相當的長,這個想必用過的人都有此感受,既然這樣我們可不可以借鑒eclipse上的多渠道打包思路,打一個包其他包通過修改設定檔實現呢?其實早有人對此做了實踐並證明是有效。
二、快速出包渠道再多也不怕
如果能直接修改apk的渠道號,而不需要再重新簽名能節省不少打包的時間。幸運的是我們找到了這種方法。直接解壓apk,解壓後的根目錄會有一個META-INF目錄,如果在META-INF目錄內添加空檔案,可以不用重新簽名應用。因此,通過為不同渠道的應用添加不同的空檔案,可以唯一標識一個渠道。思路其實很簡單(具體請參考這篇文章:1190000003763833)
/** * 從apk中擷取版本資訊 * @param context * @param channelKey * @return */ private static String getChannelFromApk(Context context, String channelKey) { //從apk包中擷取 ApplicationInfo appinfo = context.getApplicationInfo(); String sourceDir = appinfo.sourceDir; //預設放在meta-inf/裡, 所以需要再拼接一下 String key = "META-INF/" + channelKey; String ret = ""; ZipFile zipfile = null; try { zipfile = new ZipFile(sourceDir); Enumeration<?> entries = zipfile.entries(); while (entries.hasMoreElements()) { ZipEntry entry = ((ZipEntry) entries.nextElement()); String entryName = entry.getName(); if (entryName.startsWith(key)) { ret = entryName; break; } } } catch (IOException e) { e.printStackTrace(); } finally { if (zipfile != null) { try { zipfile.close(); } catch (IOException e) { e.printStackTrace(); } } } String[] split = ret.split("_"); String channel = ""; if (split != null && split.length >= 2) { channel = ret.substring(split[0].length() + 1); } return channel; }
這個方法就是從檔案名稱中擷取我們的渠道名稱(不同的包到時候會解壓後建立一個相關渠道的名的檔案),怎麼解壓後建立渠道相關的檔案呢?我們使用下面的python指令碼實現。
#!/usr/bin/python# coding=utf-8import zipfileimport shutilimport os# 空檔案 便於寫入此空檔案到apk包中作為channel檔案src_empty_file = ‘info/czt.txt‘# 建立一個空檔案(不存在則建立)f = open(src_empty_file, ‘w‘) f.close()# 擷取目前的目錄中所有的apk源包src_apks = []# python3 : os.listdir()即可,這裡使用相容Python2的os.listdir(‘.‘)apk_path = ‘E:/AndroidStudio/work/xxx/app/build/outputs/apk/‘for file in os.listdir(apk_path): fulldirfile = os.path.join(apk_path, file) if os.path.isfile(fulldirfile): extension = os.path.splitext(file)[1][1:] if extension in ‘apk‘: if file.find(‘unaligned‘) == -1: src_apks.append(fulldirfile)print(src_apks)# 擷取渠道列表channel_file = ‘info/channel.txt‘f = open(channel_file)lines = f.readlines()f.close()for src_apk in src_apks: # file name (with extension) src_apk_file_name = os.path.basename(src_apk) # 分割檔案名稱與尾碼 temp_list = os.path.splitext(src_apk_file_name) # name without extension src_apk_name = temp_list[0] # 尾碼名,包含. 例如: ".apk " src_apk_extension = temp_list[1] # 建立組建目錄,與檔案名稱相關 package_name = src_apk_name[0:src_apk_name.index(‘-‘)] output_dir = ‘G:/XXX打包/‘ + package_name +‘/output_‘ + src_apk_name + ‘/‘ # 目錄不存在則建立 if not os.path.exists(output_dir): os.mkdir(output_dir) # 遍曆渠道號並建立對應渠道號的apk檔案 for line in lines: # 擷取當前渠道號,因為從渠道檔案中獲得帶有\n,所有strip一下 target_channel = line.strip() # 拼接對應渠道號的apk target_apk = output_dir + src_apk_name + "-" + target_channel + src_apk_extension # 拷貝建立新apk shutil.copy(src_apk, target_apk) # zip擷取建立立的apk檔案 zipped = zipfile.ZipFile(target_apk, ‘a‘, zipfile.ZIP_DEFLATED) # 初始化渠道資訊 empty_channel_file = "META-INF/cztchannel_{channel}".format(channel = target_channel) # 寫入渠道資訊 zipped.write(src_empty_file, empty_channel_file) # 關閉zip流 zipped.close()
在這裡我根據我的實際工作對原本Github上的代碼進行了修改,部分地方改成了絕對路徑,並對unaligned包進行了過濾。
apk_path = ‘E:/AndroidStudio/work/xxx/app/build/outputs/apk/‘
if file.find(‘unaligned‘) == -1: src_apks.append(fulldirfile)
這樣我們無論打多少個渠道的包幾乎和打一個包的時間相同,因為其他包通過這個指令碼實現的是檔案的拷貝解壓和壓縮,幾乎用不了多少時間。
三、打包後的一些事情
前一段時間讓我鬱悶的是打包時間長,還有一件比較鬱悶的事情就是打完包後還要通過QQ發給相關同事(幾乎快要1G的包,很可怕),所以我對上面的python做了一點修改,將包按照具體的包名(AppId)建立目錄,存放到相應的目錄下(按照打包時間再細分目錄)。
output_dir = ‘G:/XXX打包/‘ + package_name +‘/output_‘ + src_apk_name + ‘/‘
這樣做就好辦了,再使用FTP將該根目錄共用出來,這樣打完包只需要給他發個震動視窗就可以了,他可以自己去取需要的包,後來我又想了想,這樣對於別人感覺尋找太麻煩了,我又添加了一個類似於目錄的東西,同樣是改python檔案。
#建立目錄索引index_file = open(‘G:/XXX打包/index.html‘, ‘r+‘);li = ‘<a href=ftp://192.168.2.54/‘ + package_name + ‘>‘ + src_apk_name + ‘</a>‘content = index_file.read()index_file.seek(0, 0)index_file.write(li+ ‘</br></br>\n‘ + content)index_file.close()
這樣就感覺方便多了,只是對我來說現在的打包已經不像之前那樣讓人煩了,我現在樂意去打包,因為我只需要點下滑鼠就可以打多個渠道的包,滿足任意需求打包。對於使用安裝包的同事來說也是一件很愉快的事情,因為我為他做的索引還是蠻不錯的。
可以看到這裡是對應的四個AppId(也就是四個應用),index.html就是索引,按照時間排序的,一般需要的都是最新的包。
AndroidStudio多AppId多渠道快速打包