Swift GYB

作者: iOS_小久 | 来源:发表于2019-06-17 22:04 被阅读50次

术语 “样板” 可以追溯到印刷媒体的早期阶段。小型地区性报纸的填充量只有几英寸,但通常缺乏编写人员来实现这一目标,因此他们中的许多人转向大型印刷品集团,以获得稳定的内容流,可以逐字添加到他们日报的后页。这些故事通常在预设板上提供,这类似于用于制造锅炉的轧制钢板,因此得名。

通过转喻的过程,内容本身被称为“样板”,并且该概念被用于包含合同中的标准化公式文本,形式字母,以及与本周关于NSHipster的代码的文章最相关。

来小编交流群 有技术的来闲聊 没技术的来学习 691040931

并非所有代码都是迷人的。实际上,许多使一切工作的低级基础设施都是一个模板。

这是斯威夫特标准库,其中包括像符号整数类型的家庭(真Int8Int16Int32Int64),其实现仅在相应类型的大小而变化。

复制粘贴代码可以作为一次性解决方案(假设您第一次设法正确),但它不可持续。每当您想要对这些派生实现进行更改时,您都可能会引入轻微的不一致性,导致实现随着时间的推移而发生分歧 - 与负责地球生命变化的随机突变不同。

语言有各种技术来处理这个问题,从C ++模板和Lisp宏到evalC预处理器语句。

Swift没有宏系统,并且因为标准库本身是用Swift编写的,所以它无法利用C ++元编程功能。相反,Swift维护者使用一个名为gyb.py的Python脚本,使用一 小组 模板标签生成源代码。

<abbr title="生成您的Boilerplate" style="box-sizing: border-box;">GYB</abbr> 是“Generate Your Boilerplate”的首字母缩写,参考 另一个Python工具, <abbr title="Generate Your Projects" style="box-sizing: border-box;">GYP</abbr> (“生成你的项目”) <abbr title="生成您的项目" style="box-sizing: border-box;"></abbr> <ins style="box-sizing: border-box; background-color: rgba(255, 128, 0, 0.2); text-decoration: none;">Haskell包 <abbr title="报废你的锅炉" style="box-sizing: border-box;">SYB</abbr> (“Scrap Your Boilerplate”) <abbr title="报废你的锅炉" style="box-sizing: border-box;"></abbr></ins>。

GYB如何运作

GYB是一个轻量级模板系统,允许您使用Python代码进行变量替换和流控制:

  • 序列评估一段Python代码%{ <var class="placeholder" style="box-sizing: border-box; background: rgb(248, 248, 242); display: inline-block; padding: 0px 0.25em; border: 1px solid rgb(40, 42, 54); border-radius: 5px; font-style: normal;">code</var> }
  • 序列管理控制流程% <var class="placeholder" style="box-sizing: border-box; background: rgb(248, 248, 242); display: inline-block; padding: 0px 0.25em; border: 1px solid rgb(40, 42, 54); border-radius: 5px; font-style: normal;">code</var>: ... % end
  • 该序列替换表达式的结果${ <var class="placeholder" style="box-sizing: border-box; background: rgb(248, 248, 242); display: inline-block; padding: 0px 0.25em; border: 1px solid rgb(40, 42, 54); border-radius: 5px; font-style: normal;">code</var> }

所有其他文本都保持不变。

可以在Codable.swift.gyb中找到GYB的一个很好的例子。在文件的顶部,基Codable类型被分配给实例变量:

%{
codable_types = ['Bool', 'String', 'Double', 'Float',
                 'Int', 'Int8', 'Int16', 'Int32', 'Int64',
                 'UInt', 'UInt8', 'UInt16', 'UInt32', 'UInt64']
}%

稍后,在实现中,迭代这些类型以生成协议要求的方法声明:Single<wbr style="box-sizing: border-box;">Value<wbr style="box-sizing: border-box;">Encoding<wbr style="box-sizing: border-box;">Container

% for type in codable_types:
  mutating func encode(_ value: ${type}) throws
% end

评估GYB模板会产生以下声明:

mutating func encode(_ value: Bool) throws
mutating func encode(_ value: String) throws
mutating func encode(_ value: Double) throws
mutating func encode(_ value: Float) throws
mutating func encode(_ value: Int) throws
mutating func encode(_ value: Int8) throws
mutating func encode(_ value: Int16) throws
mutating func encode(_ value: Int32) throws
mutating func encode(_ value: Int64) throws
mutating func encode(_ value: UInt) throws
mutating func encode(_ value: UInt8) throws
mutating func encode(_ value: UInt16) throws
mutating func encode(_ value: UInt32) throws
mutating func encode(_ value: UInt64) throws

此图案用于在整个文件中,以产生类似的方法类似地公式化的声明 ,和。总的来说,GYB将样板代码的数量减少了几千个LOC:encode(_:for<wbr style="box-sizing: border-box;">Key:)``decode(_:for<wbr style="box-sizing: border-box;">Key:)``decode<wbr style="box-sizing: border-box;">If<wbr style="box-sizing: border-box;">Present(_:for<wbr style="box-sizing: border-box;">Key:)

$ wc -l Codable.swift.gyb
2183 Codable.swift.gyb
$ wc -l Codable.swift
5790 Codable.swift

有效的GYB模板可能无法生成有效的Swift代码。如果派生文件中发生编译错误,则可能难以确定根本原因。

安装GYB

GYB不是标准Xcode工具链的一部分,因此您无法找到它xcrun。相反,您可以使用Homebrew下载它:

$ brew install nshipster/formulae/gyb

或者,您可以下载源代码并使用chmod命令生成gyb可执行文件(macOS上的默认安装应该能够运行gyb):

$ wget https://github.com/apple/swift/raw/master/utils/gyb
$ wget https://github.com/apple/swift/raw/master/utils/gyb.py
$ chmod +x gyb

如果你走这条路线,一定要将它们移动到可以从Xcode项目访问的地方,但要将它们与源文件分开(例如,Vendor项目根目录下的目录)。

在Xcode中使用GYB

在Xcode中,单击导航器中的蓝色项目文件图标,选择项目中的活动目标,然后导航到“Build Phases”面板。在顶部,您将看到一个+符号,您可以单击该符号以添加新的构建阶段。选择“添加新运行脚本阶段”,然后在源编辑器中输入以下内容:

find . -name '*.gyb' |                                               \
    while read file; do                                              \
        ./path/to/gyb --line-directive '' -o "${file%.gyb}" "$file"; \
    done

确保在编译源之前订购GYB构建阶段。

现在,当您构建项目时,任何具有.swift.gyb文件扩展名的文件都由GYB评估,GYB会输出一个.swift与项目中其余代码一起编译的文件。

何时使用GYB

与任何工具一样,知道何时使用它与了解如何一样重要。以下是一些可以打开工具箱并进入GYB的示例。

生成公式代码

您是否为一组中的元素或序列中的项复制粘贴相同的代码?具有可变替换的for-in循环可能是解决方案。

Codable前面的from 示例所示,您可以在GYB模板文件的顶部声明一个集合,然后遍历该集合以获取类型,属性或方法声明:

%{
    abilities = ['strength', 'dexterity', 'constitution',
                 'intelligence', 'wisdom', 'charisma']
}%
class Character {
  var name: String

% for ability in abilities:
  var ${ability}: Int
% end
}

使用GYB进行评估会生成以下Swift代码:

class Character {
  var name: String

  var strength: Int
  var dexterity: Int
  var constitution: Int
  var intelligence: Int
  var wisdom: Int
  var charisma: Int
}

代码中的大量重复是一种气味,可能表明有更好的方法来完成任务。协议扩展和泛型等内置语言功能可以消除大量的代码重复,因此请注意使用它们而不是使用GYB强制执行。

生成从数据派生的代码

您是在编写基于数据源的代码吗?尝试将GYB纳入您的开发中!

GYB文件可以导入Python包一样jsonxmlcsv,这样你就可以分析几乎任何类型的可能会遇到的文件:

%{ import csv }
% with open('path/to/file.csv') as file:
    % for row in csv.DictReader(file):

如果您希望看到这一点,请查看 Currencies.swift.gyb ,它为ISO 4217规范定义的每种货币生成Swift枚举 。

通过将数据下载到可以检入源代码管理的文件而不是在GYB文件中进行HTTP请求或数据库查询,可以使编译保持快速和确定性。

代码生成使您的代码与相关标准保持同步变得微不足道。只需更新数据文件并重新运行GYB即可。


斯威夫特已经做了很多与另外的编译器合成的最近削减样板 EncodableDecodable4.0, EquatableHashable4.1,和 4.2。我们希望这种势头能够在未来的语言更新中得到体现。Case<wbr style="box-sizing: border-box;">Iterable

与此同时,对于其他一切,GYB是代码生成的有用工具。

社区中另一个很棒的工具是 Sourcery,它允许你在Swift(通过Stencil)而不是Python中编写模板。

“不要重复自己”可能是编程的一种美德,但有时你必须说几次才能使事情发挥作用。当你这样做时,你会感谢有一个像GYB这样的工具来为你说。

扫码进交流群 有技术的来闲聊 没技术的来学习

691040931

原文转载地址:https://nshipster.com/swift-gyb/

相关文章

网友评论

    本文标题:Swift GYB

    本文链接:https://www.haomeiwen.com/subject/vstzfctx.html