滴滴出行小程序I18n最佳實踐

做者:sky-admin貓兒不熊javascript

背景

I18n = Internationalization,國際化,由於單詞由首末字符i/n和中間18個字母組成,簡稱i18n。對程序來講,就是要在不修改內部代碼的狀況下,能根據不一樣語言及地區顯示相應的界面,以支持不一樣語言的人順利使用程序。html

業務背景

互聯網行業進入下半場,精細化運營是關鍵。多語言支持能讓產品更好地服務境內的其餘語言用戶,也爲產品出海打下基礎,隨着 WeChat/Alipay 的全球化,你的小程序是否作好準備了呢?前端

4月初,滴滴出行小程序團隊接到支持英文版的需求,預計上線時間爲6月上旬。當前滴滴出行小程序集成的衆多業務線和各類公共庫,展現給用戶的有前端硬編碼的靜態文本和服務端下發的文案,都要同步接入多語言。考慮到小程序當前的體量,光文本收集、語料翻譯、npm package 支持,聯調,測試,溝通成本等等,而且前端開發只投入1.5人力的狀況下,時間是蠻緊迫的,可是咱們抗住了壓力,最終英文版滴滴出行小程序如期上線,截止目前運行穩定,用戶反饋良好,獲得了超出預期的收益。vue

固然這一切得益於各團隊同窗的高效工做,和各團隊的通力配合,更得益於部門技術團隊 Mpx框架優雅的多語言能力支持。劃重點來咯,所謂工欲善其事必先利其器,若是你的公司業務須要開發小程序,也須要接入多語言,那麼請搬好小板凳,咱們來看一下小程序框架 Mpx 是如何優雅支持多語言能力。相信看完這篇,能夠幫助你認識 Mpx(https://github.com/didi/mpx) ,加深對框架的理解,最終利用 Mpx 框架高效迭代小程序,年終獎多出那部分能夠打賞一下做者,買杯咖啡哈(偷笑.jpg)java

如下是滴滴出行小程序的中英文版本對比:webpack

滴滴出行微信小程序i18n

也歡迎你們在微信/支付寶裏搜索滴滴出行小程序,實際使用感覺下。PS:切換語言的方法是,打開小程序,點擊左上角用戶頭像,進入側邊欄設置頁面,點擊切換中英文便可體驗。git

技術背景

在上述業務背景下,Mpx 框架——滴滴自研的專一提高小程序開發體驗的加強型小程序框架,內建 i18n 能力便提上日程。github

與 WEB 不一樣,小程序(本文以微信小程序爲例)運行環境採用雙線程架構設計,渲染層的界面使用 WebView 進行渲染,邏輯層採用 JSCore 線程運行 JS腳本。邏輯層數據改變,經過 setData 將數據轉發到 Native(微信客戶端),Native 再將數據轉發到渲染層,以此更新頁面。因爲線程間通訊成本較高,實際項目開發時須要控制頻次和數量。另外小程序的渲染層不支持運行 JS ,一些如事件處理等操做沒法在渲染層實現,所以微信官方提供了一套腳本語言 WXS ,結合 WXML ,能夠構建出頁面的結構(不瞭解 WXS ?戳這裏)。web

基於小程序的雙線程架構設計,實現 i18n 存在一些技術上的難點與挑戰,因爲 Mpx 框架早期構建起來的強大基礎,最終得以優雅支持多語言能力,實現了和vue-i18n 基本一致的使用體驗。npm

使用

在使用上,Mpx 支持 i18n 能力提供的 API 與 vue-i18n 大致對齊,用法上也基本一致。

模板中使用 i18n

編譯階段經過用戶配置的 i18n 字典,結合框架內建的翻譯函數經過 wxs-i18n-loader 合成爲可執行的 WXS 翻譯函數,並自動注入到有翻譯函數調用的模板中,具體調用方式以下圖。

// mpx文件
<template>
  <view>
    <view>{{ $t('message.hello', { msg: 'hello' })}}</view>
    <!-- formattedDatetime計算屬性,可基於locale變動響應刷新 -->
    <view>{{formattedDatetime}}</view>
  </view>
</template>
複製代碼

JS 中使用 i18n

經過框架提供的 wxs2js 能力,將 WXS 翻譯函數轉換爲 JS 模塊注入到 JS 運行時,使運行時環境中也可以調用翻譯函數。

// mpx文件
<script> import mpx, { createComponent } from '@mpxjs/core' createComponent({ ready () { // js中使用 console.log(this.$t('message.hello', { msg: 'hello' })) // 局部locale變動,生效範圍爲當前組件內 this.$i18n.locale = 'en-US' setTimeout(() => { // 全局locale變動,生效範圍爲項目全局 mpx.i18n.locale = 'zh-CN' }, 10000) }, computed: { formattedDatetime () { return this.$d(new Date(), 'long') } } }) </script>
複製代碼

定義 i18n 字典

項目構建時傳入 i18n 配置對象,主要包括語言字典和默認語言類型。

new MpxWebpackPlugin({
  i18n: {
    locale: 'en-US',
    // messages既能夠經過對象字面量傳入,也能夠經過messagesPath指定一個js模塊路徑,在該模塊中定義配置並導出,dateTimeFormats/dateTimeFormatsPath和numberFormats/numberFormatsPath同理
    messages: {
      'en-US': {
        message: {
          hello: '{msg} world'
        }
      },
      'zh-CN': {
        message: {
          hello: '{msg} 世界'
        }
      }
    },
    // messagesPath: path.resolve(__dirname, '../src/i18n.js')
  }
})
複製代碼

若是是經過 Mpx 提供的 cli 工具生成的項目,這部分配置會在 mpx.conf.js 文件中,不光能夠直接內聯寫在該文件中,也能夠指定語言包的路徑。

以上,Mpx 的 i18n 方案接入成本低,使用優雅,體驗優秀。直觀感覺可參考下面 mpx i18n demo :github.com/didi/mpx/tr…

方案

Mpx框架的 i18n 支持幾乎徹底實現了 vue-i18n 的所有能力,下面咱們來詳細說明 Mpx 框架 i18n 能力的具體實現。

方案探索

基於小程序運行環境的雙線程架構,咱們嘗試了不一樣方案,具體探索過程以下:

方案一:基於 Mpx 框架已提供的數據加強能力 computed 計算屬性,來支持 i18n 。該方案與 uniapp 的實現思路類似(後文會進行對比分析),存在必定不足,包括線程通訊帶來的性能開銷和for循環場景下的處理較複雜等,最終放棄。
方案二:基於 WXS + JS 支持 i18n 適配。經過視圖層注入 WXS,將 WXS 語法轉換爲 JS 後注入到邏輯層,這樣視圖層和邏輯層都可實現 i18n 適配,而且在必定程度上有效減小兩個線程間的通訊耗時,提升性能。

從性能和合理性上考慮,咱們最終採用了方案二進行 Mpx 的 i18n 方案實現。

mpx-i18n內部流程示意圖

Mpx i18n 架構設計圖

因爲各大小程序平臺上,WXS 語法和使用均存在較大差別,所以該方案實現過程當中也存在一些技術上的難點,這些難點基於 Mpx 框架的早期構建起來的跨平臺能力也一一得以攻克,具體以下。

實現難點

WXS 在模板中運行的跨平臺處理

WXS 是運行在視圖層中的 JS,能夠減小與邏輯層通訊耗時,提升性能。所以 Mpx 框架在迭代初期便已支持在模板和 JS 運行環境使用 WXS 語言,而且針對小程序跨平臺 WXS 語法進行抹平。 在模板中,Mpx 自定義一個 webpack chunk template,以微信 WXS 做爲 DSL,利用 babylon 將注入的 WXS 轉化成 ast,而後遍歷 ast 節點,抹平各大平臺對 WXS 語法的處理差別,輸出各平臺能夠識別的類 WXS 文件。目前主要支持微信(WXS)、支付寶(sjs)、百度(filter)、QQ(qs)、頭條(sjs)等小程序平臺。

WXS 在邏輯層運行的跨平臺處理

WXS 與 JavaScript 是不一樣的語言,有本身的語法,並不和 JavaScript 一致。而且 WXS 的運行環境和其餘 JavaScript 代碼是隔離的,WXS 中不能調用其餘 JavaScript 文件中定義的函數,也不能調用小程序提供的API。 所以在邏輯層,Mpx 將注入的 WXS 語法轉化爲 JS,經過 webpack 注入到當前模塊。例如 WXS 全局方法 getRegExp/getDate 在 JS 中是沒法調用的,Mpx將它們分別轉化成 JS 模塊,再經過 webpack addVariable 將模塊注入到 bundle.js 中。 同理,Mpx 會將編譯時注入的 i18n wxs 翻譯函數和 i18n 配置對象掛載到全局 global 對象上,利用 mixin 混入到頁面組件,並監聽 i18n 配置對象,這樣JS和模板中便可直接調用 i18n 翻譯函數,實現數據響應。

以上即是 Mpx 框架在小程序中支持 i18n 能力的技術細節,因爲 WXS 是能夠在視圖層執行的類 JS 語法的一門語言,這樣就減小了小程序邏輯層和視圖層的通訊耗時,提高性能。可是因爲實現依賴類 WXS 能力,以及 WXS 執行環境的限制,目前模板上可直接使用的翻譯函數包括 $t/$tc/$te ,若是須要格式化數字或日期可使用對應的翻譯函數在 JS 中 Mpx 提供的計算屬性中實現。

輸出 web 時使用 i18n

Mpx同時還支持轉換產出H5,而 Mpx 提供的 i18n 能力在使用上與 vue-i18n 基本一致,輸出 web 時框架會自動引入 vue-i18n,並使用當前的 Mpx i18n 配置信息對其進行初始化,用戶無需進行任何更改,便可輸出和小程序表現徹底一致的 i18n web 項目。

對比

上面分析了 Mpx 框架的 i18n 方案的技術細節,咱們來看下和其餘方案的對比,主要是和 uniapp - 基於 Vue 編寫小程序的方案,和微信官方的方案,二者提供的 i18n 支持與Mpx的對比有何優劣。

uniapp的方案

uniapp 提供了對 i18n 能力的支持,是直接引入vue-i18n。但小程序中沒法在模板上調用 JS 方法,本質上是利用計算屬性 Computed 轉換好語言,而後利用模板插值在小程序模板中使用。

模板中: <view>{{ message.hello }}</view>

JS裏須要寫:

computed: {  
    message () {  
      return { hello: this.$t('message.hello') }
    }
  }
複製代碼

所以該方案存在一個性能問題,最終的渲染層所看到的文本仍是經過 setData 跨線程通訊完成,這樣就會致使線程間通訊增多,性能開銷較大。

而且,早期這種形式使用成本較高,後來 uniapp 也針對其作過優化,實現了能夠在模板上寫 $t() 的能力,使用上方便了很多。

這個 t ( ) 的實現是在編譯時候識別到 t() 的實現是在編譯時候識別到 t 就自動替換,幫你替換成一個 uniapp 的 computed 數據,所以數據部分仍是和以前同樣要維護兩份。尤爲是模板上的for循環,即便 for 裏只有一個數據要被轉換,整個列表都要被替換成一個計算屬性,在線程間通訊時進一步加大了性能開銷。

微信官方的方案

微信小程序自己也提供了一個 i18n 的方案,倉庫地址是:wechat-miniprogram/miniprogram-i18n 。

這個方案從 i18n 自己的實現來說和Mpx框架的設計是相似的,也是基於 WXS 實現(英雄所見略同啊)。但由於周邊配套上沒有完整的體系,總體使用體驗上就也略遜於基於Mpx框架來開發支持 i18n 的國際化小程序了。

主要的點就是,官方提供的方案,要基於 gulp 工具進行一次額外構建,同時在JS中使用時候還要額外引入一個 behavior 去讓JS中也可使用翻譯能力。

而Mpx框架經過一次統一的Webpack構建產出完整的內容,用戶無需擔憂語言包更新後忘記從新構建,在JS中使用的時候不光更方便,並且語言信息仍是個響應式的,任何組件均可以很方便地監聽語言值的變化去作一些其餘的事情。

最後,Mpx的 i18n 方案對比微信官方的方案還有個巨大的優勢,結合Mpx的跨平臺能力,能實現均以這個方案,一套代碼產出支持微信/支付寶/百度/QQ/頭條多個平臺的支持 i18n 的小程序。

總結

Mpx 框架專一小程序開發,指望爲開發者提供最溫馨的開發體驗,有衆多優秀的功能特性,幫助開發者提效。本文介紹的是其內置的 i18n 能力,經過對比分析得出相比其餘框架方案在使用成本和性能等方面有明顯的優點,歡迎各位有相關需求的同窗進行體驗嘗試。

將來 Mpx 還會持續迭代優化,提供更多更好的能力幫助小程序開發者提效。在使用過程當中遇到任何問題,歡迎你們在 Git 上提 issue,團隊成員會及時響應。同時也鼓勵你們一塊兒爲開源社區作貢獻,參與到 Mpx 共建中來,爲小程序技術發展添磚加瓦。

GitHub地址 github.com/didi/mpx Mpx文檔 mpxjs.cn/

歡迎技術交流與反饋,順便star一下鼓勵開源項目貢獻者,咱們將持續發力貢獻社區。

附:以往Mpx文章連接
滴滴開源小程序框架Mpx - https://mpxjs.cn/articles/1.0.html
滴滴小程序框架Mpx發佈2.0,支持小程序跨平臺開發,可直接轉換已有微信小程序 - https://mpxjs.cn/articles/2.0.html
小程序開發者,爲何你應該嘗試下MPX - https://mpxjs.cn/articles/mpx1.html
Mpx 小程序框架技術揭祕 - https://mpxjs.cn/articles/mpx2.html
滴滴出行小程序體積優化實踐 - https://mpxjs.cn/articles/size-control.html

相關文章
相關標籤/搜索