美文网首页iOSiOS开发
简洁使用Swift实例(isIn)

简洁使用Swift实例(isIn)

作者: 李现科 | 来源:发表于2016-08-25 13:44 被阅读227次

前提

大家应该常遇到在table view的delegate里处理若干个cell的问题.很多人的条件语句是这样的:

if indexPath.row == 0 || indexPath.row == 1 || indexPath.row == 3 {
    // do something
}

这个写法非常常规,也清晰明了.但是致命的缺点是

  • 当我有更多选项时长度已经超出字符限制
  • 没有办法实现可变数量的选项

解决方案

简单处理下,可以将所有可能放到数组或者范围(如果连续)中.然后调用其contains方法用于校验.代码如下:

if (2...8).contains(indexPath.row) {
    // do something
}
if [2,4,6,9].contains(indexPath.row) {
    // do something
}
if 2...8 ~= indexPath.row {                // 匹配运算符
    // do something
}

优化

这样虽然解决了前提所述的两个问题,但是写法不够美观,语义不够清晰.如果说可以使用isIn(:)函数处理,就显得美观多了.实现如下:

// 定义
extension Int {
    func isIn(range: Range<Int>) -> Bool {
        return range.contains(self)
    }
    
    func isIn<T: SequenceType where T.Generator.Element == Int>(ints: T) -> Bool {
        return ints.contains(self)
    }
    
    func isIn(ints: Int...) -> Bool {   //这里使用可变参数函数来避免必须将参数放入数组
        return ints.contains(self)
    }
}
// 使用
if indexPath.row.isIn(4,5,6) {
    // do something
}

这样实现了美观简洁地使用,但是还不够通用.比如UInt,Float等可以支持Range或数组的类型却无法使用.如果要更加通用,必须研究下contains函数,其定义如下:

extension SequenceType where Generator.Element : Equatable {
    /// Returns `true` iff `element` is in `self`.
    @warn_unused_result
    public func contains(element: Self.Generator.Element) -> Bool
}

那么显然我们需要扩写Equatable协议,让其实现isIn(:)函数.代码如下:

extension Equatable {
    func isIn<T: SequenceType where T.Generator.Element == Self>(collection: T) -> Bool {
        return collection.contains(self)
    }
    
    func isIn(collection: Self...) -> Bool {
        return collection.contains(self)
    }
}

这里需要说明下Range不需额外定义的理由.首先Range的定义如下:

public struct Range<Element : ForwardIndexType> : Equatable, CollectionType, CustomStringConvertible, CustomDebugStringConvertible {...}

它遵循于CollectionType(继承自SequenceType),它的Element遵循于ForwardIndexType(继承自Equatable),所以它是符合extension Equatable中第一个函数的.

后记

至此,我们用最简洁的方式写出了contains的反函数.这一切要感谢Swift强大的特性支持我们在不同维度约束条件,既安全又强大.

相关文章

网友评论

  • d1497e8e780a:那如果说你extension里面的<T: SequenceType where T.Generator.Element == Self> ==Self 不成立怎么办,或者说这个T不遵循 Equatable协议?
    李现科:@张大帅的成神之路 你有尝试去改写为: Equatable了吗?Tuple本身并不遵循Equatable.针对这点只能用特殊闭包去处理,你可以看下这个:
    public func contains(@noescape predicate: (Self.Generator.Element) throws -> Bool) rethrows -> Bool
    d1497e8e780a:@李现科 1.既然你都说了要做到真正的通用,真实的泛型,为何不尝试让一些本来不遵循Equatable协议的类遵循此协议,即重载比较运算符.
    2. <T: SequenceType where T.Generator.Element == Self> 我觉得写成 <T: SequenceType where T.Generator.Element: Equatable > 更合适吧, 这个泛型万一是一个元组呢.
    李现科:@张大帅的成神之路 连Equatable都不遵循,就用不了比较运算符,不在考虑之列

本文标题:简洁使用Swift实例(isIn)

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