OC中的位枚举在swift中的替代方案

位移枚举在OC中非常常见也是非常方便运用的,举个例子SD_image中的SDWebImageOptions
1> NS_OPTIONS与位运算
NS_OPTIONS用来定义位移相关操作的枚举值,当一个枚举变量需要携带多种值的时候就需要,我们可以参考UIKit.Framework的头文件,可以看到大量的枚举定义。例如在SDWebImage下面就会接触到SDWebImageOptions枚举值:

typedef NS_OPTIONS(NSUInteger, SDWebImageOptions) {
    SDWebImageRetryFailed = 1 << 0,
    SDWebImageLowPriority = 1 << 1,
    SDWebImageCacheMemoryOnly = 1 << 2,
    SDWebImageProgressiveDownload = 1 << 3,
    SDWebImageRefreshCached = 1 << 4,
    SDWebImageContinueInBackground = 1 << 5,
    SDWebImageHandleCookies = 1 << 6,
    SDWebImageAllowInvalidSSLCertificates = 1 << 7,
    SDWebImageHighPriority = 1 << 8,
    SDWebImageDelayPlaceholder = 1 << 9,
    SDWebImageTransformAnimatedImage = 1 << 10,
    SDWebImageAvoidAutoSetImage = 1 << 11,
    SDWebImageScaleDownLargeImages = 1 << 12
};

找了一篇位运算基础 https://baike.baidu.com/item/位运算/6888804

================分界线================
然而,在swift中使用位枚举的时候发现swift本身不支持原来的语法,我们先尝试原来的写作方法:

public enum Corner {
    case topLeft = 1 << 1
    case topRight = 1 << 2
    case bottomLeft = 1 << 3
    case bottomRight  = 1 << 4
//    case all
}

语法报错为


Snip20190624_7.png

把Corner改为Int类型仍然解决不了问题,(enum Corner: Int).
swift 的枚举不支持运算值,只支持储存值,那试试吧位移值的具体值计算出来

public enum Corner: Int {
    case topLeft = 2
    case topRight = 4
    case bottomLeft = 8
    case bottomRight  = 16
//    case all
}

虽然不报错了,但是这种写法如果枚举值很多的话,他们之间的关系不够直观,既那然枚举值不能是计算值,那么我们给枚举添加一个值的属性。事例代码如下:

public enum RectCorner: RectCornersValue {
    case topLeft
    case topRight
    case bottomLeft
    case bottomRight
    case all
}

extension RectCorner {
    fileprivate var value: Int {
        switch self {
        case .topLeft:      return 1 << 0
        case .topRight:     return 1 << 1
        case .bottomLeft:   return 1 << 2
        case .bottomRight:  return 1 << 3
        case .all:
            return RectCorner.topLeft.value | RectCorner.topRight.value | RectCorner.bottomLeft.value | RectCorner.bottomRight.value
        }
    }
}

我们先忽略RectCornersValue,给枚举RectCorner写了个私有的value属性,也就是位移枚举的计算值,
使枚举能使用位运算,需要重载运算符‘|’

infix operator | : precedence
precedencegroup precedence { associativity: left }

重载运算符‘|’后,运算值为Int类型,也就是意味着,方法接受枚举类型同时也可能是枚举位运算的Int类型,这里我们用协议扩展约束类型

public protocol RectCornersValue {}
extension Int: RectCornersValue {}

同时 enum RectCorner: RectCornersValue 枚举也要遵循协议约束,这样方法参数类型RectCornersValue支持枚举和枚举的位运算值。
下面举个例子给view的任意角添加圆角,方法声明如下:

func cornerSet (_ radius: CGFloat, corners: RectCornersValue) {    } 

当角的参数为组合枚举的时候如何检测组合的元素,这里需要位运算的知识,这里例子如下:

extension RectCornersValue {
    fileprivate func isContains(_ objEnum: RectCorner) -> Bool {
        if (self as? Int) != nil {
            let value = self as! Int
            return value & objEnum.value != 0
        } else {
            let value = (self as? RectCorner)?.value ?? 0
            return value & objEnum.value != 0
        }
    }
}

完整的代码实例为

import Foundation

infix operator | : precedence
precedencegroup precedence { associativity: left }
public protocol RectCornersValue {}
extension Int: RectCornersValue {}

/// RectCorner
///
/// - topLeft: topLeft
/// - topRight: topRight
/// - bottomLeft: bottomLeft
/// - bottomRight: bottomRight
/// - all: all
public enum RectCorner: RectCornersValue {
    case topLeft
    case topRight
    case bottomLeft
    case bottomRight
    case all
}

extension RectCorner {
    fileprivate var value: Int {
        switch self {
        case .topLeft:      return 1 << 0
        case .topRight:     return 1 << 1
        case .bottomLeft:   return 1 << 2
        case .bottomRight:  return 1 << 3
        case .all:
            return RectCorner.topLeft.value | RectCorner.topRight.value | RectCorner.bottomLeft.value | RectCorner.bottomRight.value
        }
    }
    static func | (left: RectCornersValue , right: RectCorner) -> RectCornersValue {
        if ((left as? Int) != nil) {
            let leftValue: Int = (left as! Int)
            return leftValue | right.value
        } else {
            let leftValue: RectCorner = (left as! RectCorner)
            return leftValue.value | right.value
        }
    }
}

extension RectCornersValue {
    fileprivate func isContains(_ objEnum: RectCorner) -> Bool {
        if (self as? Int) != nil {
            let value = self as! Int
            return value & objEnum.value != 0
        } else {
            let value = (self as? RectCorner)?.value ?? 0
            return value & objEnum.value != 0
        }
    }
}
extension UIView {
    /// view任意圆角弧度设置
    ///
    /// - Parameters:
    ///   - radius: 弧度半径
    ///   - corners: 角 RectCorner,支持位枚举 
    func cornerSet (_ radius: CGFloat, corners: RectCornersValue) {
        self.setNeedsLayout()
        self.layoutIfNeeded()
        var radius = radius
        let pathRef = CGMutablePath()
        let bounds: CGRect = self.bounds
        let allowdMaxRadius = min(bounds.size.width, bounds.size.height)/2.0
        if radius > allowdMaxRadius { radius = allowdMaxRadius }
        
        let leftMiddle  = CGPoint(x: bounds.minX, y: bounds.midY)
        let topLeft     = CGPoint(x: bounds.minX, y: bounds.minY)
        let topRight    = CGPoint(x: bounds.maxX, y: bounds.minY)
        let bottomLeft  = CGPoint(x: bounds.minX, y: bounds.maxY)
        let bottomRight = CGPoint(x: bounds.maxX, y: bounds.maxY)
        let topMiddle   = CGPoint(x: bounds.midX, y: bounds.minY)
        let bottomMiddle = CGPoint(x: bounds.midX, y: bounds.maxY)
        let rightMiddle = CGPoint(x: bounds.maxX, y: bounds.midY)
        
        func drawLine(p1: CGPoint, p2: CGPoint, _ corner: RectCorner) {
            if corners.isContains(corner) {
                pathRef.addArc(tangent1End: p1,
                               tangent2End: p2,
                               radius: radius,
                               transform: .identity)
            } else {
                pathRef.addLine(to: p1)
                pathRef.addLine(to: p2)
            }
        }
        pathRef.move(to: leftMiddle)
        drawLine(p1: topLeft, p2: topMiddle, .topLeft)
        drawLine(p1: topRight, p2: rightMiddle, .topRight)
        drawLine(p1: bottomRight, p2: bottomMiddle, .bottomLeft)
        drawLine(p1: bottomLeft, p2: leftMiddle, .bottomRight)
        let layer = CAShapeLayer()
        layer.path = pathRef
        self.layer.mask = layer
    } 
}

//用法:   label.cornerSet (10.0, corners:RectCornersValue.topLeft | RectCornersValue.topRight) 

总结:所运用知识有位运算,枚举,运算符重载,协议及协议约束等
总体来说,swift下的位枚举运用没有OC下简单,这里只是作为学习和研究使用。

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