已有Android项目接入ReactNative并实现热更新(二)

一、 拥有一个能运行的Android项目,项目名CodePushDemo

二、 已经看了已有Android项目接入ReactNative(一),并且按照步骤接入了RN,能打包apk并安装到手机,运行页面如下:

page-0.png
page-1.png

三、 搭建本地CodePush服务器

可以直接用微软的服务,但是服务器在国外不稳定,由于CodePushServer已经开源,所以可以down下来自己搭建。

3.1 确保已经安装mysql,codepush会自动创建数据库记录版本等信息

3.2 从git下载服务端源码


    git clone https://github.com/lisong/code-push-server.git
    
git_clone.jpg

进入code-push-server根目录,下载服务端需要的依赖:


    npm install

3.3 修改服务端配置

配置数据库:


    db: {
        username: process.env.RDS_USERNAME || "root",
        password: process.env.RDS_PASSWORD || "", //我没有设置密码,所以是空字符串
        database: process.env.DATA_BASE || "codepush",
        host: process.env.RDS_HOST || "127.0.0.1",
        port: process.env.RDS_PORT || 3306,
        dialect: "mysql",
        logging: false,
        operatorsAliases: false,
      }

配置本地存储目录:


    local: {
        storageDir: process.env.STORAGE_DIR || "C:/workspace/CodePush/code-push-server/download", //我就在项目下新建了一个download文件夹来存储上传的bundle
        downloadUrl: process.env.LOCAL_DOWNLOAD_URL || "http://127.0.0.1:3000/download",
        public: '/download'
      }

官方是要求配置jwt的,但是我没有配置也能用,最好还是配上,避免不必要的麻烦:


    jwt: {
        // Recommended: 63 random alpha-numeric characters
        // Generate using: https://www.grc.com/passwords.htm //去这个网址复制一个,不要用我已经在用的
        tokenSecret: process.env.TOKEN_SECRET ||'EKlCydP2vGppvHv7F35uvF4VLzhX3ULng15X0dOZck9gwXp1Mcfg8eXtTJjjkOa' 
      }

3.4 初始化创建数据库


    node ./bin/db init --dbhost localhost --dbuser root —dbpassword

db_init.jpg

3.5 启动CodePushServer


    node ./bin/www

www.jpg

四、 在本地服务器添加项目

4.1 安装code-push-cli


    npm install code-push-cli@latest -g

此时已经可以使用code-push命令行了。

4.2 登录code-push服务器


    code-push login http://localhost:3000

login.png

运行后会打开浏览器,需要输入账号admin密码123456,获取到token再填回命令行里,登录成功。

login-success.jpg

4.3 添加你的项目,此时还没有版本的数据,只需要一个项目名字CodePushDemo就行


    code-push add app CodePushDemo android react-native

code-push-add.jpg

添加完会提供两个key,需要记录下来,一个是生产环境Production的,一个是测试环境Staging的。

五、 配置原生项目

服务器搭好了,项目也添加了,剩下的就是如何将开发的app和服务器连接起来,关键就是上面记录的两个key。

5.1 改造application

引入CodePush模块,同时重写获取JSBundleFile的方法,由CodePush告诉我们读哪个bundle文件去加载React的内容。

application.jpg

但是此时CodePush还是爆红的,那从哪去找这个引用呢?

5.2 根目录下安装react-native-code-push


    npm install --save react-native-code-push

CodePush.jpg

安装完成,在项目根目录的node_modules下找到react-native-code-push模块,展开目录,可以看到CodePush类,手动引用进项目。

5.3 gradle配置

app的build.gradle增加配置:


    dependencies {
        ...
    
        implementation project(':react-native-code-push')
    }
    
    apply from: "../node_modules/react-native/react.gradle"
    apply from: "../node_modules/react-native-code-push/android/codepush.gradle"

project的setting.gradle增加配置:


    include ':app'
    //添加下面两行
    include ':react-native-code-push'
    project(':react-native-code-push').projectDir=new File(rootProject.projectDir, './node_modules/react-native-code-push/android/app')

重新构建完成,此时CodePush已经导入完成了,查看CodePush的构造函数:


    public CodePush(String deploymentKey, Context context, boolean isDebugMode, String serverUrl) {
            this(deploymentKey, context, isDebugMode);
            mServerUrl = serverUrl;
        }

第一个参数是保存的key,由于存在debug和release的区别,可以继续改造app的build.gradle文件,将key差异化:


    buildTypes {
            release {
                minifyEnabled false
                proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
                buildConfigField "String", "CODEPUSH_KEY", '"NwAy0d7C9yHyBZqEzdfkVaNO6izR4ksvOXqog"'
                signingConfig signingConfigs.release
            }
    
            debug {
                minifyEnabled false
                proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
                buildConfigField "String", "CODEPUSH_KEY", '"M2fp5m8A9nPrCFSnJeJoYGnvtXGX4ksvOXqog"'
            }
        }

最后一点,修改版本号为1.0.0,否则可能热更不成功

5.4 index.android.js文件修改

在打开React页面的时候进行安装热更,附上js文件:


    import React, {Component} from 'react';
    import {
        AppRegistry,
        StyleSheet,
        Text,
        View,
        AppState
    } from 'react-native';
    import CodePush from 'react-native-code-push';
    
    export default class CodePushComponent extends Component {
    
        // 增加的内容
        componentDidMount() {
            AppState.addEventListener("change", (newState) => {
                newState === 'active' && CodePush.sync();
            });
        }
    
        render() {
            return (
                <View style={styles.container}>
                    <Text style={styles.welcome}>
                        Welcome to React Native!
                    </Text>
                    <Text style={styles.instructions}>
                        我是 ReactNative
                    </Text>
                    <Text style={styles.instructions}>
                        Press Cmd+R to reload,{'\n'}
                        Cmd+D or shake for dev menu
                    </Text>
                </View>
            );
        }
    }
    
    const styles = StyleSheet.create({
        container: {
            flex: 1,
            justifyContent: 'center',
            alignItems: 'center',
            backgroundColor: '#F5FCFF',
        },
        welcome: {
            fontSize: 20,
            textAlign: 'center',
            margin: 10,
        },
        instructions: {
            textAlign: 'center',
            color: '#333333',
            marginBottom: 5,
        },
    });
    
    AppRegistry.registerComponent('CodePushComponent', () => CodePushComponent);


六、 发布更新

以上配置完成,已经可以打包了,这时在js里面做出一些修改,然后直接版本更新:


   code-push release-react CodePushDemo android --t 1.0.0 --d Staging --des '更新了一些js文字' -m false 

参数解释: --t 表示需要更新的版本
--d 表示上传bundle的环境
--des 后台记录的描述
-m 是否强制更新

命令行在服务器端查看更新情况:

deployment-key-list.png

重新打开app,如果已经打开的需要杀掉进程一次,因为没有设置强制更新,需要下一次启动可以看到变化。

最终效果:

page-last.png
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容