Swift探索(一): 类与结构体(上)

1.类与结构体

1.1 类

class PersonClass {
   
   //1.类中可以定义存储值的属性
   var age : Int = 18
   var name : String = "小明"
   
   // 2.类中可以定义方法
   func test() {
       print("%@多少%@岁", self.name, self.age);
   }
   
   // 3.类中可以定义初始化器
   // 指定初始化器 默认只有一个
   init(_ age: Int, _ name: String) {
       self.age = age
       self.name = name
   }
   
   // 便捷初始化器 必须从相同的类里调用另一个初始化器
   // 安全性能原则:self及其属性已经初始化了
   convenience init(_ age: Int) {
       self.init(18, "小明")
       self.age = age
       self.name = "小王"
   }
   
   // 可失败初始化
   init?(age: Int, name: String) {
       if age < 18 {return nil}
       self.age = age
       self.name = name
   }
   
   // 必要初始化器 所有该类的子类都必须 实现该初始化器
   required init(age1: Int, name1:String) {
       self.age = age1
       self.name = name1
   }
   
   // 类有析构函数, 结构体没有
   deinit {
   }
}

class SubPersonClass: PersonClass {
   var subName: String
   // 子类的所有属性必须在父类初始化之前初始化完成
   // 子类初始化器必须先向上委托父类初始化器,然后才能为继承的属性设置新值。如果不这样做,指定初始化器赋予的新值将被父类中的初始化器所覆盖
   // 确保成员变量访问安全
   init(_ subName : String) {
       self.subName = subName;
       super.init(18, "小明")
   }
}

1.2 结构体

// 结构体
struct PersonStruct {
    
    //1.结构体中可以定义存储值的属性
    var age : Int = 18
    var name : String = "小明"
    
    // 2.结构体中可以定义方法
    func test() {
        print("%@多少%@岁", self.name, self.age);
    }
    
    // 3.结构体中可以定义初始化器
    // 与类不同,编译器会提供默认的初始化方法(前提是我们自己没有指定初始化器)
    init(age : Int, name : String) {
        self.age = age
        self.name = name
    }
}

1.3 类和结构体的相同点

  • 定义存储值的属性
  • 定义方法
  • 定义下标以使用语法提供对其值的访问
  • 定义初始化器
  • 使用extension来拓展功能
  • 遵循协议来提供某种功能

1.4 类和结构体的不同点

  • 类有继承的特性,而结构体没有
  • 类型转换使您能在运行时检查和解释class的实例对象的类型
  • 类有析构函数用来释放其占用的资源, 而结构体没有
  • 类有引用计数,允许对一个类实例有多个引用
  • 类是引用类型,结构体是值类型

2. 引用类型和值类型

  • 引用类型的变量并不直接存储具体的实例对象,而是对当前存储具体实例内存地址的引用。
  • 值类型的变量存储的就是具体的实例(或者说具体的值)
  • 引用类型存储在
  • 值类型存储在

借助几个LLDB指令来查看当前变量的内存结构

  • po 只会输出对应的值
  • p 则会输出返回值的类型以及命令结果
  • x/8g: 读取内存中的值(8g: 8字节格式输出)
  • frame varibale -L xxx:查看各实例内存信息

2.1 引用类型

引用赋值给varlet或者给函数传参,是将内存地址拷贝一份。类似于制作一个文件的替身(快捷方式、链接),指向的是同一个文件。属于浅拷贝

p和p1都引用了同一个PersonClass的实例地址.png

pp1都引用了同一个PersonClass的实例地址

p和p1在内存中的地址.png

pp1在内存中的地址相差8个字节(存储着当前实例对象的内存地址)

类的内存地址在堆空间.png

2.2 值类型

值类型赋值给 varlet或者给函数传参,是直接将所有内容拷贝一份。类似于对文件进行复制粘贴操作,产生了全新的文件副本。属于深拷贝。

ps和ps1是内存当中存储的值.png

psps1是内存当中存储的值,并且更改ps1当中的agenameps没有影响,因为psps1存储的是两个不同的值

结构体的内存地址在栈空间.png

3.类的生命周期

iOS开发的语言不管是OC还是Swift后端都是通过LLVM进行编译的,如下图所示:

OC和Swift都是通过LLVM进行编译.png

OC通过clang 编译器,编译成IR,然后再生成可执行文件 .o(这里也就是我们的机器码)
Swift则是通过Swift编译器编辑成IR,然后再生成可执行文件,具体编译过程如下:
Swift编译流程.png

  • swift code 经过-dump-parse的命令来进行语法分析,解析成我们的AST(也就是抽象语法树);
  • 通过-dump-ast语义分析来检查我们的语义,(例如类型检查是否准确安全);
  • 完成之后Swift代码将会降级为SIL,也就是Swift中间语言/代码(Swift intermediate language);
  • SIL分为Raw SIL(原生的,没有开启优化选项) 和SILOpt Canonical SIL(经过优化的);
  • 最后通过LLVM降级为IR,然后通过后端代码编译为不同架构的机器码

我们也可以通过swiftc命令去生成对应的文件

// 分析输出AST
swiftc main.swift -dump-parse
// 分析并且检查类型输出AST 
swiftc main.swift -dump-ast
// 生成中间体语言(SIL),未优化
swiftc main.swift -emit-silgen
// 生成中间体语言(SIL),优化后的 
swiftc main.swift -emit-sil
// 生成LLVM中间体语言 (.ll文件) 
swiftc main.swift -emit-ir
// 生成LLVM中间体语言 (.bc文件) 
swiftc main.swift -emit-bc
// 生成汇编
swiftc main.swift -emit-assembly
// 编译生成可执行.out文件 
swiftc -o main.o main.swift

SIL文件中的重要标识可参考:SIL官方文档

接下来开始分析类的初始化

class Person {
    var age : Int = 18
    var name : String = "小明"
}

var p = Person()

通过swiftc main.swift -emit-sil > ./main.sil生成.sil文件,在.sil文件中 注意到__allocating_init()创建实例对象的函数

// Person.__allocating_init()
sil hidden [exact_self_class] @$s4main6PersonCACycfC : $@convention(method) (@thick Person.Type) -> @owned Person {
// %0 "$metatype"
bb0(%0 : $@thick Person.Type):
  %1 = alloc_ref $Person                          // user: %3
  // function_ref Person.init()
  %2 = function_ref @$s4main6PersonCACycfc : $@convention(method) (@owned Person) -> @owned Person // user: %3
  %3 = apply %2(%1) : $@convention(method) (@owned Person) -> @owned Person // user: %4
  return %3 : $Person                             // id: %4
} // end sil function '$s4main6PersonCACycfC'

其中Person.Type元类型的作用类似于 OC 里的isa
alloc_ref根据官方文档中的解释

Allocates an object of reference type T. The object will be initialized with retain count 1; its state will be otherwise uninitialized. The optional objc attribute indicates that the object should be allocated using Objective-C's allocation methods (+allocWithZone:)(可以理解为去堆区申请内存空间)

进入汇编我们可以看到__allocating_init的调用

__allocating_init的调用.png

进入到__allocating_init内部
__allocating_init内部.png

可以看到__allocating_init内部调用了一个函数swift_allocObject
通过 Swift源码 中的HeapObject.cpp文件可以发现

HeapObject *swift::swift_allocObject(HeapMetadata const *metadata,
                                     size_t requiredSize,
                                     size_t requiredAlignmentMask) {
  CALL_IMPL(swift_allocObject, (metadata, requiredSize, requiredAlignmentMask));
}

SWIFT_RUNTIME_EXPORT
HeapObject *(*SWIFT_RT_DECLARE_ENTRY _swift_allocObject)(
    HeapMetadata const *metadata, size_t requiredSize,
    size_t requiredAlignmentMask) = _swift_allocObject_;

swift_allocObject走的是内部函数_swift_allocObject_,进入到_swift_allocObject_

static HeapObject *_swift_allocObject_(HeapMetadata const *metadata,
                                       size_t requiredSize,
                                       size_t requiredAlignmentMask) {
  assert(isAlignmentMask(requiredAlignmentMask));
  auto object = reinterpret_cast<HeapObject *>(
      swift_slowAlloc(requiredSize, requiredAlignmentMask));

  // NOTE: this relies on the C++17 guaranteed semantics of no null-pointer
  // check on the placement new allocator which we have observed on Windows,
  // Linux, and macOS.
  new (object) HeapObject(metadata);

  // If leak tracking is enabled, start tracking this object.
  SWIFT_LEAKS_START_TRACKING_OBJECT(object);

  SWIFT_RT_TRACK_INVOCATION(object, swift_allocObject);

  return object;
}

跳转至swift_slowAlloc

void *swift::swift_slowAlloc(size_t size, size_t alignMask) {
  void *p;
  // This check also forces "default" alignment to use AlignedAlloc.
  if (alignMask <= MALLOC_ALIGN_MASK) {
#if defined(__APPLE__) && SWIFT_STDLIB_HAS_DARWIN_LIBMALLOC
    p = malloc_zone_malloc(DEFAULT_ZONE(), size);
#else
    p = malloc(size);
#endif
  } else {
    size_t alignment = (alignMask == ~(size_t(0)))
                           ? _swift_MinAllocationAlignment
                           : alignMask + 1;
    p = AlignedAlloc(size, alignment);
  }
  if (!p) swift::crash("Could not allocate memory.");
  return p;
}

因此我们可以看出Swift对象的内存分配流程如下
__allocating_init--->swift_allocObject ---> _ swift_allocObject ---> swift_slowAlloc--->Malloc
_swift_allocObject_函数里我们可以看到返回的是HeapObject类型,因此Swift对象内存结构为HeapObject

进入到HeapObject

/// The Swift heap-object header.
/// This must match RefCountedStructTy in IRGen.
struct HeapObject {
  /// This is always a valid pointer to a metadata object.
  HeapMetadata const *__ptrauth_objc_isa_pointer metadata;

  SWIFT_HEAPOBJECT_NON_OBJC_MEMBERS;

#ifndef __swift__
  HeapObject() = default;

  // Initialize a HeapObject header as appropriate for a newly-allocated object.
  constexpr HeapObject(HeapMetadata const *newMetadata) 
    : metadata(newMetadata)
    , refCounts(InlineRefCounts::Initialized)
  { }
  
  // Initialize a HeapObject header for an immortal object
  constexpr HeapObject(HeapMetadata const *newMetadata,
                       InlineRefCounts::Immortal_t immortal)
  : metadata(newMetadata)
  , refCounts(InlineRefCounts::Immortal)
  { }

#ifndef NDEBUG
  void dump() const SWIFT_USED;
#endif

#endif // __swift__
};

我们可以看到HeapObject有两个属性,默认占用 16 字节大小:

  • 一个是 HeapMatadata类型的metadata,8字节,
  • 一个是refCount,8字节

通过源码最后可以得出

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

推荐阅读更多精彩内容