【自问自答】关于 Swift 的几个疑问
感觉自己给自己释疑,也是一个极为有趣的过程。这次,我还新增了“猜想”一栏,来尝试回答一些暂时没有足够资料支撑的问题。
Swift 版本是:4.0.3。不同版本的 Swift,可能无法复现问题。
个人记录,仅供参考,不保证严格意义上的正确性。
swift 中,如何在函数内,声明 static 变量 ?
问题描述:
以下语句,是编译不过的,提示:“static properties may only be declared on a type”
func add() -> Int {
static var base = 0
base += 1
return base
}
add()
add()
add()
解决方案:
可以用内嵌类型的 static 属性来解决,如:
func add() -> Int {
struct Temp{
static var base = 0
}
Temp.base += 1
return Temp.base
}
add() // --> 1
add() // --> 2
add() // --> 3
参考:https://stackoverflow.com/a/25354915
猜想:
同一作用域的同名内嵌类型,多次执行,只会真正定义一次.
swift 有没有可以进行全局埋点的黑魔法机制?
问题描述:
全局埋点,依赖于 runtime 机制, 所以换种问法就是: swift 中如何继续使用 objc 的runtime 机制.
解决方案:
纯Swift类没有动态性,但在方法、属性前添加dynamic修饰可以获得动态性。
继承自NSObject的Swift类,其继承自父类的方法具有动态性,其他自定义方法、属性需要加dynamic修饰才可以获得动态性。
若方法的参数、属性类型为Swift特有、无法映射到Objective-C的类型(如Character、Tuple),则此方法、属性无法添加dynamic修饰(会编译错误)
参考: http://www.infoq.com/cn/articles/dynamic-analysis-of-runtime-swift
快速验证,可使用:
class A{
@objc dynamic func funcA(){
print("funcA")
}
}
func methodSwizze(cls: AnyClass, originalSelector: Selector, swizzledSelector:Selector){
let originalMethod = class_getInstanceMethod(cls, originalSelector)
let swizzledMethod = class_getInstanceMethod(cls, swizzledSelector)
if let originalMethod = originalMethod, let swizzledMethod = swizzledMethod {
method_exchangeImplementations(originalMethod, swizzledMethod)
}
}
extension A{
@objc dynamic func funcB(){
print("funcB")
}
}
methodSwizze(cls: A.self, originalSelector: #selector(A.funcA), swizzledSelector: #selector(A.funcB))
let a = A()
a.funcB() // --> funcA
a.funcA() // --> funcB
注意: swift 4 中, 加 dynamic 的同时,也必须加 @objc -- 即不允许单独加 dynamic 标记.
猜想:
dynamic 是在用性能换灵活性.生产环境下,未来更可能的方案,可能是:
通过协议,约定必须实现的统计相关的方法 --> 通过单元测试,来保证遵循特定统计协议的类型,在特定的时机一定会调用协议规定的统计方法.
extension 中覆盖某个自定义的 framework 中的 open/public class 中的 private 方法,会发生什么事?
问题描述:
模块A:
open class Book: NSObject {
private func funcA(){
print("private funcA")
}
public func callFuncA(){
funcA()
}
}
模块B:
public extension Book {
func funcA(){
print("public funcA")
}
}
问题:
模块B 中,以下代码的输出是?
let book = Book()
book.funcA() // --> ?
book.callFuncA() // --> ?
解决方案:
可以直接运行观察:
let book = Book()
book.funcA() // --> public funcA
book.callFuncA() // --> private funcA
所以: 通过 extension 覆盖其他模块open类的private方法,不会有任何诡异的问题.两个实现,都对彼此透明.
更进一步: 模块B以 Optional 方式引入模块A. 如果是在模块B中,通过 extension 覆盖模块A的private 方法.然后在模块 C 中同时引入了模块 A 和 B,此时模块C中类似的函数调用,会是哪个模块的方法实现生效?
let book = Book()
book.funcA() // --> public funcA
book.callFuncA() // --> private funcA
可以看到,仍然是模块B中的 public 级别的方法生效.
再进一步,如果模块 A 中的方法,由 private 改为 public,即:
open class Book: NSObject {
public func funcA(){
print("original public funcA")
}
public func callFuncA(){
funcA()
}
}
此时模块C 中的调用,会报错:
error: ambiguous use of 'funcA()'
book.funcA()
^
A.Book:2:17: note: found this candidate
public func funcA()
^
B.Book:2:17: note: found this candidate
public func funcA()
如果模块 B 以 Required 方式引入模块A,模块C,只引入模块B,此时的调用结果,会不会有什么不同? --> 然而,并没有什么不同,依然是同样的 ambiguous 错误.
总结一下:
可以安全地在 extension 中覆盖其他模块中open/public类中定义的非 public 方法.对于原有模块,会继续使用自身的非 public 的方法定义;定义其他模块,可以正确使用 extension 版本中的模块代码.
不要尝试在 extension 中定义其他模块中 open/public类中定义的 public 方法.虽然可以定义,但是使用时,会引起 ambiguous 错误.
在使用 extension 扩展其他模块中定义的类时,最好还是给自己扩展的方法加上特定前缀,不然第三方模块万一暴露的同名方法,自己的代码就彻底跪了.
猜想:
扩展第三方模块类时,使用自定义的前缀,总是一个好的习惯.
嵌套定义的类型,如果外层类型是 private, 内层类型是 open,内层类型.那么内层类型有可能在其他模块中被使用吗 ?
问题描述:
open class Book: NSObject {
private class InnerBook{
open class DeeperBook{
}
}
}
在另一个 swift 模块中,能使用类似下面的类型初始化代码吗?
var book = Book.InnerBook.DeeperBook()
解决方案:
直接调用,会报错:
error: 'InnerBook' is inaccessible due to 'private' protection level
尝试修改为:
open class Book: NSObject {
open class InnerBook{
open class DeeperBook{
}
}
}
依然报错:
error: 'Book.InnerBook.DeeperBook' initializer is inaccessible due to 'internal' protection level
根据提示,再修改下 DeeperBook 的初始化方法的访问级别:
open class Book: NSObject {
open class InnerBook{
open class DeeperBook{
public init() {
}
}
}
}
猜想:
内嵌类型的方法的访问级别,并不会随着类型本身访问级别的宽松更变得比默认的 internal 更宽松.
疑问: 为什么函数定义外的 closure 不会引起作用域内其他变量引用计数的变化?
问题描述:
仔细观察以下不同代码片段的不同输出:
片段A:
class Book{
let name: String
lazy var whoami:(()->String)? = {
return self.name
}
init(name:String) {
self.name = name
}
deinit {
print("\(name) is being deinitialized")
}
}
var aBook:Book? = Book(name: "风之影")
print(aBook!.whoami!())
aBook = nil
/*
输出:
风之影
*/
片段B:
class Book{
let name: String
lazy var whoami:(()->String)? = {
return self.name
}
init(name:String) {
self.name = name
}
deinit {
print("\(name) is being deinitialized")
}
}
var aBook:Book? = Book(name: "风之影")
print(aBook!.whoami!())
aBook?.whoami = nil
aBook = nil
/*
输出:
风之影
风之影 is being deinitialized
*/
片段C:
class Book{
let name: String
lazy var whoami:(()->String)? = {
return self.name
}
init(name:String) {
self.name = name
}
deinit {
print("\(name) is being deinitialized")
}
}
var aBook:Book? = Book(name: "风之影")
aBook?.whoami = {
return aBook!.name + " new"
}
print(aBook!.whoami!())
aBook = nil
/*
输出:
风之影 new
风之影 is being deinitialized
*/
片段A, aBook 内存泄露,经典的 closure self 循环引用问题.
片段B,是 closure self 循环引用的一个可选解决方案,即 self 主动切断对 closure 的引用.
片段C,比较诡异. aBook 引用了一个新的 closure,新的 closure 内又引用了 aBook 一次,但是 aBook 竟然还是可以正确释放,并没有预期中的内存泄露问题.令人费解!?
解决方案:
片段 D:
class Book{
let name: String
lazy var whoami:(()->String)? = {
return self.name
}
init(name:String) {
self.name = name
}
deinit {
print("\(name) is being deinitialized")
}
}
var aBook:Book? = Book(name: "风之影")
aBook?.whoami = {
[aBook] in
return aBook!.name + " new"
}
print(aBook!.whoami!())
aBook = nil
/*
输出:
风之影 new
*/
可以看到,这样 aBook 就会泄露了.片段 D 与 片段 C 的区别在于 closure 中的那句 [aBook] in .这个语法,是我"杜撰"的,语义上近似于以强引用方式捕捉 aBook 对应的真实对象.官方文档中并没有提到有这种语法.
另外,参考 objc 中block 的行为,我尝试搜索相关 swift 中 栈(stack) block 的相关信息.如果 closure 也区分栈和堆,倒是还可以勉强解释.不过,并没有相关的信息,而且 closure 本身也是不支持 copy 操作的.
注意: 当前复现此问题用的是 swift 4.0.3 版本,不同版本中的 closure 的行为可能不一致.
猜想:
或许 swift 中,只有内部有可能直接使用 self 的 closure,才需要特别考虑closure引起的内存泄露问题.
个人猜测,可能是因为 self 比较特殊, closure 只能直接捕捉其真实对象.
【自问自答】关于 Swift 的几个疑问的更多相关文章
- [python]自问自答:python -m参数?
python -m xxx.py 作用是:把xxx.py文件当做模块启动 但是我一直不明白当做模块启动到底有什么用.python xxx.py和python -m xxx.py有什么区别! 自问自答: ...
- 自问自答之VR遐想
先让我组织一下语言,作为表达能力超弱的战五渣来讲,归纳总结什么的最要命了. 我可以给你分析个1到N条出来,但是一般来讲没什么顺序,想到什么就说什么.而且我属于线性思维,有一个引子就可以按着话头一步步发 ...
- [python]自问自答:python -m参数? (转)
python -m xxx.py 作用是:把xxx.py文件当做模块启动但是我一直不明白当做模块启动到底有什么用.python xxx.py和python -m xxx.py有什么区别! 自问自答: ...
- [python]自问自答:python -m参数? (转) ( python2.7 版本 )
原文地址: http://www.cnblogs.com/xueweihan/p/5118222.html python -m xxx.py 作用是:把xxx.py文件当做模块启动 但是我一直不明白当 ...
- css自问自答(二)
css自问自答(二) 7.掌握定位的一些属性 position 和 display 属性,以及如何浮动(float)和清除(clear)元素,z-index属性 三个属性控制: position 属性 ...
- css自问自答(一)
css自问自答(一) 1.块级元素和行内元素特性与区别? 块级:display:block <div>.<p>.<h1>...<h6>.<ol&g ...
- vue 源码自问自答-响应式原理
vue 源码自问自答-响应式原理 最近看了 Vue 源码和源码分析类的文章,感觉明白了很多,但是仔细想想却说不出个所以然. 所以打算把自己掌握的知识,试着组织成自己的语言表达出来 不打算平铺直叙的写清 ...
- 区块链自问自答 day1
区块链自问自答 day1 简要介绍区块链是什么? 区块链(Blockchain)是一种对等网络下的分布式数据库系统 数据结构中的单向链表是通过每个节点包含一个节点的指针实现"链" ...
- 区块链自问自答 day2
区块链自问自答 day2 区块链的自治性是如何达成的?为什么能够在去信任的环境下自由安全地交换数据? 区块链中有众多的节点,包含了恶意节点.故障节点.正常节点,想要这些节点共同做出一致的决定就需要 ...
- golang 数组的一些自问自答
所有代码基于Go-1.17.一些研究Go数组的自问自答,可以考虑作为面试题. 问题:静态存储区是什么?和堆/栈有什么区别? 回答: 可以参考下列图 堆上存放new产生的大块内存 栈上存放的是程序运行的 ...
随机推荐
- GIT常用命令(图片版)
Git 是一个很强大的分布式版本控制系统.它不但适用于管理大型开源软件的源代码,管理私人的文档和源代码也有很多优势. 本来想着只把最有用.最常用的 Git 命令记下来,但是总觉得这个也挺有用.那个也用 ...
- 简陋的斗地主,js实现
最近闲了两天没事做,用js写了个斗地主,练习练习.代码和功能都很简陋,还有bug,咋只是聊聊自己的思路. 这里说说斗地主主要包含的功能:洗牌,发牌,玩家出牌.电脑出牌,出牌规则的验证,输赢啥的没有判断 ...
- 使用背景图修改radio、checkbox样式
如果觉得设置样式太麻烦,或者页面上选中的样式太复杂,也可以用背景图去修改样式<div class=""> <label><input type=&qu ...
- arguments,caller,callee之理解
arguments对象代表正在执行的函数和调用它的函数的参数,arguments是一个不是数组但类似 数组的对象,它具有同数组一样的访问性质及方式,可以由arguments[n]来访问对应单个参数的值 ...
- 从源代码到Runtime发生的重排序
源代码和Runtime时执行的代码很可能不一样,这是因为编译器.处理器常常会为了追求性能对改变执行顺序.然而改变顺序执行很危险,很有可能使得运行结果和预想的不一样,特别是当重排序共享变量时. 从源 ...
- UVA 540(队列)
Description Team Queue Queues and Priority Queues are data structures which are known to most comp ...
- Android之不须要自己定义View(ViewfindView.java)最简单的二维码扫描
不废话,先爆照 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/d ...
- Java方法的概念及使用
方法 将一段逻辑或者功能提取出来,这种提取的形式就是函数 格式 修饰符 返回值类型 函数名(参数列表){ 方法体: return 返回值; } //明确返回值类型---求两个整数的和,确定结果一定是整 ...
- Tomcat在Linux服务器上的BIO、NIO、APR模式设置
一.BIO.NIO.AIO 先了解四个概念: 同步 : 自己亲自出马持银行卡到银行取钱(使用同步IO时,Java自己处理IO读写). 异步 : 委托一小弟拿银行卡到银行取钱,然后给你(使用异步IO时, ...
- Softmax多分类算法
List<double[]> inputs_x = new List<double[]>(); inputs_x.Add(new double[] { 0.2, 0.3 }); ...