react-native 熱更新(ios&android)

由于公司項(xiàng)目需要,我抽時(shí)間研究了下react native的熱更新功能漂问,其實(shí)并不復(fù)雜赖瞒,自己在原來的demo上做了實(shí)驗(yàn),還是蠻成功的蚤假。廢話不多說栏饮,我們來看看是怎么實(shí)現(xiàn)的,還以之前的BaiDuMapDemo為例勤哗。

前提

使用熱更新,你必須安裝Android NDK掩驱,并設(shè)置環(huán)境變量ANDROID_NDK_HOME芒划,指向你的NDK根目錄。例如:/Users/tdzl2003/Downloads/android-ndk-r10e欧穴。
具體配置方法請(qǐng)自行百度构罗。

一妇萄、安裝

在你的項(xiàng)目根目錄下運(yùn)行以下命令:

npm install -g react-native-update-cli rnpm
npm install --save react-native-update@4.x
react-native link react-native-update

如果訪問極慢或者顯示網(wǎng)絡(luò)失敗,請(qǐng)?jiān)O(shè)置使用淘寶鏡像(僅需設(shè)置一次):

npm config set registry https://registry.npm.taobao.org --global
npm config set disturl https://npm.taobao.org/dist --global```

>說明:
1.`npm install -g react-native-update-cli rnpm`這一句在每一臺(tái)電腦上僅需運(yùn)行一次。
2.如果RN版本低于0.29纵顾,請(qǐng)使用`rnpm link`代替`react-native link`命令。
3.我使用的react-native版本為0.42.3丧肴,所以對(duì)應(yīng)的是4.x版本蘸劈。詳細(xì)的版本請(qǐng)參考下方表格。

####版本對(duì)應(yīng)表
因?yàn)镽eact Native不同版本代碼結(jié)構(gòu)不同弦叶,因而請(qǐng)按下面表格對(duì)號(hào)入座:

React Native版本 | react-native-update版本 
----|------
<= 0.26 | 1.0.x  
0.27 - 0.28 | 2.x  
0.29 - 0.33 | 3.x  
0.34 - 當(dāng)前 | 4.x

####二俊犯、手動(dòng)link
如果第一步的react-native link已成功(iOS工程和安卓工程均能看到依賴),可以跳過此步驟伤哺。
######(一)IOS
1. 在XCode中的Project Navigator里燕侠,右鍵點(diǎn)擊Libraries ? Add Files to BaiDuMapDemo
2. 進(jìn)入node_modules ? react-native-update ? ios 并選中RCTHotUpdate.xcodeproj
3. 在XCode中的project navigator里者祖,選中BaiDuMapDemo,在 Build Phases ? Link Binary With Libraries 中添加 libRCTHotUpdate.a
4. 繼續(xù)在Build Settings里搜索Header Search Path绢彤,添加$(SRCROOT)/../node_modules/react-native-update/ios

######(二)Android
1.在android/settings.gradle中添加如下代碼:

include ':react-native-update'
project(':react-native-update').projectDir = new File(rootProject.projectDir, '../node_modules/react-native-update/android')

2.在android/app/build.gradle的 dependencies 部分增加如下代碼:

compile project(':react-native-update')

3.檢查你的RN版本七问,如果是0.29及以上,,打開android/app/src/main/java/.../MainApplication.java茫舶,否則打開android/app/src/main/java/.../MainActivity.java械巡;在文件開頭增加 

import cn.reactnative.modules.update.UpdatePackage;

4.在getPackages()方法中增加 

new UpdatePackage()

####三、配置Bundle URL
######(一)IOS
1.在工程target的Build Phases->Link Binary with Libraries中加入libz.tbd奇适、libbz2.1.0.tbd
![20170601-sybil052-1.png](http://upload-images.jianshu.io/upload_images/5357992-92823e9fcc3f37e3.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

2.在你的AppDelegate.m文件中增加如下代碼:

// ... 其它代碼

import "RCTHotUpdate.h"

  • (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
    {

if DEBUG

// 原來的jsCodeLocation
jsCodeLocation = [NSURL URLWithString:@"http://localhost:8081/index.ios.bundle?platform=ios&dev=true"];

else

jsCodeLocation=[RCTHotUpdate bundleURL];

endif

// ... 其它代碼
}

######(二)Android
0.29及以后版本:在你的MainApplication中增加如下代碼:

// ... 其它代碼

import cn.reactnative.modules.update.UpdateContext;
public class MainApplication extends Application implements ReactApplication {

private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) {
@Override
protected String getJSBundleFile() {
return UpdateContext.getBundleUrl(MainApplication.this);
}
// ... 其它代碼
}
}

0.28及以前版本:在你的MainActivity中增加如下代碼:

// ... 其它代碼

import cn.reactnative.modules.update.UpdateContext;

public class MainActivity extends ReactActivity {

@Override
protected String getJSBundleFile() {
    return UpdateContext.getBundleUrl(this);
}
// ... 其它代碼

}

####四坟比、iOS的ATS例外配置
從iOS9開始,蘋果要求以白名單的形式在Info.plist中列出外部的非https接口嚷往,以督促開發(fā)者部署https協(xié)議葛账。在我們的服務(wù)部署https協(xié)議之前,請(qǐng)?jiān)贗nfo.plist中添加如下例外(右鍵點(diǎn)擊Info.plist皮仁,選擇open as - source code)
![20170601-sybil052-2.png](http://upload-images.jianshu.io/upload_images/5357992-bbd643e4e05e85a9.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

####五籍琳、登錄與創(chuàng)建應(yīng)用
1.首先請(qǐng)?jiān)赱http://update.reactnative.cn](http://update.reactnative.cn/)注冊(cè)帳號(hào),然后在你的項(xiàng)目根目錄下運(yùn)行以下命令:

$ pushy login
email: <輸入你的注冊(cè)郵箱>
password: <輸入你的密碼>

這會(huì)在項(xiàng)目文件夾下創(chuàng)建一個(gè).update文件贷祈,注意不要把這個(gè)文件上傳到Git等CVS系統(tǒng)上趋急。你可以在.gitignore末尾增加一行.update來忽略這個(gè)文件。

2.登錄之后可以創(chuàng)建應(yīng)用势誊。注意iOS平臺(tái)和安卓平臺(tái)需要分別創(chuàng)建:

$ pushy createApp --platform ios
App Name: <輸入應(yīng)用名字>
$ pushy createApp --platform android
App Name: <輸入應(yīng)用名字>

兩次輸入的名字可以相同呜达,這沒有關(guān)系。

3.選擇或者創(chuàng)建過應(yīng)用后粟耻,你將可以在文件夾下看到update.json文件:
![20170601-sybil052-3.png](http://upload-images.jianshu.io/upload_images/5357992-93634ce6452b4653.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
至此應(yīng)用的創(chuàng)建/選擇就已經(jīng)成功了查近。上張圖:

![20170601-sybil052-4.png](http://upload-images.jianshu.io/upload_images/5357992-98acce61902fff96.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

####六、添加熱更新功能代碼
在BaiDuMapDemo.js中添加代碼:

...
// 其他代碼
import {
isFirstTime,
isRolledBack,
packageVersion,
currentVersion,
checkUpdate,
downloadUpdate,
switchVersion,
switchVersionLater,
markSuccess,
} from 'react-native-update';

import _updateConfig from './update.json';
const {appKey} = _updateConfig[Platform.OS];

export default class BaiduMapDemo extends Component {
...
//其他代碼
this.checkUpdate = this.checkUpdate.bind(this);
}
componentWillMount() {
if (isFirstTime) {
Alert.alert('提示', '這是當(dāng)前版本第一次啟動(dòng),是否要模擬啟動(dòng)失敗?失敗將回滾到上一版本', [
{text: '是', onPress: ()=>{throw new Error('模擬啟動(dòng)失敗,請(qǐng)重啟應(yīng)用')}},
{text: '否', onPress: ()=>{markSuccess()}},
]);
} else if (isRolledBack) {
Alert.alert('提示', '剛剛更新失敗了,版本被回滾.');
}
}

doUpdate = info => {
    downloadUpdate(info).then(hash => {
        Alert.alert('提示', '下載完畢,是否重啟應(yīng)用?', [
            {text: '是', onPress: ()=>{switchVersion(hash);}},
            {text: '否',},
            {text: '下次啟動(dòng)時(shí)', onPress: ()=>{switchVersionLater(hash);}},
        ]);
    }).catch(err => {
        Alert.alert('提示', '更新失敗.');
    });
};

checkUpdate () {
    console.log('appKey.....',appKey);
    checkUpdate(appKey).then((info) => {
        console.log('info...', info);
        if (info.expired) {
            Alert.alert('提示', '您的應(yīng)用版本已更新,請(qǐng)前往應(yīng)用商店下載新的版本', [
                {text: '確定', onPress: ()=>{info.downloadUrl && Linking.openURL(info.downloadUrl)}},
            ]);
        } else if (info.upToDate) {
            Alert.alert('提示', '您的應(yīng)用版本已是最新.');
        } else {
            Alert.alert('提示', '檢查到新的版本'+info.name+',是否下載?\n'+ info.description, [
                {text: '是', onPress: ()=>{this.doUpdate(info)}},
                {text: '否',},
            ]);
        }
    }).catch(err => {
        console.log('err,,...',err);
        Alert.alert('提示', '更新失敗.');
    });
};

render() {
return (
<View style={styles.container}>
...
// 其他代碼
<View style={styles.row}>
<Button title="Traffic" onPress={() => {
this.setState({
trafficEnabled: !this.state.trafficEnabled
});
}} />
...
// 其他代碼
<Button title="檢查更新" onPress={() => {
console.log('我這行了嗎');
this.checkUpdate();
}} />
</View>
</View>
);
}
}
...
// 其他代碼

####七挤忙、發(fā)布應(yīng)用
下面我們要分成兩部分來搞了霜威,先來搞Android發(fā)布應(yīng)用。
#####(一)Android
######1.生成簽名密鑰
使用如下命令生成密鑰:

$ keytool -genkey -v -keystore my-release-key.keystore -alias my-key-alias -keyalg RSA -keysize 2048 -validity 10000

######2.設(shè)置gradle變量
(1)把my-release-key.keystore文件放到你工程中的android/app文件夾下册烈。
(2)編輯~/.gradle/gradle.properties(沒有這個(gè)文件你就創(chuàng)建一個(gè))戈泼,添加如下的代碼(注意把其中的****替換為相應(yīng)密碼)。

MYAPP_RELEASE_STORE_FILE=my-release-key.keystore
MYAPP_RELEASE_KEY_ALIAS=my-key-alias
MYAPP_RELEASE_STORE_PASSWORD=*****
MYAPP_RELEASE_KEY_PASSWORD=*****

######3.添加簽名到項(xiàng)目的gradle配置文件
編輯你項(xiàng)目目錄下的android/app/build.gradle赏僧,添加如下的簽名配置:

...
android {
...
defaultConfig { ... }
signingConfigs {
release {
storeFile file(MYAPP_RELEASE_STORE_FILE)
storePassword MYAPP_RELEASE_STORE_PASSWORD
keyAlias MYAPP_RELEASE_KEY_ALIAS
keyPassword MYAPP_RELEASE_KEY_PASSWORD
}
}
buildTypes {
release {
...
signingConfig signingConfigs.release
}
}
}
...

######4.生成發(fā)行APK包
在終端中運(yùn)行以下命令:

$ cd android && ./gradlew assembleRelease

生成的APK文件位于android/app/build/outputs/apk/app-release.apk大猛,它已經(jīng)可以用來發(fā)布了。
######5.發(fā)布
運(yùn)行如下命令:

$ pushy uploadApk android/app/build/outputs/apk/app-release.apk

######6.發(fā)布新的熱更新版本
修改一行代碼淀零,例如:
![20170601-sybil052-6.png](http://upload-images.jianshu.io/upload_images/5357992-ae9d6d76a6027ca7.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
在根目錄下輸入命令:

$ pushy bundle --platform android
Bundling with React Native version: 0.22.2
<各種進(jìn)度輸出>
Bundled saved to: build/output/android.1459850548545.ppk
Would you like to publish it?(Y/N)
輸入Y
Uploading [========================================================] 100% 0.0s
Enter version name: 1.0.1
Enter description: 測(cè)試
Enter meta info: {"ok":1}
Ok.
Would you like to bind packages to this version?(Y/N)
輸入Y

  1. 1.0(normal) (newest)
    Total 1 packages.
    Enter packageId:9310(上面列表中前面的數(shù)字)
版本綁定完畢后胎署,客戶端就應(yīng)當(dāng)可以檢查到更新并進(jìn)行更新了。
![20170601-sybil052-7.png](http://upload-images.jianshu.io/upload_images/5357992-612a59db3e7bb6d4.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

至此為止窑滞,android已經(jīng)完成了植入代碼熱更新的全部工作琼牧。
#####(二)IOS
######1.生成發(fā)行APK包
>注:ios打包有很多種方法恢筝,因?yàn)楸綿emo在Xcode中歸檔失敗,總是報(bào)Linker command failed with exit code 1錯(cuò)誤巨坊,暫未找到錯(cuò)誤原因撬槽。因此我將APP直接運(yùn)行到手機(jī)上,其他打包方式請(qǐng)自行百度趾撵。

![QQ20170602-110558@2x.png](http://upload-images.jianshu.io/upload_images/5357992-bab8897607e8afd5.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

1.終端cd到項(xiàng)目目錄侄柔,執(zhí)行命令

react-native bundle --entry-file index.ios.js --platform ios --dev false --bundle-output ./ios/bundle/index.ios.jsbundle --assets-dest ./ios/bundle

2.導(dǎo)入生成的文件到XCode
![QQ20170602-114650@2x.png](http://upload-images.jianshu.io/upload_images/5357992-2007aee1418da7f3.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

>注:詳細(xì)內(nèi)容見文章[RN-iOS打包真機(jī)測(cè)試](http://www.reibang.com/p/a3d486346edc)

3.在Xcode中選擇Product -> Scheme -> Edit Scheme (cmd + <),然后選擇Run選項(xiàng)卡占调,將Build Configuration設(shè)置為release
如圖:
![20170601-sybil052-8.png](http://upload-images.jianshu.io/upload_images/5357992-cfed355a03787552.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

4.選擇設(shè)備

![QQ20170601-194204@2x.png](http://upload-images.jianshu.io/upload_images/5357992-831931b5ad395f70.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

5.選擇Product -> Build(command+B)暂题,build成功后,找到目錄下product文件夾中的BaiDuMapDemo.app究珊,右鍵Show in Finder薪者,如圖:
![QQ20170602-100826@2x.png](http://upload-images.jianshu.io/upload_images/5357992-4d6734bcf45b2fd7.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

5.打開iTunes,將BaiduMapDemo文件拖到應(yīng)用中剿涮,再將iTunes中的BaiduMapDemo拖到桌面言津,并在終端輸入以下命令:

pushy uploadIpa /Users/xxx/Desktop/BaiDuMapDemo.ipa

6.連接真機(jī),在Xcode中選擇真機(jī)取试,點(diǎn)擊運(yùn)行悬槽,即可安裝app。
######2.發(fā)布新的熱更新版本
修改一行代碼瞬浓,在根目錄下輸入命令:
![20170601-sybil052-9.png](http://upload-images.jianshu.io/upload_images/5357992-c0050be7db24732f.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

版本綁定完畢后初婆,客戶端就應(yīng)當(dāng)可以檢查到更新并進(jìn)行更新了。
至此為止猿棉,ios完成了植入代碼熱更新的全部工作磅叛。
上幾張效果圖:
![sybil052_1413.PNG](http://upload-images.jianshu.io/upload_images/5357992-b34a80af229c8579.PNG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

![sybil052_1414.PNG](http://upload-images.jianshu.io/upload_images/5357992-a00b1d77b56ff9e5.PNG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市铺根,隨后出現(xiàn)的幾起案子宪躯,更是在濱河造成了極大的恐慌乔宿,老刑警劉巖位迂,帶你破解...
    沈念sama閱讀 222,627評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異详瑞,居然都是意外死亡掂林,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,180評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門坝橡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來泻帮,“玉大人,你說我怎么就攤上這事计寇÷嘣樱” “怎么了脂倦?”我有些...
    開封第一講書人閱讀 169,346評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)元莫。 經(jīng)常有香客問我赖阻,道長(zhǎng),這世上最難降的妖魔是什么踱蠢? 我笑而不...
    開封第一講書人閱讀 60,097評(píng)論 1 300
  • 正文 為了忘掉前任火欧,我火速辦了婚禮,結(jié)果婚禮上茎截,老公的妹妹穿的比我還像新娘苇侵。我一直安慰自己,他們只是感情好企锌,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,100評(píng)論 6 398
  • 文/花漫 我一把揭開白布榆浓。 她就那樣靜靜地躺著,像睡著了一般霎俩。 火紅的嫁衣襯著肌膚如雪哀军。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,696評(píng)論 1 312
  • 那天打却,我揣著相機(jī)與錄音杉适,去河邊找鬼。 笑死柳击,一個(gè)胖子當(dāng)著我的面吹牛猿推,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播捌肴,決...
    沈念sama閱讀 41,165評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼蹬叭,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了状知?” 一聲冷哼從身側(cè)響起秽五,我...
    開封第一講書人閱讀 40,108評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎饥悴,沒想到半個(gè)月后坦喘,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,646評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡西设,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,709評(píng)論 3 342
  • 正文 我和宋清朗相戀三年瓣铣,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片贷揽。...
    茶點(diǎn)故事閱讀 40,861評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡棠笑,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出禽绪,到底是詐尸還是另有隱情蓖救,我是刑警寧澤洪规,帶...
    沈念sama閱讀 36,527評(píng)論 5 351
  • 正文 年R本政府宣布,位于F島的核電站循捺,受9級(jí)特大地震影響淹冰,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜巨柒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,196評(píng)論 3 336
  • 文/蒙蒙 一樱拴、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧洋满,春花似錦晶乔、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,698評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至驻民,卻和暖如春翻具,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背回还。 一陣腳步聲響...
    開封第一講書人閱讀 33,804評(píng)論 1 274
  • 我被黑心中介騙來泰國打工裆泳, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人柠硕。 一個(gè)月前我還...
    沈念sama閱讀 49,287評(píng)論 3 379
  • 正文 我出身青樓工禾,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親蝗柔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子闻葵,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,860評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容