readwrite,readonly,assign,retain,copy,nonatomic 属性的作用
@property是一个属性访问声明,括号内支持以下几个属性:
getter=getName,setter=setName,设置setter与 getter的方法名
readwrite,readonly,设置可供访问级别
assign,setter方法直接赋值,不进行任何retain操作,为了解决原类型与环循引用问题
retain,setter方法对参数进行release旧值再retain新值,所有实现都是这个顺序
copy,setter方法进行Copy操作,与retain处理流程一样,先旧值release,再 Copy出新的对象,retainCount为1。这是为了减少对上下文的依赖而引入的机制。
nonatomic,非原子性访问,不加同步, 多线程并发访问会提高性能。注意,如果不加此属性,则默认是两个访问方法都为原子型事务访问。锁被加到所属对象实例级
在一个对象的方法里面:self.name= “object”;和 name =”object” 有什么不同吗?
self.name ="object":会调用对象的setName()方法;
name = "object":会直接把object赋值给当前对象的name属性。
请简述self.name= nil的机制,以及与[namerelease]的区别?
self.name =nil;
[name release]生成的访问器将自动释放以前的name对象
请简要说明viewDidLoad和viewDidUnload何时调用?
viewDidLoad在view从nib文件初始化时调用,
loadView在controller的view为nil时调用。
此方法在编程实现view时调用,view控制器默认会注册memory warning notification,
当view controller的任何view没有用的时候,
viewDidUnload会被调用,在这里实现将retain的view release,如果是retain的IBOutlet view 属性则不要在这里release,IBOutlet会负责release 。
实例化一个UITableView对象,要求写出关键语句
UITableView *my = [[UITableView alloc]initWithFrame:<(CGRect)frame> style:<(UITableViewStyle)style>];
my.delegate = self;
my.dataSource = self;
首先需要分配空间设置表格类型
然后需要设置两个必须的委托对象。
数组和指针的区别
(1)数组可以申请在栈区和数据区;指针可以指向任意类型的内存块
(2)sizeof作用于数组时,得到的是数组所占的内存大小;作用于指针时,得到的都是4个字节的大小
(3)数组名表示数组首地址,值不可以改变,如不可以将++作用于数组名上;普通指针的值可以改变,如可将++作用于指针上
(4)用字符串初始化字符数组是将字符串的内容拷贝到字符数组中;用字符串初始化字符指针是将字符串的首地址赋给指针,也就是指针指向了该数组
static的作用
(1)函数体内static 变量的作用范围为该函数体,不同于 auto 变量,该变量的内存只被分配一次,
因此其值在下次调用时仍维持上次的值;
(2)在模块内的static 全局变量可以被模块内所用函数访问,但不能被模块外其它函数访问;
(3)在模块内的static 函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明
它的模块内;
(4)在类中的static 成员变量属于整个类所拥有,对类的所有对象只有一份拷贝;
(5)在类中的static 成员函数属于整个类所拥有,这个函数不接收 this 指针,因而只能访问类的static 成员变量。
简述内存分区情况
(1)代码区:存放函数二进制代码
(2)数据区:系统运行时申请内存并初始化,系统退出时由系统释放。存放全局变量、静态变量、常量
(3)堆区:通过malloc等函数或new等操作符动态申请得到,需程序员手动申请和释放
(4)栈区:函数模块内申请,函数结束时由系统自动释放。存放局部变量、函数参数
简述Swift的派发机制
swift派发的目的是让CPU知道被调用的函数在哪里。
1)函数的派发机制:静态派发(直接派发)、函数表派发、消息派发 2)Swift派发机制总结:Swift中所有ValueType(值类型:Struct、Enum)使用直接派发; Swift中协议的Extensions使用直接派发,初始声明函数使用函数表派发; Swift中Class中Extensions使用直接派发,初始声明函数使用函数表派发,dynamic修饰的函数使用消息派发; Swift中NSObject的子类用@nonobjc或final修饰的函数使用直接派发,初始声明函数使用函数表派发,dynamic修饰的Extensions使用消息派发; 3)Swift中函数派发查看方式: 可将Swift代码转换为SIL(中间码) swiftc -emit-silgen -O example.swift
Swift如何显示指定派发方式?
添加final关键字的函数使用直接派发 添加static关键字函数使用直接派发 添加dynamic关键字函数使用消息派发 添加@objc关键字的函数使用消息派发 添加@inline关键字的函数会告诉编译器可以使用直接派发
Struct和Class的区别?
1)Struct不支持继承,Class支持继承 2)Struct是值类型,Class是引用类型 3)Struct使用let创建不可变,Class使用let创建可变 4)Struct无法修改自身属性值,函数需要添加mutating关键字 5)Struct不需要deinit方法,因为值类型不关系引用计数,Class需要deinit方法 6)Struct初始化方法是基于属性的
?,??的区别
?用来声明可选值,如果变量未初始化则自动初始化nil;在操作可选值时,如果可选值时nil则不响应后续的操作;使用as?进行向下转型操作;
?? 用来判断左侧可选值非空(not nil)时返回左侧值可选值,左侧可选值为空(nil)则返回右侧的值。
Swift中mutating的作用
Swift中协议是可以被Struct和Enum实现的,mutating关键字是为了能在被修饰的函数中修改Struct或Enum的变量值,对Class完全透明。
Set(集合类型)的使用场景
Set存储值类型相同、无序、去重
final关键词的用法
final关键词的作用:它修饰的类、方法、变量是不能被继承或重写的,编译器会报错。另外,通过它可以显示的指定函数的派发机制。
lazy关键词的用法
lazy关键词的作用:指定延时加载(懒加载),懒加载存储属性只会在首次使用时才会计算初始值属性。懒加载属性必须声明(var)为变量,因为常量属性(let)初始化之前会有值。lazy修饰的属性非线程安全的。
Swift中的访问控制权限
Open:实体可被同一模块内所有实体访问,模块外可导入该模块即可访问,模块外可被继承和重写。 Public:实体可被同一模块内所有实体访问,模块外可导入该模块即可访问,模块外不能被继承和重写。 Internal:实体可被同一模块内所有实体访问,模块外无法访问。大部分实体默认是Internal级别。 fileprivate:限制实体只能在它定义的文件内部(源文件)访问。 private: 限制实体只能在它定义的作用域内及同一文件extension中访问。