iOS程序犭袁iOS技术点征服Swift

在Swift3.1中 initialize被警告未来会禁用(di

2017-04-05  本文已影响1314人  黑暗中的孤影

3月29号,苹果正式发布了iOS10.3,作为iOS开发者,自然也很关注每次伴随iOS更新而发布的XCode,
这次苹果发布的是XCode8.3,同时Swift3.1也一起发布。Swift3.1的新特性网上已经有很多相关文章说明了。
但是有一点都没有提到:就是initialize方法被标记在未来的Swift版本将不能再使用。
下面我来告诉大家在Swift3.1时面用什么来替换initialize方法。

在未来的Swift版本中initialize将不可用

虽然现在的Swift3.1版本还可以用,但是已经被警告了,
发布到Cocopods将不被接受。所以还是要想办法移除这个警告。
我百度这个问题,发现基本没有答案,所以只好在万能的Stackoverflow中寻找。
果然,第一问题在高票答案完美地解决了这个问题。

http://stackoverflow.com/questions/42824541/swift-3-1-deprecates-initialize-how-can-i-achieve-the-same-thing

下面我尽量翻译过来让大家看明白

容易简单的解决方案

大部分App的入口点是AppDelegate

func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool
{
    your code
}

方法。当App启动完后会调用该方法。所以你只要在你自己的类定义一个静态初始化方去,然后再在里面调用该静态方法就可以了。
比如经典的IQKeyboardManager就是这样启动的。

有讲究深度的解决方案

上面那个方法可能有点不省心,如果你想创建一个框架伴随APP启动而启动,但是又也不想在AppDelegate的方法里来初始化。
而是在APP启动时就让它默默调用。那么可通知以下方案

第一步:定义协议

定义下面的swift代码,包含一个protocalclass。目的是为所有类提供一个简单的入口点来达到类似initialize的效果。你可以自定义一个类
来实现SelfAware协议。SelfAware协议内有一个awake方法,该方法是初始化代码的核心方法。

public protocol SelfAware:class { // 定义SelfAware协议
    static func awake()
}

class NothingToSeeHere{
    static func harmlessFunction(){
        let typeCount = Int(objc_getClassList(nil, 0))
        let  types = UnsafeMutablePointer<AnyClass?>.allocate(capacity: typeCount)
        let autoreleaseintTypes = AutoreleasingUnsafeMutablePointer<AnyClass?>(types)
        objc_getClassList(autoreleaseintTypes, Int32(typeCount)) //获取所有的类
        for index in 0 ..< typeCount{
            (types[index] as? SelfAware.Type)?.awake() //如果该类实现了SelfAware协议,那么调用awake方法
        }
        types.deallocate(capacity: typeCount)
    }
}

上面的代码对于大部分的iOS开发者可能看不习惯,因为我们平时开发很少使用到指针。但是看懂并不难。简单来说就是找出该项目所有的class
,如果该class实现了SelfAware协议,那么调用awake方法。这样你就可以自定义你自己的类来实现SelfAware协议达到这个效果

第二步:让APP启动时只执行一次harmlessFunction方法

有了上面的代码,现在还有一种重要的事情,就是让APP启动的时侯运行NothingToSeeHere.harmlessFunction方法。先前,这个一般最好使用Objc来
实现这个功能,但是在这在里只能使用Swift,参考以下代码

extension UIApplication{
    private static let runOnce:Void = { //使用静态属性以保证只调用一次(该属性是个方法)
        NothingToSeeHere.harmlessFunction()
    }()
    
    open override var next: UIResponder?{ //重写next属性
        UIApplication.runOnce
        return super.next
    }
}

在这里扩展了UIApplication并重写了next属性,关于UIApplicationnext属性有兴趣的开发者可以去看iOS事件传递机制文章。这里就不说了
在iOS的事件响应过程中next会调用多次。但是这里使用静态的runOnce属性来保证只会调用一次。然后再在该属性里调用NothingToSeeHereharmlessFunction方法

第三步:�自定义类实现SelfAware协议,重写awake方法

接下来的事情就很简单了,现在我们在App启动的时侯有了入口点,还有在启动过程中勾住启动点的类,剩下你只要自定义一个类实现SelfAware即可

class ProjectStart: SelfAware {
    static func awake() {
         your code
    }
}

上面有些是翻译stackoverflow原回答,自己也加了些东西。�相信读者应该不能理解。虽然在大部分开发者在项目里使用iOS runtime的东西并不多,
但是有时侯这些黑魔法还是非常好用的。 另外从这个警告也可以看出,苹果在将来会慢慢在swift中移除Objc的一些方法。所以使用swift实现iOS runtime
要谨慎。

上一篇下一篇

猜你喜欢

热点阅读