//MARK: - 主题更改时,自动执行

extension NSObject {

fileprivate struct AssociatedKeys {

static var thmemChanged = "thmemChanged"

}

/// 当前主题更改时、第一次设置时 自动调用的闭包

public typealias ThemeChangedClosure = @convention(block) (_ style:String) -> Void

/// 自动调用的闭包

/// 设置时,会设置一个KVO监听,当V2Style.style更改时、第一次赋值时 会自动调用这个闭包

var thmemChangedHandler:ThemeChangedClosure? {

get {

let closureObject: AnyObject? = objc_getAssociatedObject(self, &AssociatedKeys.thmemChanged) as AnyObject?

guard closureObject != nil else{

return nil

}

let closure = unsafeBitCast(closureObject, to: ThemeChangedClosure.self)

return closure

}

set{

guard let value = newValue else{

return

}

let dealObject: AnyObject = unsafeBitCast(value, to: AnyObject.self)

objc_setAssociatedObject(self, &AssociatedKeys.thmemChanged,dealObject,objc_AssociationPolicy.OBJC_ASSOCIATION_RETAIN_NONATOMIC)

//设置KVO监听

self.kvoController.observe(V2EXColor.sharedInstance, keyPath: "style", options: [.initial,.new] , block: {[weak self] (nav, color, change) -> Void in

self?.thmemChangedHandler?(V2EXColor.sharedInstance.style)

}

)

}

}

}




class V2EXColor :NSObject {

fileprivate static let STYLE_KEY = "styleKey"

static let V2EXColorStyleDefault = "Default"

static let V2EXColorStyleDark = "Dark"

fileprivate static var _colors:V2EXColorProtocol?

static var colors :V2EXColorProtocol {

get{

if let c = V2EXColor._colors {

return c

}

else{

if V2EXColor.sharedInstance.style == V2EXColor.V2EXColorStyleDefault{

return V2EXDefaultColor.sharedInstance

}

else{

return V2EXDarkColor.sharedInstance

}

}

}

set{

V2EXColor._colors = newValue

}

}

dynamic var style:String

static let sharedInstance = V2EXColor()

fileprivate override init(){

if let style = V2EXSettings.sharedInstance[V2EXColor.STYLE_KEY] {

self.style = style

}

else{

self.style = V2EXColor.V2EXColorStyleDefault

}

super.init()

}

func setStyleAndSave(_ style:String){

if self.style == style {

return

}

if style == V2EXColor.V2EXColorStyleDefault {

V2EXColor.colors = V2EXDefaultColor.sharedInstance

}

else{

V2EXColor.colors = V2EXDarkColor.sharedInstance

}

self.style = style

V2EXSettings.sharedInstance[V2EXColor.STYLE_KEY] = style

}

}



dynamic关键字

如果您有过OC的开发经验,那一定会对OC中@dynamic关键字比较熟悉,它告诉编译器不要为属性合成getter和setter方法。

Swift中也有dynamic关键字,它可以用于修饰变量或函数,它的意思也与OC完全不同。它告诉编译器使用动态分发而不是静态分发。OC区别于其他语言的一个特点在于它的动态性,任何方法调用实际上都是消息分发,而Swift则尽可能做到静态分发。

因此,标记为dynamic的变量/函数会隐式的加上@objc关键字,它会使用OC的runtime机制。

虽然静态分发在效率上可能更好,不过一些app分析统计的库需要依赖动态分发的特性,动态的添加一些统计代码,这一点在Swift的静态分发机制下很难完成。这种情况下,虽然使用dynamic关键字会牺牲因为使用静态分发而获得的一些性能优化,但也依然是值得的。

classKraken{dynamicvarimADynamicallyDispatchedString:StringdynamicfuncimADynamicallyDispatchedFunction(){//Hoorayfordynamicdispatch!}}

使用动态分发,您可以更好的与OC中runtime的一些特性(如CoreData,KVC/KVO)进行交互,不过如果您不能确定变量或函数会被动态的修改、添加或使用了Method-Swizzle,那么就不应该使用dynamic关键字,否则有可能程序崩溃。