- 921. Minimum Add to Make Parentheses Valid 使括号有效的最少添加
- 915. Partition Array into Disjoint Intervals 分割数组
- 932. Beautiful Array 漂亮数组
- 940. Distinct Subsequences II 不同的子序列 II
Swift 提供的泛型可以让我们写出灵活且可重用的函数和类型
Swift 标准库是通过泛型代码构建出来的
Swift 的数组和字典类型都是泛型集
我们可以创建一个 Int 数组,也可创建一个 String 数组,或者甚至于可以是任何其他 Swift 的类型数据数组
下面的代码是一个非泛型函数 exchange 用来交换两个 Int 值
import Cocoa
// 定义一个交换两个变量的函数
func swapTwoInts(_ a: inout Int, _ b: inout Int)
{
let temporaryA = a
a = b
b = temporaryA
}
var numb1 = 100
var numb2 = 200
print("交换前数据: \(numb1) 和 \(numb2)")
swapTwoInts(&numb1, &numb2)
print("交换后数据: \(numb1) 和 \(numb2)")
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
交换前数据: 100 和 200
交换后数据: 200 和 100
函数swapTwoInts 只能交换整数 Int 类型的变量
如果想要交换两个 String 值或者 Double 值,就得重新写个对应的函数
import Cocoa
// 交换两个字符串变量的值
func swapTwoStrings(_ a: inout String, _ b: inout String)
{
let temporaryA = a
a = b
b = temporaryA
}
// 交换两个 Double 变量的值
func swapTwoDoubles(_ a: inout Double, _ b: inout Double)
{
let temporaryA = a
a = b
b = temporaryA
}
这三个函数从代码上来看,它们功能代码是相同的,只是类型上不一样
像这样的情况,我们可以使用泛型,从而避免重复编写代码
泛型使用了占位类型名(在这里用字母 T 来表示)来代替实际类型名(例如 Int、String 或 Double)
func swapTwoValues<T>(_ a: inout T, _ b: inout T)
swapTwoValues 后面跟着占位类型名(T),并用尖括号括起来(<T>
)
这个尖括号告诉 Swift 那个 T 是 swapTwoValues(::) 函数定义内的一个占位类型名
因此Swift 不会去查找名为 T 的实际类型
下面的代码使用一个泛型函数 exchange 用来交换两个 Int 和 String 值
import Cocoa
// 定义一个交换两个变量的函数
func swapTwoValues<T>(_ a: inout T, _ b: inout T)
{
let temporaryA = a
a = b
b = temporaryA
}
var numb1 = 100
var numb2 = 200
print("交换前数据: \(numb1) 和 \(numb2)")
swapTwoValues(&numb1, &numb2)
print("交换后数据: \(numb1) 和 \(numb2)")
var str1 = "A"
var str2 = "B"
print("交换前数据: \(str1) 和 \(str2)")
swapTwoValues(&str1, &str2)
print("交换后数据: \(str1) 和 \(str2)")
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
交换前数据: 100 和 200
交换后数据: 200 和 100
交换前数据: A 和 B
交换后数据: B 和 A
Swift 允许开发者定义自己的泛型类型
自定义类、结构体和枚举作用于任何类型,如同 Array 和 Dictionary 的用法
下面我们来编写一个名为 Stack (栈)的泛型集合类型,栈只允许在集合的末端添加新的元素(称之为入栈),且也只能从末端移除元素(称之为出栈)
图片中从左到右步骤如下:
1、 三个值在栈中;
2、 第四个值被压入到栈的顶部;
3、 现在有四个值在栈中,最近入栈的那个值在顶部;
4、 栈中最顶部的那个值被移除,或称之为出栈;
5、 移除掉一个值后,现在栈又只有三个值了;
下面的代码一个非泛型版本的栈,以 Int 型的栈为例
import Cocoa
struct IntStack
{
var items = [Int]()
mutating func push(_ item: Int)
{
items.append(item)
}
mutating func pop() -> Int
{
return items.removeLast()
}
}
这个结构体在栈中使用一个名为 items 的 Array 属性来存储值
Stack 提供了两个方法:push(_:) 和 pop(),用来向栈中压入值以及从栈中移除值
这些方法被标记为 mutating,因为它们需要修改结构体的 items 数组
但这个IntStack 结构体只能用于 Int 类型
我们可以定义一个泛型 Stack 结构体,从而能够处理任意类型的值
import Cocoa
struct Stack<Element>
{
var items = [Element]()
mutating func push(_ item: Element)
{
items.append(item)
}
mutating func pop() -> Element
{
return items.removeLast()
}
}
var stackOfStrings = Stack<String>()
print("字符串元素入栈: ")
stackOfStrings.push("google")
stackOfStrings.push("DDKK.COM 弟弟快看,程序员编程资料站")
print(stackOfStrings.items);
let deletetos = stackOfStrings.pop()
print("出栈元素: " + deletetos)
var stackOfInts = Stack<Int>()
print("整数元素入栈: ")
stackOfInts.push(1)
stackOfInts.push(2)
print(stackOfInts.items);
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
字符串元素入栈:
["google", "DDKK.COM 弟弟快看,程序员编程资料站"]
出栈元素: DDKK.COM 弟弟快看,程序员编程资料站
整数元素入栈:
[1, 2]
Stack 基本上和 IntStack 相同,占位类型参数 Element 代替了实际的 Int 类型
上面的的代码中,Element 在如下三个地方被用作占位符:
1、 创建items属性,使用Element类型的空数组对其进行初始化;
2、 指定push(_:)方法的唯一参数item的类型必须是Element类型;
3、 指定pop()方法的返回值类型必须是Element类型;
扩展一个泛型类型的时候(使用 extension 关键字),并不需要在扩展的定义中提供类型参数列表
更加方便的是,原始类型定义中声明的类型参数列表在扩展里是可以使用的,并且这些来自原始类型中的参数名称会被用作原始定义中类型参数的引用
下面的代码扩展了泛型类型 Stack,为其添加了一个名为 topItem 的只读计算型属性,它将会返回当前栈顶端的元素而不会将其从栈中移除
import Cocoa
struct Stack<Element>
{
var items = [Element]()
mutating func push(_ item: Element)
{
items.append(item)
}
mutating func pop() -> Element {
return items.removeLast()
}
}
extension Stack
{
var topItem: Element?
{
return items.isEmpty ? nil : items[items.count - 1]
}
}
var stackOfStrings = Stack<String>()
print("字符串元素入栈: ")
stackOfStrings.push("google")
stackOfStrings.push("DDKK.COM 弟弟快看,程序员编程资料站")
if let topItem = stackOfStrings.topItem {
print("栈中的顶部元素是:\(topItem).")
}
print(stackOfStrings.items)
范例中topItem 属性会返回一个 Element 类型的可选值
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
字符串元素入栈:
栈中的顶部元素是:DDKK.COM 弟弟快看,程序员编程资料站.
["google", "DDKK.COM 弟弟快看,程序员编程资料站"]
我们也可以通过扩展一个存在的类型来指定关联类型。
例如Swift 的 Array 类型已经提供 append(_:) 方法,一个 count 属性,以及一个接受 Int 类型索引值的下标用以检索其元素
这三个功能都符合 Container 协议的要求,所以我们只需简单地声明 Array 采纳该协议就可以扩展 Array
下面的代码创建一个空扩展即可
extension Array: Container {}
类型约束指定了一个必须继承自指定类的类型参数,或者遵循一个特定的协议或协议构成
我们可以写一个在一个类型参数名后面的类型约束,通过冒号分割,来作为类型参数链的一部分
这种作用于泛型函数的类型约束的基础语法如下所示(和泛型类型的语法相同)
func someFunction<T: SomeClass, U: SomeProtocol> (someT: T, someU: U)
{
// 这里是泛型函数的函数体部分
}
上面这个函数有两个类型参数
1、 第一个类型参数T,有一个要求T必须是SomeClass子类的类型约束;
2、 第二个类型参数U,有一个要求U必须符合SomeProtocol协议的类型约束;
import Cocoa
// 非泛型函数,查找指定字符串在数组中的索引
func findIndex(ofString valueToFind: String, in array: [String]) -> Int?
{
for (index, value) in array.enumerated() {
if value == valueToFind {
// 找到返回索引值
return index
}
}
return nil
}
let strings = ["google", "weibo", "taobao", "DDKK.COM 弟弟快看,程序员编程资料站", "facebook"]
if let foundIndex = findIndex(ofString: "DDKK.COM 弟弟快看,程序员编程资料站", in: strings)
{
print("DDKK.COM 弟弟快看,程序员编程资料站 的索引为 \(foundIndex)")
}
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
DDKK.COM 弟弟快看,程序员编程资料站 的索引为 3
Swift 中使用 associatedtype 关键字来设置关联类型实例
下面的代码定义了一个 Container 协议,该协议定义了一个关联类型 ItemType
Container 协议只指定了三个任何遵从 Container 协议的类型必须提供的功能
遵从协议的类型在满足这三个条件的情况下也可以提供其它额外的功能
import Cocoa
// Container 协议
protocol Container
{
associatedtype ItemType
// 添加一个新元素到容器里
mutating func append(_ item: ItemType)
// 获取容器中元素的数
var count: Int { get }
// 通过索引值类型为 Int 的下标检索到容器中的每一个元素
subscript(i: Int) -> ItemType { get }
}
// Stack 结构体遵从 Container 协议
struct Stack<Element>: Container
{
// Stack<Element> 的原始实现部分
var items = [Element]()
mutating func push(_ item: Element) {
items.append(item)
}
mutating func pop() -> Element {
return items.removeLast()
}
// Container 协议的实现部分
mutating func append(_ item: Element) {
self.push(item)
}
var count: Int {
return items.count
}
subscript(i: Int) -> Element {
return items[i]
}
}
var tos = Stack<String>()
tos.push("google")
tos.push("DDKK.COM 弟弟快看,程序员编程资料站")
tos.push("taobao")
// 元素列表
print(tos.items)
// 元素个数
print( tos.count)
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
["google", "DDKK.COM 弟弟快看,程序员编程资料站", "taobao"]
3
类型约束能够确保类型符合泛型函数或类的定义约束
我们可以在参数列表中通过 where 语句定义参数的约束
可以写一个 where 语句,紧跟在在类型参数列表后面
where 语句后跟一个或者多个针对关联类型的约束,以及(或)一个或多个类型和关联类型间的等价(equality)关系
下面的范例定义了一个名为 allItemsMatch 的泛型函数,用来检查两个 Container 实例是否包含相同顺序的相同元素
如果所有的元素能够匹配,那么返回 true,反之则返回 false
import Cocoa
// Container 协议
protocol Container
{
associatedtype ItemType
// 添加一个新元素到容器里
mutating func append(_ item: ItemType)
// 获取容器中元素的数
var count: Int { get }
// 通过索引值类型为 Int 的下标检索到容器中的每一个元素
subscript(i: Int) -> ItemType { get }
}
// // 遵循Container协议的泛型TOS类型
struct Stack<Element>: Container {
// Stack<Element> 的原始实现部分
var items = [Element]()
mutating func push(_ item: Element) {
items.append(item)
}
mutating func pop() -> Element {
return items.removeLast()
}
// Container 协议的实现部分
mutating func append(_ item: Element) {
self.push(item)
}
var count: Int {
return items.count
}
subscript(i: Int) -> Element {
return items[i]
}
}
// 扩展,将 Array 当作 Container 来使用
extension Array: Container {}
func allItemsMatch<C1: Container, C2: Container>
(_ someContainer: C1, _ anotherContainer: C2) -> Bool
where C1.ItemType == C2.ItemType, C1.ItemType: Equatable {
// 检查两个容器含有相同数量的元素
if someContainer.count != anotherContainer.count {
return false
}
// 检查每一对元素是否相等
for i in 0..<someContainer.count {
if someContainer[i] != anotherContainer[i] {
return false
}
}
// 所有元素都匹配,返回 true
return true
}
var tos = Stack<String>()
tos.push("google")
tos.push("DDKK.COM 弟弟快看,程序员编程资料站")
tos.push("taobao")
var aos = ["google", "DDKK.COM 弟弟快看,程序员编程资料站", "taobao"]
if allItemsMatch(tos, aos) {
print("匹配所有元素")
} else {
print("元素不匹配")
}
编译运行以上 Swift 范例,输出结果为
$ swift main.swift
匹配所有元素
有没有办法在 .swift 文件(编译成 .swift 模块)中声明函数,如下所示: 你好.swift func hello_world() { println("hello world")
我正在尝试使用 xmpp_messenger_ios 和 XMPPFramework 在 iOS 上执行 MUC 这是加入房间的代码。 func createOrJoinRoomOnXMPP()
我想在我的应用程序上创建一个 3D Touch 快捷方式,我已经完成了有关快捷方式本身的所有操作,它显示正确,带有文本和图标。 当我运行这个快捷方式时,我的应用程序崩溃了,因为 AppDelegate
我的代码如下: let assetTag = Expression("asset_tag") let query2 = mdm.select(mdm[assetTag],os, mac, lastRe
我的 swift 代码如下所示 Family.arrayTuple:[(String,String)]? = [] Family.arrayTupleStorage:String? Family.ar
这是我的 JSON,当我读取 ord 和 uniq 数据时出现错误 let response2 : [String: Any] = ["Response":["status":"SUCCESS","
我想将 swift 扩展文件移动到 swift 包中。但是,将文件移动到 swift 包后,我遇到了这种错误: "Type 'NSAttributedString' has no member 'ma
使用CocoaPods,我们可以设置以下配置: pod 'SourceModel', :configurations => ['Debug'] 有什么方法可以用 Swift Package Manag
我正在 Xcode 中开发一个 swift 项目。我将其称为主要项目。我大部分都在工作。我在日期选择器、日期范围和日期数学方面遇到了麻烦,因此我开始了另一个名为 StarEndDate 的项目,其中只
这是 ObjectiveC 代码: CCSprite *progress = [CCSprite spriteWithImageNamed:@"progress.png"]; mProgressBar
我正在创建一个命令行工具,在 Xcode 中使用 Swift。我想使用一个类似于 grunt 的配置文件确实如此,但我希望它是像 Swift 包管理器的 package.swift 文件那样的快速代码
我假设这意味着使用系统上安装的任何 swift 运行脚本:#!/usr/bin/swift 如何指定脚本适用的解释器版本? 最佳答案 Cato可用于此: #!/usr/bin/env cato 1.2
代码说完全没问题,没有错误,但是当我去运行模拟器的时候,会出现这样的字样: (Swift.LazyMapCollection (_base:[ ] 我正在尝试创建一个显示报价的报价应用。 这是导入
是否可以在运行 Swift(例如 Perfect、Vapor、Kitura 等)的服务器上使用 RealmSwift 并使用它来存储数据? (我正在考虑尝试将其作为另一种解决方案的替代方案,例如 no
我刚开始学习编程,正在尝试完成 Swift 编程书中的实验。 它要求““编写一个函数,通过比较两个 Rank 值的原始值来比较它们。” enum Rank: Int { case Ace = 1 ca
在您将此问题标记为重复之前,我检查了 this question 它对我不起作用。 如何修复这个错误: error: SWIFT_VERSION '5.0' is unsupported, suppo
从 Xcode 9.3 开始,我在我的模型中使用“Swift.ImplicitlyUnwrappedOptional.some”包裹了我的字符串变量 我不知道这是怎么发生的,但它毁了我的应用程序! 我
这个问题在这里已经有了答案: How to include .swift file from other .swift file in an immediate mode? (2 个答案) 关闭 6
我正在使用 Swift Package Manager 创建一个应用程序,我需要知道构建项目的配置,即 Debug 或 Release。我试图避免使用 .xcodeproj 文件。请有人让我知道这是否
有一个带有函数定义的文件bar.swift: func bar() { println("bar") } 以及一个以立即模式运行的脚本foo.swift: #!/usr/bin/xcrun s
我是一名优秀的程序员,十分优秀!