SwiftObject 杂记
一、前言
看了一段时间的Swift,慢慢转变了一些对Swift的看法。
Swift作为苹果新晋的开发语言、具有模板编程、函数编程、协议多继承、vTable静态绑定、值引用类型区分、Option类型等动态语法的多种特性。
Swift作为一门跨平台的语言,非常强调性能,静态绑定是Swift跟OC语言的动态派发迥异的区别。Objetive-C中的Runtime优秀的设计让OC这门语言有着优秀的动态特性,
Swift的语言设计中一部分为了和OC桥接,一部分也参考了OC的设计让Swift具有一些动态特性,比如自省。
Swift强调安全,Optional代替OC中的空指针,同时优化了性能。Optional调用链遇到中间的Optional不会继续执行。同时值类型的语义的大量使用可以规避线程安全带来的问题。
二、本次文章主要探究Swift编译运行机制
1)编写一个控制台程序,将可执行文件拖入Hopper进行反编译
注:下一部分代码借鉴 https://mp.weixin.qq.com/s/zIkB9KnAt1YPWGOOwyqY3Q 《Swift 对象内存模型探究(一)》
//
// main.swift
// TestSwift2
//
// Created by lunli on 2018/10/9.
// Copyright 2018年 lunli. All rights reserved.
// import Foundation
import ObjectiveC class People{
var xxxxx = 1
} class Driver:People{
var yyyyy = 2
} class SellerPeople{
var zzzzz = 3
} struct Person {
var name: String
var age: Int
} class ViewController {
var x = 123456
var person: Person? {
didSet{ print("Called after setting the new value")
if let name = person?.name {
print("New name is \(name) and old name is \(String(describing: oldValue?.name))")
}
}
willSet(myNewValue) { print("Called before setting the new value")
if let newName = myNewValue?.name {
print("New name is \(newName)")
}
}
} } var p = Driver()
print(type(of: p))
print(String.init(format: "%s", object_getClassName(p))) var v = ViewController()
v.person = Person(name: "1234", age: 123) class Human : ViewController {
var age: Int?
var name: String?
var nicknames: [String] = [String]() //返回指向 Human 实例头部的指针
func headPointerOfClass() -> UnsafeMutablePointer<Int8> {
let opaquePointer = Unmanaged.passUnretained(self as AnyObject).toOpaque()
let mutableTypedPointer = opaquePointer.bindMemory(to: Int8.self, capacity: MemoryLayout<Human>.stride)
return UnsafeMutablePointer<Int8>(mutableTypedPointer)
}
} let human = Human()
let arrFormJson = ["goudan","zhaosi", "wangwu"] //拿到指向 human 堆内存的 void * 指针
let humanRawPtr = UnsafeMutableRawPointer(human.headPointerOfClass()) //nicknames 数组在内存中偏移 64byte 的位置(16 + 16 + 32)
let humanNickNamesPtr = humanRawPtr.advanced(by: 64).assumingMemoryBound(to: Array<String>.self)
//human.nicknames
//[]
let test = humanRawPtr.advanced(by: 0).assumingMemoryBound(to: AnyObject.self)
let test2 = humanRawPtr.advanced(by: 8).assumingMemoryBound(to: Int64.self)
print(test) humanNickNamesPtr.initialize(to: arrFormJson)
print(human.nicknames) //["goudan","zhaosi", "wangwu"] print((human as AnyObject).isKind(of: ViewController.self))
拖入Hopper之后,可以看到

我们可以清晰看到SellerPeople的类的定义,和其MetaClass的定义;其定义跟现有的OC Class类型定义比较类似,一个指针指向父类,一个指向元类,Cache未知、一个vtable表示方法调用列表、data表示类中的变量数据

上面的截图可以看到SellerPeople中ivar的定义和名字方法的定义、通过这些信息可以完全实现一个对象类型的检测、属性列表、方法列表等。
在DEBUG模式生成的可执行文件下面,我们用命令行对其中的符号进行恢复,可以清晰看到类名、变量名。
LUNLI-MC1:Swift 二进制分析 lunli$ nm TestSwift2|xcrun swift-demangle|grep "ViewController"
0000000100002120 t TestSwift2.ViewController.person.didset : TestSwift2.Person?
0000000100002ae0 T TestSwift2.ViewController.person.getter : TestSwift2.Person?
0000000100002d70 T TestSwift2.ViewController.person.materializeForSet : TestSwift2.Person?
0000000100002d40 t closure #1 : () in TestSwift2.ViewController.person.materializeForSet : TestSwift2.Person?
00000001004f3cd8 S direct field offset for TestSwift2.ViewController.person : TestSwift2.Person?
0000000100002110 T variable initialization expression of TestSwift2.ViewController.person : TestSwift2.Person?
0000000100002b60 T TestSwift2.ViewController.person.setter : TestSwift2.Person?
00000001000026f0 t TestSwift2.ViewController.person.willset : TestSwift2.Person?
0000000100002e10 T TestSwift2.ViewController.__allocating_init() -> TestSwift2.ViewController
0000000100002f60 T TestSwift2.ViewController.init() -> TestSwift2.ViewController
0000000100511110 s reflection metadata field descriptor TestSwift2.ViewController
0000000100589e38 b lazy cache variable for type metadata for TestSwift2.ViewController
0000000100001cd0 T type metadata accessor for TestSwift2.ViewController
0000000100584d60 d full type metadata for TestSwift2.ViewController
0000000100584d38 D metaclass for TestSwift2.ViewController
00000001004f3cf0 S nominal type descriptor for TestSwift2.ViewController
0000000100584d70 D type metadata for TestSwift2.ViewController
0000000100003030 T TestSwift2.ViewController.__deallocating_deinit
0000000100002fd0 T TestSwift2.ViewController.deinit
0000000100589ad8 S TestSwift2.v : TestSwift2.ViewController
0000000100507a0e s _symbolic ____ 10TestSwift214ViewControllerC
2)既然能看到任意swift定义的class结构,那么 swift是否存在一个共同的父类结构?
搜索之后看到一个答案:


原文:https://stackoverflow.com/questions/24137368/why-is-there-no-universal-base-class-in-swift
我写了一点代码来模拟这个特性,一段OC和Swift混编的代码
import Foundation
class People{
var xxxxx = 1
}
class Driver:People{
var yyyyy = 2
}
class SellerPeople{
var zzzzz = 3
}
struct Person {
var name: String
var age: Int
}
class ViewController1 : NSObject {
var obj = SellerPeople()
@objc var str = "Hello World"
var x = 123456
var person: Person? {
didSet{
print("Called after setting the new value")
if let name = person?.name {
print("New name is \(name) and old name is \(String(describing: oldValue?.name))")
}
}
willSet(myNewValue) {
print("Called before setting the new value")
if let newName = myNewValue?.name {
print("New name is \(newName)")
}
}
}
@objc func getSwiftObj() -> AnyObject {
return obj
}
}
class Human : ViewController1 {
var age: Int?
var name: String?
var nicknames: [String] = [String]()
//返回指向 Human 实例头部的指针
func headPointerOfClass() -> UnsafeMutablePointer<Int8> {
let opaquePointer = Unmanaged.passUnretained(self as AnyObject).toOpaque()
let mutableTypedPointer = opaquePointer.bindMemory(to: Int8.self, capacity: MemoryLayout<Human>.stride)
return UnsafeMutablePointer<Int8>(mutableTypedPointer)
}
}
注意getSwiftObject方法返回的是一个Swift对象,我们将这个对象拿到OC中用调试器进行调试

可以看到调试器中打印出来,一个纯Swift没有指定父类的类默认继承的对象是SwiftObject,参考这个类的实现如下
https://github.com/apple/swift/blob/f4db1dd7a4abba2685247e1a7415d4fcb91f640d/stdlib/public/runtime/SwiftObject.h
//===--- SwiftObject.h - Native Swift Object root class ---------*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This implements the Objective-C root class that provides basic `id`-
// compatibility and `NSObject` protocol conformance for pure Swift classes.
//
//===----------------------------------------------------------------------===// #ifndef SWIFT_RUNTIME_SWIFTOBJECT_H
#define SWIFT_RUNTIME_SWIFTOBJECT_H #include "swift/Runtime/Config.h"
#include <cstdint>
#include <utility>
#include "swift/Runtime/HeapObject.h"
#if SWIFT_OBJC_INTEROP
#include "llvm/Support/Compiler.h"
#include <objc/NSObject.h>
#endif #if SWIFT_OBJC_INTEROP #if SWIFT_DARWIN_ENABLE_STABLE_ABI_BIT
// Source code: "SwiftObject"
// Real class name: mangled "Swift._SwiftObject"
#define SwiftObject _TtCs12_SwiftObject
#else
// Pre-stable ABI uses un-mangled name for SwiftObject
#endif #if __has_attribute(objc_root_class)
__attribute__((__objc_root_class__))
#endif
SWIFT_RUNTIME_EXPORT @interface SwiftObject<NSObject> {
@private
Class isa;
SWIFT_HEAPOBJECT_NON_OBJC_MEMBERS;
} - (BOOL)isEqual:(id)object;
- (NSUInteger)hash; - (Class)superclass;
- (Class)class;
- (instancetype)self;
- (struct _NSZone *)zone; - (id)performSelector:(SEL)aSelector;
- (id)performSelector:(SEL)aSelector withObject:(id)object;
- (id)performSelector:(SEL)aSelector withObject:(id)object1 withObject:(id)object2; - (BOOL)isProxy; + (BOOL)isSubclassOfClass:(Class)aClass;
- (BOOL)isKindOfClass:(Class)aClass;
- (BOOL)isMemberOfClass:(Class)aClass;
- (BOOL)conformsToProtocol:(Protocol *)aProtocol; - (BOOL)respondsToSelector:(SEL)aSelector;
+ (BOOL)instancesRespondToSelector:(SEL)aSelector;
- (IMP)methodForSelector:(SEL)aSelector;
+ (IMP)instanceMethodForSelector:(SEL)aSelector; - (instancetype)retain;
- (oneway void)release;
- (instancetype)autorelease;
- (NSUInteger)retainCount; - (NSString *)description;
- (NSString *)debugDescription;
@end namespace swift { NSString *getDescription(OpaqueValue *value, const Metadata *type); } #endif #endif
这个类包含一个实例变量isa,这个Class是一个指针,指向的是OC Runtime中类定义的结构;同时这个类实现了NSObject的协议,猜测这样的设计是为了跟OC进行桥接。
3)Swift的对象的内存布局
这里的内容主要来自第一篇提到的文章,一个Swift引用指向的内存区域,
第一个指针应该是isa,
第二个8字节(32位机器为4字节)为引用计数,
第三块区域是vtable指针,
接下来的是继承的变量
接下来是这个类实现的变量
可见这里的对象内存布局和C++有些类似,但是增加了一个isa的东西,让这个对象具有了一些动态特征。方法执行采用vtable寻址的方式,快速高效。
SwiftObject 杂记的更多相关文章
- [Erlang 0118] Erlang 杂记 V
我在知乎回答问题不多,这个问题: "对你职业生涯帮助最大的习惯是什么?它是如何帮助你的?",我还是主动回答了一下. 做笔记 一开始笔记软件做的不好的时候就发邮件给自己, ...
- Ubuntu杂记——Ubuntu下用虚拟机共享上网
由于最近把自己电脑环境换成了Ubuntu,但学校的网络是电信的闪讯,大学里用过的人都知道这货有多坑,而且没有Linux客户端,上网都是问题,怪不得国内用Linux的人那么少,特别是高校的学生(让我瞎逼 ...
- 一个ubuntu phper的自我修养(杂记)
ubuntu使用杂记 1.flatabulous安装使用. flatabulous是一个ubuntu图标主题. 使用它,必须得安装tweak插件. sudo add-apt-repository pp ...
- 有关Java的日期处理的一些杂记
在企业应用开发中,经常会遇到日期的相关处理,说实话JDK自带的日期方法很难用.就我个人而言我一般都会采用joda-time来替代JDK自身的日期. 这篇文章是杂记,所以写的比较零散,希望大家不要见怪. ...
- 分布式系统之CAP理论杂记[转]
分布式系统之CAP理论杂记 http://www.cnblogs.com/highriver/archive/2011/09/15/2176833.html 分布式系统的CAP理论: 理论首先把分布式 ...
- Redis杂记
参考资料: Redis 教程 | 菜鸟教程 : http://www.runoob.com/redis/redis-tutorial.html Redis快速入门 :http://www.yiibai ...
- MySQL杂记
参考资料: w3school SQL 教程 : http://www.w3school.com.cn/sql/index.asp 21分钟 MySQL 入门教程 : http://www.cnblo ...
- Android之开发杂记(一)
1.cygwin环境变量设置 可在Cygwin.bat 中设置 set NDK_ROOT=P:/android/android-ndk-r8e 或者在home\Administrator\.bash_ ...
- ios程序开发杂记
ios程序开发杂记 一.程序构建 与一般的程序构建无太大区别,都是源文件编译链接这一套,通常是在mac上做交叉编译,也就是利用xcode里带的ios编译工具集去生成arm架构的ios程序(或是x86的 ...
- [Flask]学习杂记--模板
这个学习杂记主要不是分享经验,更多是记录下falsk的体验过程,以后做东西在深入研究,因为django之前用的时间比较长,所以很多概念都是一看而过,做个试验了解下flask的功能. flask中使用是 ...
随机推荐
- Web前端 - Vue
<!-- id标识vue作用的范围 --> <div id="app"> <!-- {{}} 插值表达式,绑定vue中的data数据 --> { ...
- IDEA操作MyBatis实现数据库增删改查
"感谢您阅读本篇博客!如果您觉得本文对您有所帮助或启发,请不吝点赞和分享给更多的朋友.您的支持是我持续创作的动力,也欢迎留言交流,让我们一起探讨技术,共同成长!谢谢!" 前置环境 ...
- Advanced .Net Debugging 7:托管堆与垃圾收集
一.简介 这是我的<Advanced .Net Debugging>这个系列的第七篇文章.这篇文章的内容是原书的第二部分的[调试实战]的第五章,这一章主要讲的是从根本上认识托管堆和垃圾回收 ...
- 【pytorch学习】之概率
6 概率 简单地说,机器学习就是做出预测.根据病人的临床病史,我们可能想预测他们在下一年心脏病发作的概率.在飞机喷气发动机的异常检测中,我们想要评估一组发动机读数为正常运行情况的概率有多大.在强化学习 ...
- CF1832B Maximum Sum 题解
[题目描述] 给定一个长度为 \(n\) 的数列,其中每个元素互不相同,进行 \(k\) 次操作,每次可以选择删除序列中最小的两个数或最大的一个数.求操作后剩余数的和的最大值. [思路] 我们构造一组 ...
- 一看即会:Serverless 应用开发的 7 个实战小技巧
简介:干货满满,马住收藏! Serverless 应用开发的 7 个经验心得 作者说:Serverless 架构下的应用开发,与传统架构的应用开发还是有比较大的区别点的,例如天然分布式架构会让很 ...
- SpringBoot Admin2.0 集成 Java 诊断神器 Arthas 实践
简介: 项目最初使用 Arthas 主要有两个目的: 1. 通过 arthas 解决实现测试环境.性能测试环境以及生产环境性能问题分析工具的问题. 2. 通过使用 jad.mc.redefine 功能 ...
- Flink 在有赞的实践和应用
简介: 本文介绍了Flink 在有赞的实践和应用,内容包括:Flink 的容器化改造和实践.Flink SQL 的实践和应用.未来规划. 作者:沈磊 一.Flink 的容器化改造和实践 1. 有赞的集 ...
- Vite + React 组件开发实践
简介: 毫不夸张的说,Vite 给前端带来的绝对是一次革命性的变化.或者也可以说是 Vite 背后整合的 esbuild . Browser es modules.HMR.Pre-Bundling 等 ...
- [FAQ] Cordova 模拟器中不能访问域名, 未联网 ?
首先保证电脑已联网,然后打开模拟器的浏览器输入常用网址,看看是否能够联网. 如果访问失败,在本机中在 cmd 中 ping www.baidu.com 获得百度的ip地址,然后在浏览器中输入 http ...