前言:
最近在研究swift發現很久沒有更新文章了,看著時差那么久沒有更新有點急躁了,但又不知道要分享內容好一點又實用的文章,最近寫項目上線的次數比較頻繁,上線時對于Xcode的配置也是諸多操作,問題是,一些web服務的url指向了測試服務器,同時API密鑰被配置用于測試環境。在提交app給蘋果審核前,你需要修改所有這些API密鑰和URL以適應生產版本,相較于在開發環境和生產環境之間來回修改相關數值,對于這種繁瑣的操作和弊端想著要優化了,所以這兩天就去研究了一下Targets管理項目。
對于iOS開發的老鳥來說這已經不是事兒了,但對于初學者來說很多人就會覺得奇怪,為什么需要使用兩個單獨的數據庫和環境。原因是當你繼續構建新的功能特性或繼續開發你的應用,你希望區分開現有的公開版本和生產版本和預發布版本。
區分版本的方法一:
因為公司的開發團隊的嚴謹把數據庫的接口區分為公開環境、預發布環境、測試環境,那么就必須把項目使用三個獨立的環境區分開來。我之前使用的一個辦法就是定義一個全局的變量,它會將應用程序初始化為公開版本、測試版本、預發布版本。
Objective-C:
在PCH文件中定義成宏(這里就不教大家怎么新建配置PCH了)
要哪個環境就打開注釋,關閉另外兩個注釋
#define LocalEnvironment? ? ? ? //測試環境
//#define ReleaseEnvironment ? ? //預發布環境
//#define OfficialEnvironment? ? ? //公開環境
#ifdef LocalEnvironment
#define APIURL @"http://www.test.myapi.net"? ? //APIURL 為后臺接口
#endif?
#ifdef ReleaseEnvironment
#define APIURL @"http://www.release.myapi.net"
#endif
#ifdef OfficialEnvironment
#define APIURL @"http://www.myapi.net"
#endif
Swift3.0后:
新建一個 Swift File 文件,定義成全局常量
enum environmentType {
case local, release, official ? ? ? //測試環境,預發布環境,公開環境
}
//給測試人員的時候切換他們想要的環境,上線的時候切換official環境
let environment: environmentType = .official ? ?
let APIURL: String = ""? ? ? ? ? ? //APIURL 為后臺接口
switch environment {
case .local:
APIURL = "http://www.test.myapi.net" ? ? ? ??
case .release:
APIURL = "http://www.release.myapi.net"
case .official:
APIURL = "http://www.myapi.net"?
}?
弊端:這種方法需要你每次切換環境時改變全局變量。雖然這種方法也許快捷,方便,但是它有一些重要的限制。首先,因為我們在開發和生產兩個環境中使用一個Bundle ID,你不能在一臺設備上安裝應用的兩個版本。當你需要要測試開發版本的應用程序時,同時仍在該設備上使用生產版本的應用,這就變的不方便了。此外,這種方法很有可能將應用的開發版本上傳到應用商店。如果你忘記了改變這個全局變量,你將會上傳錯誤的應用給你的用戶。
區分版本的方法二:
創建新的Target,我將使用一個事例項目引導學者們一步步探討整個過程(通用Objective-C、Swift)
1.首先新建一個項目取名Happy,可以看到只有一個TARGETS,我們把這個target作為公開版本。
2.配置測試版本和預發布版本。
? ?3.選擇 Duplicate Only,需要新建兩個因為需要再新建兩個版本
4.現在打開你開發target的“Build Settings”,滾動到“Packaging”,并將值改為開發的plist文件。注意:這三個targets對應的代碼相同,但Plist和設置是不同的
5.可選操作:但強烈推薦。如果你想簡單地區分公開版本和預發布版本構建,你應該為每個版本使用單獨的icon和啟動頁。這將使測試人員更清晰地知道正在使用哪個app,防止上傳開發版本。
跳到 `Assets.xcassets` 添加一個新的圖標。右擊圖標 > App Icons & Launch Images > New iOS App Icon. 新圖標重命名為“AppIcon-Release”同時添加自己的圖片。
現在回到項目設置,選擇您的開發target,并改變Bundle Identifier。你可以簡單地將“Release”追加到原來的ID上。如果執行了上一步驟,請確保更改應用app icon,設置為在上一步中創建的。(PS:改變Bundle Identifier的話需要重新去官網弄好區分這個Bundle Identifier的測試證書)
6.最后,我們會為生產和開發target配置預處理宏/編譯器標識。之后我們就可以使用該標識在我們的代碼來檢測應用程序正在運行的版本。
Objective-C
去到`Build Settings`下`Apple LLVM 7.0 - Preprocessing`。拓展`Preprocessor Macros`在Rebug和Release區域添加一個變量。對于公開版本target(即Happy),將該值設置為`DEVELOPMENT=2`。對于預發布版本target(即Happy Release),將該值設置為`DEVELOPMENT=1`。對于測試版本target(即Happy Test),將該值設置為`DEVELOPMENT=0`。
Swift
編譯器不再支持預處理指令。作為替代,它使用編譯時的屬性和build配置。選中開發target,添加一個標識表示開發版本。找到`Build Setting`往下滾動到`Swift Compiler - Custom Flags`部分。將值設為`-DTEST`表示這個target作為測試版本。將值設為`-DRELEASE`表示這個target作為預發布版本。將值設為`-DOFFICIAL`表示這個target作為公開版本。
使用Target和宏
根據已配置的宏DEV_VERSION,我們可以在代碼中利用它動態地編譯項目。下面是一個簡單的例子:
Objective-C: 在PCH文件中
#if DEVELOPMENT==0
#define LocalEnvironment ? ? //測試
#elif DEVELOPMENT==1
#define ReleaseEnvironment //預發布
#elif DEVELOPMENT==2
#define OfficialEnvironment ? ?//公開
#endif
#ifdef LocalEnvironment
#define APIURL @"http://www.test.myapi.net"? ? //APIURL 為后臺接口
#endif
#ifdef ReleaseEnvironment
#define APIURL @"http://www.release.myapi.net"
#endif
#ifdef OfficialEnvironment
#define APIURL @"http://www.myapi.net"
#endif
Objective-C中你可以使用`#if`檢查`DEVELOPMENT`的環境,并相應的設置URLs/ API密鑰。
Swift: 在新建的Swift File文件中,聲明成全局常量
#if TEST ? ? ? ? ? ? ? ? ? ? //測試
let SERVER_URL = "http://www.test.myapi.net"
#elseif RELEASE ? ? ? //預發布
let SERVER_URL = "http://www.release.myapi.net"
#else ? ? ? ? ? ? ? ? ? ? ? ? ? //公開
let SERVER_URL = "http://www.myapi.net"
#endif
像這樣
Swift中你仍然可以使用`#if`判定build的參數動態編譯。然而,除了使用`#define`定義基本常量,在swift中我們也可以用`let`定義一個全局常量。
驗證:
最后,管理多個target的一些注意事項:
1.當你添加新的文件到項目中,不要忘記選擇兩個target,以保持你的代碼同步在這三個版本。
2.如果你使用的CocoaPods,不要忘了添加新的target到你的Podfile中。
Build setting搜索user header search paths添加 ?${SRCROOT}(表示當前目錄) 防止Import的時候出現錯誤
多Target控制項目的版本操作已經完成,喜歡的朋友動動手指喜歡一下,發現有錯誤希望能和大家溝通提點一下,謝謝大家繼續支持。