新聞中心
本文轉(zhuǎn)載自微信公眾號(hào)「程序員魚(yú)皮」,作者魚(yú)皮 。轉(zhuǎn)載本文請(qǐng)聯(lián)系程序員魚(yú)皮公眾號(hào)。

創(chuàng)新互聯(lián)公司長(zhǎng)期為1000多家客戶(hù)提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊(duì)從業(yè)經(jīng)驗(yàn)10年,關(guān)注不同地域、不同群體,并針對(duì)不同對(duì)象提供差異化的產(chǎn)品和服務(wù);打造開(kāi)放共贏平臺(tái),與合作伙伴共同營(yíng)造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為湯旺企業(yè)提供專(zhuān)業(yè)的成都做網(wǎng)站、成都網(wǎng)站設(shè)計(jì),湯旺網(wǎng)站改版等技術(shù)服務(wù)。擁有十年豐富建站經(jīng)驗(yàn)和眾多成功案例,為您定制開(kāi)發(fā)。
大家好,我是魚(yú)皮,今天給大家分享企業(yè)項(xiàng)目開(kāi)發(fā)的重要知識(shí) —— 多環(huán)境。
本文大綱:
魚(yú)皮 - 多環(huán)境技術(shù)大綱
什么是多環(huán)境?
先思考一個(gè)問(wèn)題。
假如我們有一個(gè)數(shù)百萬(wàn)用戶(hù)正在用的網(wǎng)站,網(wǎng)頁(yè)文件部署在幾臺(tái)服務(wù)器上。那現(xiàn)在我們要開(kāi)發(fā)上線一個(gè)新功能,應(yīng)該怎么做呢?
老弟小阿巴問(wèn):寫(xiě)好代碼后,直接更新服務(wù)器上的網(wǎng)頁(yè)文件么?
我一巴掌抽過(guò)去:那萬(wàn)一你的代碼有 Bug,不就影響到線上用戶(hù)的使用了么?
老弟思考了下:那寫(xiě)好代碼后,在本地測(cè)試運(yùn)行沒(méi)有問(wèn)題后,再發(fā)布上線?
我:思路不錯(cuò),但問(wèn)題在于,如果本地和線上運(yùn)行項(xiàng)目時(shí),連接的是同一個(gè)數(shù)據(jù)庫(kù),那么當(dāng)你在本地測(cè)試向數(shù)據(jù)庫(kù)中插入亂七八糟的假數(shù)據(jù)、或者修改數(shù)據(jù)庫(kù)表結(jié)構(gòu)時(shí),不就會(huì)影響到線上的數(shù)據(jù)了么?
老弟一拍手:對(duì)哦,那如何讓本地的測(cè)試不影響到線上項(xiàng)目呢?
這就需要 多環(huán)境 。根據(jù)實(shí)際需要,將同一個(gè)項(xiàng)目(或同一套代碼)按照一定方法進(jìn)行區(qū)分,并將所需資源和項(xiàng)目本身部署到不同的機(jī)器上。不同環(huán)境的項(xiàng)目可以有 不同的行為 ,且能夠 同時(shí)存在、互不影響 。
舉個(gè)例子,可以給線上項(xiàng)目搭建一套開(kāi)發(fā)環(huán)境,開(kāi)發(fā)環(huán)境的數(shù)據(jù)存儲(chǔ)在獨(dú)立的開(kāi)發(fā)數(shù)據(jù)庫(kù),并且為了調(diào)試方便,不需要登錄也能夠訪問(wèn)所有的用戶(hù)數(shù)據(jù):
這樣一來(lái),本地和線上的項(xiàng)目就完全隔離開(kāi)了,開(kāi)發(fā)者在本地想怎么折騰就怎么折騰!這便是多環(huán)境的好處。
常用環(huán)境
多環(huán)境聽(tīng)起來(lái)雖然挺爽的,但事實(shí)上,環(huán)境不是區(qū)分的越多越好!
一方面是搭建多環(huán)境需要額外的工作量;另一方面是項(xiàng)目依賴(lài)的資源越多,成本就越高,而且維護(hù)起來(lái)也更麻煩。
因此,企業(yè)中常用的環(huán)境也就那么幾種,都快成為一種約定俗成的規(guī)范了,下面給大家介紹一下。
不同團(tuán)隊(duì)區(qū)分環(huán)境的方式可能不同,僅供參考。
本地環(huán)境
一般用 local 標(biāo)識(shí),是指前端或后端獨(dú)立開(kāi)發(fā)、自主測(cè)試的環(huán)境。通常就是讓項(xiàng)目和依賴(lài)在我們自己的電腦上運(yùn)行,比如數(shù)據(jù)庫(kù)、緩存、隊(duì)列等各種服務(wù),可能需要自己在本地搭建。
本地環(huán)境
開(kāi)發(fā)環(huán)境
一般用 dev 標(biāo)識(shí),是指前端和后端(或者多個(gè)程序員)一起協(xié)作開(kāi)發(fā)、聯(lián)調(diào)的環(huán)境。通常將項(xiàng)目和依賴(lài)放在員工電腦可以直接訪問(wèn)的開(kāi)發(fā)機(jī)上,不用自己搭建,直接跑起來(lái)項(xiàng)目,提高開(kāi)發(fā)和協(xié)作效率。
對(duì)規(guī)模不大的團(tuán)隊(duì)來(lái)說(shuō),開(kāi)發(fā)和本地環(huán)境其實(shí)有一套就夠了,畢竟本地也可以連接公用的數(shù)據(jù)庫(kù)等服務(wù)。
開(kāi)發(fā)環(huán)境
測(cè)試環(huán)境
一般用 test 標(biāo)識(shí),是指前端和后端開(kāi)發(fā)和聯(lián)調(diào)完成,做出完整的新功能后,交給測(cè)試同學(xué)去找 Bug 的環(huán)境。
通常在測(cè)試環(huán)境需要有獨(dú)立的測(cè)試數(shù)據(jù)庫(kù)和其他服務(wù),讓測(cè)試同學(xué)大顯身手。每次修改完 Bug 后,也都要再次發(fā)布項(xiàng)目到測(cè)試環(huán)境,讓測(cè)試同學(xué)重新驗(yàn)證。
測(cè)試環(huán)境
預(yù)發(fā)布環(huán)境
一般用 pre 標(biāo)識(shí),這是和線上項(xiàng)目最接近的環(huán)境,一般是測(cè)試驗(yàn)證通過(guò)、產(chǎn)品經(jīng)理體驗(yàn)過(guò)后,才能將項(xiàng)目發(fā)布到這個(gè)環(huán)境。
實(shí)際上,預(yù)發(fā)布環(huán)境的項(xiàng)目調(diào)用的后端接口、連接的數(shù)據(jù)庫(kù)、服務(wù)等都 和線上項(xiàng)目一致 ,和線上唯一的區(qū)別就是前端訪問(wèn)的域名不同。
正因如此,預(yù)發(fā)布環(huán)境看到的都是真實(shí)的用戶(hù)數(shù)據(jù),可以發(fā)現(xiàn)更多測(cè)試環(huán)境因?yàn)閿?shù)據(jù)不足而沒(méi)查出來(lái)的 Bug。
預(yù)發(fā)布環(huán)境
生產(chǎn)環(huán)境
一般用 prod 標(biāo)識(shí),又叫線上環(huán)境,是給所有真實(shí)用戶(hù)使用的環(huán)境。
因此不能隨意修改,且發(fā)布項(xiàng)目到該環(huán)境時(shí)必須格外小心。線上的數(shù)據(jù)庫(kù)、機(jī)器等資源一般也是由專(zhuān)業(yè)的運(yùn)維來(lái)負(fù)責(zé),想要登錄機(jī)器、修改配置,都需要經(jīng)過(guò)嚴(yán)格審批。
生產(chǎn)環(huán)境
如何實(shí)現(xiàn)?
最后再介紹下多環(huán)境的實(shí)現(xiàn)方式,其實(shí)大同小異,遵循 3 個(gè)步驟:抽象配置類(lèi) + 配置文件化 + 注入環(huán)境參數(shù),就能輕松實(shí)現(xiàn)。
抽象配置類(lèi)
將項(xiàng)目代碼中需要根據(jù)環(huán)境的變化而更改的變量整理到一個(gè)或多個(gè)配置類(lèi)中,集中管理。
舉個(gè)例子,連接數(shù)據(jù)庫(kù)時(shí),我們需要數(shù)據(jù)庫(kù) IP、端口、配置等信息,代碼如下:
- // 數(shù)據(jù)庫(kù)基本信息
- DB db = new DB();
- db.setIp("10.0.0.1");
- db.setPort(3306);
- // 數(shù)據(jù)庫(kù)連接配置
- DBConnection c = new DBConncetion();
- c.setTimeout(1000);
我們可以將這些代碼中寫(xiě)死的值全部替換成變量,將同類(lèi)變量放到一個(gè)類(lèi)中:
- // 數(shù)據(jù)庫(kù)配置類(lèi)
- class DBConfig {
- String ip = "10.0.0.1";
- int port = 3306;
- long timeout = 1000L;
- }
然后從這個(gè)類(lèi)中讀取變量的值:
- DB db = new DB();
- DBConfig cf= new DBConfig()
- // 從類(lèi)中獲取
- db.setIp(cf.getIp());
- db.setPort(cf.getPort());
- DBConnection c = new DBConncetion();
- c.setTimeout(cf.getTimeout());
這樣的好處是,如果代碼中還有其他地方用到了這些變量,也都可以從同一個(gè)類(lèi)去獲取,而不是把 死值 重復(fù)寫(xiě)多次,難以維護(hù)。
配置文件化
我們可以用專(zhuān)門(mén)的配置文件來(lái)維護(hù)配置,從而讓用戶(hù)修改配置更方便,不用再去找代碼、改代碼。
常見(jiàn)的配置文件格式有 properties、yaml、yml、json 等,比如新建一個(gè)數(shù)據(jù)庫(kù)配置文件 db.properties :
- db.ip=10.0.0.1
- db.port=3306
- db.timeout=1000
接下來(lái)在初始化數(shù)據(jù)庫(kù)時(shí),就可以將配置文件中的值加載到上一步寫(xiě)好的配置類(lèi)中,然后讀取啦:
- // 從文件讀取配置的值
- DBConfig cf = new DBConfig("db.properties");
- db.setIp(cf.getIp());
- db.setPort(cf.getPort());
- ...
其實(shí)只不過(guò)是把配置的值從代碼中移到了文件中而已。
但這樣一來(lái),我們想加載哪個(gè)配置文件就能加載哪個(gè)!
比如要搞一套測(cè)試環(huán)境的配置,只需再新建一個(gè) db-test.properties 文件(文件名中加個(gè)環(huán)境名稱(chēng)),就能在這個(gè)文件中編寫(xiě)?yīng)毩⒌呐渲昧耍缓笤诖a中加載該文件即可:
- new DBConfig("db-test.properties");
無(wú)論是前端還是后端,大部分的多環(huán)境實(shí)現(xiàn)都是這個(gè)原理 —— 搞多套配置,所以總能在項(xiàng)目中看到類(lèi)似的配置文件:
多環(huán)境配置文件
注入環(huán)境參數(shù)
到目前為止,其實(shí)我們還是在代碼中寫(xiě)了 死值 ,來(lái)告訴程序應(yīng)該加載哪個(gè)名稱(chēng)的配置文件。
比如在本地開(kāi)發(fā)時(shí),加載 db-dev.properties ,開(kāi)發(fā)完成后、正式上線前,再改代碼為加載 db-prod.properties。
但這樣不僅麻煩,而且可能忘了修改,把開(kāi)發(fā)環(huán)境的項(xiàng)目發(fā)布到了線上。
最理想的效果應(yīng)該是:無(wú)論項(xiàng)目要切換到哪個(gè)環(huán)境,整個(gè)項(xiàng)目都完全不用修改。
因此,我們可以將 指定環(huán)境 這件事放到最后,在通過(guò)命令去打包或者啟動(dòng)項(xiàng)目時(shí),將環(huán)境參數(shù)寫(xiě)進(jìn)去。
舉個(gè)例子,我們?cè)趩?dòng) java 項(xiàng)目時(shí),給 env 系統(tǒng)變量傳遞不同參數(shù):
- # 測(cè)試環(huán)境
- java -jar -Denv=test dist.jar
- # 生產(chǎn)環(huán)境
- java -jar -Dend=prod dist.jar
然后在程序中讀取該參數(shù),加載對(duì)應(yīng)的配置即可:
- // 讀取 env 參數(shù)
- String env = System.getProperty("env");
- new DBConfig("db-" + env + ".properties");
同理,對(duì)于前端項(xiàng)目,可以在打包構(gòu)建時(shí)傳入環(huán)境變量,然后自己在代碼中讀取,或者交給 Webpack 之類(lèi)的打包工具處理:
- {
- "scripts": {
- "serve": "env=dev serve",
- "build:test": "env=test build"
- "build": "env=prod build"
- }
- }
網(wǎng)頁(yè)標(biāo)題:企業(yè)項(xiàng)目開(kāi)發(fā)的重要知識(shí)之多環(huán)境
網(wǎng)頁(yè)網(wǎng)址:http://m.fisionsoft.com.cn/article/dhpeoeg.html


咨詢(xún)
建站咨詢(xún)
