swift 泛型 return 第一个和最后一个元素
swift generics return first and last element
我正在尝试习惯泛型(从未在 objc 中使用过它们)并想编写一个玩具函数,它接受任何类型的对象 () 和 returns 第一个和最后一个元素。假设地,我只会在数组或字符串上使用它——我不断收到没有下标成员的错误。我完全理解错误消息告诉我 swift 不知道 T 可能持有一个确实有下标的类型——我只是想知道如何解决这个问题。
func firstAndLastFromCollection<T>(a:T?) {
var count: Int = 0
for item in a as! [AnyObject] {
count++
}
if count>1 {
var first = a?[0]
var last = a?[count-1]
return (first, last)
}
return something else here
}
我是否需要在此处的某处进行类型转换(这有点违背此处的目的,因为我需要向下转换为字符串或数组,添加代码并降低此函数的通用性)?
如果您想 return 第一个和最后一个元素,那么假设输入参数是某种类型的数组可能是安全的。
所以你可以这样实现你的功能
func firstAndLast<T>(list:[T]) -> (first:T, last:T)? {
guard let first = list.first, last = list.last else { return nil }
return (first, last)
}
该函数 return 一个包含 2 个元素的元组,它们都具有相同类型的输入数组的通用元素。
returned 元组是一个选项,因为如果数组为空,则 nil 是 returned.
例子
let nums = firstAndLast([1,2,3,4])
let words = firstAndLast(["One", "Two", "Three"])
正如您可以验证的那样,数组中的泛型元素类型成为元组中元素的类型。
在上面的例子中 nums
被推断为 (Int, Int)?
和 words
(Words, Words)?
更多示例
let emptyList: [String] = []
firstAndLast(emptyList) // nil
扩展
最后你也可以把这段代码写成Array的扩展。
extension Array {
var firstAndLast: (first:Element, last:Element)? {
guard let first = self.first, last = self.last else { return nil }
return (first, last)
}
}
现在你可以写了
let aCoupleOfShows = ["Breaking Bad", "Better Call Saul", "Mr Robot"].firstAndLast
同样,如果您检查常量 aCoupleOfShows
的类型,您会发现它是 (first: String, last: String)?
。 Swift 自动推断出正确的类型。
最后一个例子
在评论中你说你想要字符串的第一个和最后一个字符。这是代码,如果你使用上面的扩展
if let chars = Array("Hello world".characters).firstAndLast {
print("First char is \(chars.first), last char is \(chars.last) ")
}
//>> First char is H, last char is d
如果我们谈论集合,让我们使用 CollectionType
:
func firstAndLastFromCollection<T: CollectionType>(a: T) -> (T.Generator.Element, T.Generator.Element)? {
guard !a.isEmpty else {
return nil
}
return (a.first!, a.lazy.reverse().first!)
}
print(firstAndLastFromCollection(["a", "b", "c"])) // ("a", "c")
print(firstAndLastFromCollection("abc".characters)) // ("a", "c")
print(firstAndLastFromCollection(0..<200)) // (0, 199)
print(firstAndLastFromCollection([] as [String])) // nil
如果您指定泛型类型也符合双向索引:
func firstAndLastFromCollection<T: CollectionType where T.Index : BidirectionalIndexType>(...) -> ...
那么你可以直接调用last
:
return (a.first!, a.last!)
如果我们决定使用类别来实现它,我们根本不需要泛型:
extension CollectionType {
func firstAndLast() -> (Generator.Element, Generator.Element)? {
guard !self.isEmpty else {
return nil
}
return (self.first!, self.lazy.reverse().first!)
}
}
extension CollectionType where Index: BidirectionalIndexType {
func firstAndLast() -> (Generator.Element, Generator.Element)? {
guard !self.isEmpty else {
return nil
}
return (self.first!, self.last!)
}
}
print("abc".characters.firstAndLast())
Swift 是一种面向 协议 的语言。通常你会发现自己扩展协议比扩展 类 或结构更多。
我正在尝试习惯泛型(从未在 objc 中使用过它们)并想编写一个玩具函数,它接受任何类型的对象 () 和 returns 第一个和最后一个元素。假设地,我只会在数组或字符串上使用它——我不断收到没有下标成员的错误。我完全理解错误消息告诉我 swift 不知道 T 可能持有一个确实有下标的类型——我只是想知道如何解决这个问题。
func firstAndLastFromCollection<T>(a:T?) {
var count: Int = 0
for item in a as! [AnyObject] {
count++
}
if count>1 {
var first = a?[0]
var last = a?[count-1]
return (first, last)
}
return something else here
}
我是否需要在此处的某处进行类型转换(这有点违背此处的目的,因为我需要向下转换为字符串或数组,添加代码并降低此函数的通用性)?
如果您想 return 第一个和最后一个元素,那么假设输入参数是某种类型的数组可能是安全的。
所以你可以这样实现你的功能
func firstAndLast<T>(list:[T]) -> (first:T, last:T)? {
guard let first = list.first, last = list.last else { return nil }
return (first, last)
}
该函数 return 一个包含 2 个元素的元组,它们都具有相同类型的输入数组的通用元素。 returned 元组是一个选项,因为如果数组为空,则 nil 是 returned.
例子
let nums = firstAndLast([1,2,3,4])
let words = firstAndLast(["One", "Two", "Three"])
正如您可以验证的那样,数组中的泛型元素类型成为元组中元素的类型。
在上面的例子中 nums
被推断为 (Int, Int)?
和 words
(Words, Words)?
更多示例
let emptyList: [String] = []
firstAndLast(emptyList) // nil
扩展
最后你也可以把这段代码写成Array的扩展。
extension Array {
var firstAndLast: (first:Element, last:Element)? {
guard let first = self.first, last = self.last else { return nil }
return (first, last)
}
}
现在你可以写了
let aCoupleOfShows = ["Breaking Bad", "Better Call Saul", "Mr Robot"].firstAndLast
同样,如果您检查常量 aCoupleOfShows
的类型,您会发现它是 (first: String, last: String)?
。 Swift 自动推断出正确的类型。
最后一个例子
在评论中你说你想要字符串的第一个和最后一个字符。这是代码,如果你使用上面的扩展
if let chars = Array("Hello world".characters).firstAndLast {
print("First char is \(chars.first), last char is \(chars.last) ")
}
//>> First char is H, last char is d
如果我们谈论集合,让我们使用 CollectionType
:
func firstAndLastFromCollection<T: CollectionType>(a: T) -> (T.Generator.Element, T.Generator.Element)? {
guard !a.isEmpty else {
return nil
}
return (a.first!, a.lazy.reverse().first!)
}
print(firstAndLastFromCollection(["a", "b", "c"])) // ("a", "c")
print(firstAndLastFromCollection("abc".characters)) // ("a", "c")
print(firstAndLastFromCollection(0..<200)) // (0, 199)
print(firstAndLastFromCollection([] as [String])) // nil
如果您指定泛型类型也符合双向索引:
func firstAndLastFromCollection<T: CollectionType where T.Index : BidirectionalIndexType>(...) -> ...
那么你可以直接调用last
:
return (a.first!, a.last!)
如果我们决定使用类别来实现它,我们根本不需要泛型:
extension CollectionType {
func firstAndLast() -> (Generator.Element, Generator.Element)? {
guard !self.isEmpty else {
return nil
}
return (self.first!, self.lazy.reverse().first!)
}
}
extension CollectionType where Index: BidirectionalIndexType {
func firstAndLast() -> (Generator.Element, Generator.Element)? {
guard !self.isEmpty else {
return nil
}
return (self.first!, self.last!)
}
}
print("abc".characters.firstAndLast())
Swift 是一种面向 协议 的语言。通常你会发现自己扩展协议比扩展 类 或结构更多。