R.Swift - iOS高效资源引用框架探究

前言

Android中的资源引用 & "R"

在接触iOS开发之前,我也曾搞过一段时间的Android开发。在Android开发中,其对于资源的引用非常方便,我们可以通过"R机制"来对Android中的图片(Drawable)、布局(Layout)、动画配置(Anim)、国际化字符串(String)、尺寸(Dimen)等等资源进行非常方便快捷的引用。什么是R机制?在我们创建一个Android项目的时候,IDE会自动帮我们创建一个名为R的类型,它所在的文件名称也是叫做R.javaR类型中没有任何方法,包含的是代表不同类型资源的内部静态类,而这些内部静态类中,也只有静态的属性,每个属性代表一个资源,故我们要引用某个资源类型中的某个资源,可用R.资源类型.资源名来引用。下面就是代码中的实例:

//  在Activity中设置其自己的视图布局
RootActivity.this.setContentView(R.layout.activity_root);

//  从图片资源文件夹中加载名为"image_test"的图片以其创建位图
Bitmap aBitmap = BitmapFactory.decodeResource(this.getResources(), R.drawable.image_test);

//  以id为基准找到某个视图控件
this.showTextView = (TextView)this.findViewById(R.id.show_text_view);

巧妙的是,我们无需对R类进行任何的修改或定义,IDE会帮我们自动完成:当我们把一张图片复制入drawable文件夹中,R类的drawable内部类会自动声明并定义了一个名字与图片相同的静态属性,属性的类型为int,用于对资源的唯一标识,同样的,当我们创建一个layout布局文件,或者在定义colordimensstringxml配置文件中新增新配置资源,IDE都会根据资源的改变自动配置R类。

iOS中的资源引用

当我从Android转至iOS后,我瞬间觉得iOS的资源引用略为恶心,跟Android比起来,它更加蛋疼麻烦,举几个例子:

  1. 我现在Assets图片库中有一张图片,名为image_test,好,我现在要创建一个UIImage对象,让其解码这张图片。代码是这样子的:
let aImage = UIImage(named: "image_test")

在这里,我使用的UIImage构造方法需要传入一个字符串参数,这个字符串要求为图片的名字。在调用这个构造方法的时候,我的心里有股蛋蛋的忧伤... what? 我要自己输入一个字符串,才能构造出这个实例?就算我不手打,直接复制图片名字也好,我还要跳到Finder或者Assets中“Command + C”再回到代码区中“Command + V”,不仅如此,UIImage(named:)构造方法所返回的类型是Optional类型,原因是我在输入图片名字构造的时候可能误将名字打错了,又或者图片在之前不小心被我删除了,Swift不能保证构造出的UIImage一定非空... 略坑😑

  1. 很多时候,我们会用到Nib来自定义一个视图,比如UITableViewCell;当我们在Nib中布局配置好视图后,我们就需要在代码区中引用它,下面就是UITableView注册一个以Nib构建的Cell的例子:
let tableView = UITableView(frame: CGRectZero, style: .Plain)
tableView.registerNib(UINib(nibName: "TanTableViewCell", bundle: nil), forCellReuseIdentifier: ViewController.CELL_IDENTIFIER)

首先我在之前已经创建好了一个名为TanTableViewCellNib,然后我们在为UITableViewC注册Cell的时候,则需要以我们创建的Cell的Nib文件名字符串为参数传入UINib的构造方法中。这里的问题跟上面所说的创建UIImage一样,使用字符串作为参数构建某些资源实例时,效率会较为低下,并且由于输入的错误,最终或许会导致实例无法构建,这是iOS开发API中一处蛋疼的地方。

R.Swift? 什么东东?

根据前言中关于Android资源引用的内容,顾名思义,R.Swift是一款基于Swift平台,针对iOS以及TVOS开发的仿Android资源引用框架。它所针对的问题,就是类似于刚刚我在上方提到的一样,避免使用字符串来构造某些资源实例。R.Swift能够在我们修改项目资源后动态地进行配置,最终,你能够使用类似语法R.资源类型.资源名称来对某资源进行引用构建。R.Swift有着动态生成代码的机制,由此,它具有以下极具魅力的优点:

  • 代码自动补全 :就像输入其他的代码一样,R.Swift支持IDE的代码自动补全,当你的资源量非常庞大时,你只需把资源调用语法的前沿部分敲出来,IDE就会自动给予代码提示以及补全,这对资源的引用效率有着显著的提高。
  • 自动检测 : 当我们写入某些错误的代码时,IDE会向我们反馈错误以及警告,如我们调用某个类中不存在的方法时,IDE就会立即检测出代码的错误,并给予反馈。在R.Swift中,若我们将资源重命名,或干脆将资源删除了,R.Swift会自动重新进行配置,而在之前我们对该资源进行引用的地方,IDE也会相应的向我们报错,表明资源的不存在,如此一来,我们就不必担心资源的修改删除对代码的蝴蝶效应了。

总之一句话,R.Swift猴犀利!

R.Swift的安装配置

在这里我们使用的是CocoaPods来对R.Swift进行安装,相关的步骤也不必累赘说明,一句pod install后我们把R.Swift安装进来了。下面要进行R.Swift的配置,主要有几个步骤:

  1. 进入项目的配置界面,在左边的TARGETS项下面选择我们的项目,并在右边点击Build Phases这个tab。
  2. 进入Build Phases这个tab后,我们看到左上角有一个"+"按钮,点击并在弹出的选项卡中选择New Run Script Phase
  3. 我们会看到界面的下方多出了一个Run Script项,展开它,并在脚本输入区域输入"$PODS_ROOT/R.swift/rswift" "$SRCROOT" (第二对双引号括起来所代码的是项目的根目录,你也可以放到根目录下的其他目录中,只需将其修改为"$SRCROOT/XXX",XXX为目标目录名)。
  4. 我们按住新建的这个Run Script项向上移动,移到Compile Source项的上方,不过也要保证此时它也在Check Pods Manifest.lock项的下方。
  5. Command + B,编译一下,编译成功后,在Finder进入到刚刚我们制定的目录中,此时我们会看到一个名为R.generated.swift的文件已经创建了,直接把此文件拖入Xcode项目中,记住不要勾选Copy items if needed项。
  6. 配置到此完成,我们可以构建自己的项目了

R.Swift对资源修改并编译后都会在R.generated.swift文件中自动生成代码,我们也可以打开这个文件看一下现在资源的构造情况,这里只截了一小段代码展示:

  /// This `R.font` struct is generated, and contains static references to 0 fonts.
  struct font {
    private init() {}
  }
  
  /// This `R.image` struct is generated, and contains static references to 1 images.
  struct image {
    /// Image `kafei`.
    static let kafei = ImageResource(bundle: _R.hostingBundle, name: "kafei")
    
    /// `UIImage(named: "kafei", bundle: ..., traitCollection: ...)`
    static func kafei(compatibleWithTraitCollection traitCollection: UITraitCollection? = nil) -> UIImage? {
      return UIImage(resource: R.image.kafei, compatibleWithTraitCollection: traitCollection)
    }
    
    private init() {}
  }
  
  /// This `R.nib` struct is generated, and contains static references to 1 nibs.
  struct nib {
    /// Nib `TanTableViewCell`.
    static let tanTableViewCell = _R.nib._TanTableViewCell()
    
    /// `UINib(name: "TanTableViewCell", bundle: ...)`
    static func tanTableViewCell(_: Void) -> UINib {
      return UINib(resource: R.nib.tanTableViewCell)
    }
    
    private init() {}
  }

R.Swift的使用

在这里要说明一下,每当我们修改了资源,我们需要Command + B来编译一下项目从而让R.Swift自动进行配置更新。


下面就来用实例演示一下R.Swift的使用,并与原生的API进行比对:

Image - 图片

//  不使用R.Swift
let pImage = UIImage(named: "image_test")
//  使用R.Swift
let nImage = R.image.image_test()

File - 数据文件

//  不使用R.Swift
let pFile = NSBundle.mainBundle().pathForResource("DataFile", ofType: "json")
//  使用R.Swift
let nFile = R.file.dataFileJson.path()

Font - 字体

//  不使用R.Swift
let pFont = UIFont(name: "chalkduster", size: 35)
//  使用R.Swift
let nFont = R.font.chalkduster(size: 35)
//  你看,非常神奇,在上面的方法中你不仅可以选择字体类型,还能设置字体大小

Color - 颜色

颜色这里我觉得有必要细讲一下,一个较为庞大的项目,其颜色资源配置也是相应比较复杂,如某些视图的背景颜色、字体的颜色等等,以往我们进行颜色资源的配置,一般会定义一个全局的Config文件或类,里面就存有各种颜色,每种颜色用有意义的名称去标识。R.Swift与其不同的是,它还能有更赞的方式导入颜色资源:

Color Palette

呵呵,这个方式可以说是非常的花式,配合R.Swift使用起来简直6得飞起!
当我们在可视化视图编辑界面(如Storyboard)中为一个视图配置某些颜色时,我们可以进入颜色调色板来选择更多的颜色,如图:

调色板

在其中,我们可以创建自己的调色板,如现在我创建了一个名为MyAppColor的调色板:
自定义调色板

这个自定义的调色板是不局限于现在的项目,以后我们创建的每一个项目都可以使用这个调色板。如果你的UI设计师够醒目的话,他也可以给我们提供一个设计师调好颜色的调色板,现在问题来了,设计师在他的电脑里定义好了调色板,怎么share出去呢?其实我们每自定义一个调色板,它都会以二进制文件的形式储存在电脑中,位置在~/Library/Colors/(隐藏文件目录,要访问它要不取消Finder隐藏,要不直接控制台open跳进来),文件的名字为调色板名.clr,把它直接copy下来就行。
R.Swift能够非常花式地使用调色板,有多花式?
就现在来说,我直接把我创建的调色板的文件MyAppColor.clr拖进项目中,Command + B编译下,然后,我现在可以这样子得到颜色了:

let appRedColor = R.color.myAppColor.red()

简直6到飞起!🌚

Nib

//  不使用R.Swift
tableView.registerNib(UINib(nibName: "TanTableViewCell", bundle: nil), forCellReuseIdentifier: CELL_IDENTIFIER)
//  使用R.Swift
tableView.registerNib(R.nib.tanTableViewCell(), forCellReuseIdentifier: CELL_IDENTIFIER)

UITableView中的复用

上面的代码让TableView注册了其cell所属的Nib,并输入了cell的复用标识符,而在另一种情况下,比如我们在设计cell的Nib时已经在Nib中设置好了cell的复用标识符,我们就没必要在TableView注册cell的时候再配置一遍了,幸亏,R.Swift给我们提供了高效率的cell注册以及复用方法:

  1. 首先在Cell的Nib中设置好复用标识符(Identifier)
  2. 进行Cell的注册
tableView.registerNib(R.nib.tanTableViewCell)
  1. 编写TableView数据源的Cell返回方法
func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCellWithIdentifier(R.nib.tanTableViewCell.identifier, forIndexPath: indexPath)
    //  Do Something
    return cell
}

R.Swift也支持对UICollectionView中Cell的高效注册与复用,方式跟UITableView也是差不多,在这里就不累赘说明了。


以上说明了使用R.Swift进行各种资源的引用构造,除了上面的资源外,R.Swift还能作用于Localized string(本地化字符串)StoryboardsSegues这些资源,其大大提高了iOS开发的资源引用构造效率。

运行时检测

R.Swift提供运行时检测功能,在项目运行的时候能够检测其生成的代码是否和此时项目的资源完全匹配,调用的是R.assertValid(),这个方法在release模式下是不做任何操作的,它只作用于debug(调试)模式,检测的内容为:

  • 是否所有在Storyboard中使用的图片都可用。
  • 是否所有定义在Storyboard中,且需要用标识符去加载的视图控制器都能够成功加载。
    官方建议把这个检测方法放到项目的AppDelegate中。

相关链接

R.Swift的Github地址

Github - R.Swift

参考资料

R.Swift官方说明文档

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

推荐阅读更多精彩内容

  • 转载自:https://github.com/Tim9Liu9/TimLiu-iOS 目录 UI下拉刷新模糊效果A...
    袁俊亮技术博客阅读 11,904评论 9 105
  • 嗯哼嗯哼蹦擦擦~~~ 转载自:https://github.com/Tim9Liu9/TimLiu-iOS 目录 ...
    philiha阅读 4,835评论 0 6
  • 你买了件好衣服,买了双好鞋,却总等着“大日子”才穿,最后衣服旧了,鞋子过时了。我不明白你为什么只有在特定的日子才想...
    微尘九回阅读 255评论 0 0