原文出处:hherima
本文旨在总结iOS知识网络,知识点,该知识网络罗列出常见UIKit、Foundation的对象特点和一些使用经验,可以看成是一本书;文本编辑采用树的形式,对知识点进行罗列,并标注一些使用经验(★)希望对初学者有用或给一些解决疑难杂症者提供思路;某些知识点会深入探讨;通过总结希望站在一个较高平台的角度全观Objective-C。
知识树中有些是原创文章,有些则是转载网络上iOS大神的文章。由于篇幅的限制笔者会简洁地介绍各个知识点,读者可通过链接了解详情。当然一个人的知识面是相当有限的,在给各位读者提供知识参考的同时,欢迎大家对本文提意见。
/->UIViewController
|ViewController在iOS只是一个非常重要的概念,其在一个App中所扮演的角色:
|(1) View Management:管理View (2) Data Marshalling:管理数据
|(3) User Interactions:响应用户交互 (4) Resource Management:管理资源
|(5) Adaptivity:适配不同的屏幕尺寸空间的变化
|+ (void)initialize +(void)load 的调用时机,区别【点击】
|ViewDidLoad调用时机:当view被首次使用的时候,某些情况可提升性能
|横竖屏的坑【点击】。APP整体是竖屏,单个controller可以是横屏的
|★两种交互方式:push和present
|左右滑动 - (void)pushViewController:(UIViewController *) animated:(BOOL)
|模态,从下往上弹出 - (void)presentViewController:(UIViewController *) animated: (BOOL) completion:
|★还有一种:直接把Controller的view添加到另一个Controller上。
|->UIView
|frame 和bound 的区别【点击】
|frame 是相对父试图坐标的值; bound是本身坐标系统的值
|layoutSubviews【点击】需要将[super layoutSubviews];放到最后,不然iOS7有可能会有这个崩溃
|★“Auto Layout still required after executing -layoutSubviews” iOS7上崩溃sdk缺陷【点击】
|每一个视图有唯一的父视图【点击】。addsubview操作把它从上一个父试图中移除
|善于使用hidden 使用animateWithDuration简单地控制页面切换效果
|使用animateWithDuration简单地控制页面切换效果【点击】
|★简单动画 animateWithDuration【点击】
|->CALayer
|CALayer是个简单的类,它是用来在屏幕上显示内容展示的矩形区域.【容芳志出品点击】
|直接从NSObject继承,少了UIResponder类,固CALayer悲催的不能响应任何用户事件【点击】
|->UIWindow
|每一个IOS程序都有一个UIWindow
|UIWindow有三个层级,分别是Normal,StatusBar,Alert【点击】
|keyWindow是指定的用来接收键盘以及非触摸类的消息,
|而且程序中每一个时刻只能有一个window是keyWindow。
|->UIImage
|加载图片几种方式【点击】
|[UIImage imageNamed:@“xxx”] 系统缓存到cache中
|[UIImage imageWithContentsOfFile:path] 不缓存
|[UIImage imageWithData:data] 不缓存
|★拉伸图片,四角保持不变resizableImageWithCapInsets:
|★加载gif图片【点击】
|->UILabel【点击】
|没有上下居中对齐,可以使用TTTAttributedLabel
|★重写drawTextInRect:方法,可以自定义绘制区域,比如可设置Inset
|[super drawTextInRect:UIEdgeInsetsInsetRect(rect, self.textInsets)];
/
/->UIKit
|
||->UIButton
||设置颜色,文字一定要指定button状态
||善于使用contentEdgeInsets,可以设置文本边距【点击】
||设置圆角可layer.cornerRadius
||->UITextfield
||隐藏键盘,[textfield resignFirstResponder]
||★任意页面隐藏键盘【点击】
||->UIScrollerView
||上拉下拉原理【点击】
||->UITableView
||复用,注意重写 - (void)prepareForReuse
||dequeueReusableCellWithIdentifier 从重用池中获取,可能是nil
||dequeueReusableCellWithReuseIdentifier 同上,但是不会是nil
||插入,删除,移动section或item的顺序,需遵循下面两个步骤【点击】
||1. 更新dataSource的数据
||2. 调用相应的collection view方法删除或者插入section或item
||★非常严格的条件:,在更新collection view之前,先更新datasource,
||因为collection view总是假设你已经准备好打他source了 否则collection view收到错误的item,并造成crash
||右侧音序条【点击】
||UITableView上拉、下拉原理【点击】
||AsyncDisplayKit 流畅的解决方法【点击】
||★tableView正在滚动的时候,如果reloadData,偶尔发生App crash的情况【点击】
||★UITableview Deceleration 加速滑动(惯性滑动)、弹性回归原理【点击】
||->UIDevice
||设备名 [UIDevice currentDevice].name,
||系统版本号 [[UIDevice currentDevice].systemVersion doubleValue];【点击】
||屏幕旋转方向 [[UIDevice currentDevice] orientation]
||区分iPad还是iPhone [UIDevice currentDevice].userInterfaceIdiom);
||->UIScreen
||如何正确的绘制1像素的线【点击】
||保证边距不变,内容等比例拉伸【点击】
||->UIEdgeInsets
||实际显示边距,跟设置边距的距离【点击】
|->自动布局
|Masonry
|storyboard
|
|/->NSObject既是对象也是协议,可以将对象自动置nil 比如 int = 0 bool = NO
||几乎所有类的基类或者协议【点击】
||isKindOfClass:和isMemberOfClass:,通过这两种方法可以确定一个类的从属关系
||后者测试一个接收器是否是一个指定类的实例;而后者可以测试类的从属关系。
||respondsToSelector: 方法测试一个接收器是否通过selector实现(implements)了一个标志符话的方法
||description方法,允许一个对象返回一个字符串来描述它的内容;这个常用于调试debug
||encodeWithCoder: 和 initWithCoder:方法,NSCoding协议中仅有的组成成员
||第一个允许对象编译它的实例变量,第二个允许一个对象初始化它自身的解码实例变量。
||conformsToProtocol:方法,测试接收器(对象或者类)符合一个给定的协议(protocol)
||★类对象中的 isa 指向类结构被称作 metaclass【点击】跟[object class]有点区别,比如KVO的时候
||★__weak如何实现对象值自动设置为nil的【点击】
||->NSString & NSMutableString
||NSString作为属性时候,用copy还是strong修饰?
||strong是单纯的增加对象的引用计数,而copy操作是执行了一次深拷贝【点击】
||->NSArray & NSMutableArray
||NSArray 各种遍历方式,倒序遍历【点击】
||NSArray简便初始化方法@[@"1",@"2"];
||浅拷贝。数组本身使用地址,但是数组item仍是旧对象【Apple 官方解释点击】
||无论copy、arrayWithArray、copyWithZone 数组内对象并没有变。
||只是copy出来的array是新地址,arrayWithArray出来的数组也是新地址。
||深拷贝。数组本身使用地址,但是数组item是新地址
||[[NSArray alloc] initWithArray:someArray copyItems: YES];
||深拷贝时候,数组中的item必须实现NSCopying协议并实现copyWithZone:
||★防止NSArray was mutated while being enumerated
||array包含array的情况深拷贝。NSArray* trueDeepCopyArray = [NSKeyedUnarchiver
||unarchiveObjectWithData: [NSKeyedArchiver archivedDataWithRootObject:oldArray]];
||★containsObject 注意:在对比数组中元素的时候,调用元素的isEqual的返回值。
||->NSDictionary & NSMutableDictionary
||取值时候,最好判断object的类型。 if ([object isKindOfClass:[NSString class]]){ //todo};
||->NSNumber 和 NSInteger NSRange
||前者专门用来装基础类型的对象,把整型、单精度、双精度、字符型等基础类型存储为对象
||->NSNullFMDB数据库,使用的时候崩溃
||JsonKit转换以后会生出相应的[NSNull null]对象【点击】
||->NSData字节缓冲区
||+ (nullable instancetype)dataWithContentsOfURL:(NSURL *)url
||dataWithContentsOfURL 虽然是同步的,但可以结合gcd 异步加载网络图片【点击】
||->NSUserDefaults【点击】
||可用于APP setting默认值不好用,SDK bug
||设置WebVIew的UA【点击】
||->NSDate & NSDateFormatter &NSCalendar
||可判断过去几个小时,还是几天 - (NSDateComponents *) components:fromDate:toDate:options:
||可获取时间戳
||有时候有8小时的时差,解决办法【点击】
||在开发iOS程序时对日期处理的总结【点击】
||->NSCoding & NSCoder仅有的两个方法,数据的序列号和反序列化【点击】
||- (void)encodeWithCoder:(NSCoder *)aCoder;
||- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder;
||->NSCopying & NSZone
||+ (instancetype)allocWithZone:深拷贝,类似于memcpy这样的C方法【点击】
||->NSAutoreleasePool
||降低内存峰值【点击】
||->NSFileManager删除文件的时候先判断是否存在是个好习惯
||->NSTimer
||NSTimer 简单使用【点击】
||★NSTimer定时器时间并不精确,类似于公交车进站,堵车就不准时【点击】
||->NSLog暴力打印,常用于测试【点击】
||->NSClassFromString从字符串获取类。NSStringFromClass,从类名获取字符串
||★不要小瞧这两个API,配合使用,他可以做到代码逻辑解藕的效果。
|/
|->Foundation
|
||->NSIndexPath链式结构;tableview用的比较多【点击】
||初始化 [NSIndexPath indexPathForRow:0 inSection:1];
||->NSError网络变成经常用到
||->NSException
||->NSStringEncoding NSString的编码格式,了解即可【点击】
||->NSProgressIndicator
||->NSBundle 是个目录,包含了程序使用的资源,如图像,声音,编译好的代码,nib文件【点击】
||->NSNetServiceBrowser
||->NSValue可以包装任意一个对象,可以用NSValue将struct存到NSArray和NSDictionary中。【点击】
||->NSURLConnectioniOS9已不再使用
||->NSURLSession&NSURLSessionTask【点击】NSURLSession提供的功能:
||通过URL将数据下载到内存【点击】
||通过URL将数据下载到文件系统
||将数据上传到指定URL
||在后台完成上述功能【点击】
||->NSURLRequest 包装了网络请求的信息【点击】
||->NSInputStream & NSOutputStream socket编程【点击】
||->NSPredicate
||谓语查询,原理和用法都类似于SQL中的where【点击】
||->NSLayoutConstraint
||->NSLock & NSRecursiveLock & NSCondition多线程锁
||最基本的同步锁【点击】
||@synchronized{//todo} 同样也是同步锁【点击】
||事实上信号量也能实现锁的目的,信号量和锁的区别【点击】第二篇【点击】
||->NSMethodSignature
/|配合NSInvocation实现消息转发【点击】
iOS|->NSInvocation直接调用 某个对象的消息【点击】
|iOS中可以直接调用 某个对象的消息 方式有2种performSelector:withObject: 和NSInvocation
||当然,还以用C语言的函数指针,参见下面的“方法调配技术”
||->NSSet无序的对象集合,用处少
||->NSUrl基本使用,包含File URL和File path【点击】
||->AVPlayer基本使用【点击】
||获取视频时间长度【点击】
|->NSNotificationCenter同步的机制【点击】注意防止重复,相似的机制还有delegate,observer,block
|
|/->创建push原理介绍、证书制作、测试push 专辑【点击】
||“iOS push全方位解析(一)【译文】”——iOS PUSH概述【点击】
||“iOS push全方位解析(二)【译文】”——生成OpenSSL证书,Provisioning Profile【点击】
||“iOS push全方位解析(三)【译文】”——一个极简的demo,并测试一下push【点击】
|/
|->Push
|
||★iOS6、7、8、9 Push的演化【点击】,但目前还是不尽人意(APP 无法获取通知栏消息数目)
||★php写的可以在本机发送iOS push程序【点击】
||iOS7 Background Remote Notification(后台远程通知——静默push)【点击】
|->有一些三方push SDK:极光push
|
|->block必须掌握
|block专辑【点击】;Block带有局部变量的匿名函数;iOS开发尤其实用
|【block编程第一篇】 block编程热点介绍(官方文档翻译的)【点击】
|【block编程第二篇】 block捕获变量和对象【点击】
|【block编程第三篇】block内存管理——如何验证block在栈上,还是堆上【点击】
|【block编程第五篇】block中使用 weak–strong dance 技术避免循环引用【点击】
|->多线程
|iOS有三种多线程编程的技术,分别是:【点击】
|1、NSThread 下面会讲到
|2、Cocoa NSOperation 下面会讲到
|3、GCD 下面会讲到
|这三种编程方式从上到下,抽象度层次是从低到高的,抽象度越高的使用越简单。
|dispatch_once 执行一次,用于创建单例【点击】可满足线程安全
|iOS 不像C++ 那样,可以直接将构造函数设成private。所以创建绝对单例模型需重写allocWithZone,【点击】
|
|/->引用计数(retainCount)
||retain 引用计数+1 对象alloc时,引用计数为1, release引用计数-1.引用计数=0时候,真正释放
||autoRelease,自动释放对象【点击】
||->便捷构造方法 iOS
||对象在自动释放池中,不需要开发者手动释放,比如下面的方法
||NSString的stringWithString
||NSArray的arrayWithObjects:和arrayWithArray:
||UIImage的imageNamed:
||->内存管理高级指南【官方译点击】
||->内存管理实践【点击】
|/
|->内存管理【专栏点击】
|
||->MRC手动引用计数。release和retain成对儿
|->ARC自动引用计数
|★强烈建议使用ARC
|★禁止在函数内返回局部变量指针,不然就是野指针!
|★容易引起循环引用的地方【点击】
|★- (id)performSelector:(SEL)aSelector withObject:(id)object;引起警告
|warning:performSelector may cause a leak because its selector【点击】
|
|iOS可通过URL Scheme,调用别的APP(iOS内的应用调用协议),APP 实现 - (BOOL)application: openURL: options:
|
|/->Runtime运行时特点【《运行时之一:类与对象》南峰子出品点击】
||Objective-C程序员可以在程序运行时创建,检 查,修改类,对象和它们的方法【点击】
||Objective-C runtime库也负责找出方法的最终执行代码
||★class Objective-C类是由Class类型来表示的,它实际上是一个指向objc_class结构体的指针。
||struct objc_class {
||Class isa OBJC_ISA_AVAILABILITY;
||#if !__OBJC2__
||Class super_class OBJC2_UNAVAILABLE; // 父类
||const char *name OBJC2_UNAVAILABLE; // 类名
||long version OBJC2_UNAVAILABLE; // 类的版本信息,默认为0
||long info OBJC2_UNAVAILABLE; // 类信息,供运行期使用的一些位标识
||long instance_size OBJC2_UNAVAILABLE; // 该类的实例变量大小
||struct objc_ivar_list *ivars OBJC2_UNAVAILABLE; // 该类的成员变量链表
||struct objc_method_list **methodLists OBJC2_UNAVAILABLE; // 方法定义的链表
||struct objc_cache *cacheOBJC2_UNAVAILABLE; // 方法缓存
||struct objc_protocol_list *protocols OBJC2_UNAVAILABLE; // 协议链表
||#endif
||} OBJC2_UNAVAILABLE;
||1.isa:所有的类自身也是一个对象,这个对象的Class里面也有一个isa指针,它指向metaClass
||2.super_class:指向该类的父类,如果该类已经是最顶层的根类,则super_class为NULL。
||3.cache:用于缓存最近使用的方法。提高方法查找效率
||★objc_object与idobjc_object是表示一个类的实例的结构体
||struct objc_object {
||Class isa OBJC_ISA_AVAILABILITY;
||};
||typedef struct objc_object *id;
||当创建一个类的实例对象时,分配的内存包含objc_object数据结构,然后是类的实例变量的数据。
||NSObject类的alloc和allocWithZone:方法使用函数class_createInstance来创建objc_object数据结构。
||另外还有我们常见的id,它是一个objc_object结构类型的指针
||★meta class元类,是一个类对象的类;它存储着一个类的所有类方法。
||当我们向一个对象发送消息时,runtime会在这个对象所属的这个类的方法列表中查找方法;
||而向一个类发送消息时,会在这个类的meta-class的方法列表中查找。
||★继承体系【点击】
||动态创建类【点击】
||objc_setAssociatedObject【点击】给对象增加属性,一般配合类别使用
||object_getClass 得到一个实例的类【点击】
||objc_copyImageNames 获取指定类所在动态库【南峰子出品点击】
||objc_copyClassList 创建并返回一个指向所有已注册类的指针列表【点击】
||class_xxx系列函数【点击】
||class_copyPropertyList 获取类的属性
||class_addMethod 为类添加方法
||class_isMetaClass 判断是否为元类
||class_getName 获取类名
||class_copyIvarList 拷贝类的实例变量列表
||class_getInstanceMethod 获取实例方法
||->Runnloop【ibireme出品点击】
||RunLoop 实际上就是一个对象,这个对象管理了其需要处理的事件和消息
||并提供了一个入口函数来执行上面 Event Loop 的逻辑
||Run Loop并非iOS平台专属的概念,在任何平台的多线程编程中,为控制线程生命周期【点击】
||接收处理异步消息,都需要类似Run Loop的循环机制来实现:从简单的一个无限顺序
||do{sleep(1);//执行消息}while(true),到高级平台,如Android的Looper,都是类似的机制。
||★PerformSelecter 当调用 NSObject 的 performSelecter:afterDelay: 后,实际上其内部会创建一个 Timer
||并添加到当前线程的 RunLoop 中;所以如果当前线程没有 RunLoop,则这个方法会失效【点击】
|/
|->iOS 动态机制
|
||->消息传递objc_msgSend【点击】【南峰子出品点击】
||obj-c脱胎于smalltalk的消息处理。所有方法调用都是发消息。消息是啥?一串字符【点击】
||如果向某对象传递消息,那就会使用动态绑定机制来决定需要调用的方法
||objc_msgSend有两个隐藏参数,消息接收对象 方法的selector ,即(self,_cmd)
||->消息转发message forwarding【点击】
||就是对象在接收到无法解读的消息之后会发生什么情况
||(1) +(BOOL) resolveInstanceMethod:(SEL)selector 类是否新增一个实例方法
||(2) resolveClassMethod 是否新增了类方法
||(3) -(id) forwardingTargetForSelector:(SEL)selector 能不能把这条消息转给其他接收者来注册
||(4)-(void) forwardInvocation:(NSInvocation*)invocation 消息派发系统
||(5)最后若都不能处理消息,则doesNotRecognizeSelector: 会抛出异常
||★当我们不能确定一个对象是否能接收某个消息时,会先调用respondsToSelector:来判断一下
||->SEL【点击】
||SEL又叫选择器,是表示一个方法的selector的指针,每一个方法都对应着一个SEL。
||OC在编译的时候,会根据方法的名字(包括参数序列),生成一个用 来区分这个方法的唯一的一个ID
||这个ID就是SEL类型的。需要注意的是,只要方法的名字(包括参数序列)相同,那么它们的ID都是相同的。
||就是 说,不管是超类还是子类,不管是有没有超类和子类的关系,只要名字相同那么ID就是一样的
||★方法的定义体里面,我们可以通过访问_cmd得到这个方法自己的SEL。
||->方法调配技术method swizzling 用于调试【南峰子出品点击】
||IMP 它是objetive-C 方法(method)实现代码块的地址,实际上是函数指针,指向方法实现的首地址
||IMP imp = [requestItem.delegateTarget methodForSelector:selector];
||void (*func)(id, SEL, ResponseItem *) = (void *)imp;
||func(requestItem.delegateTarget, selector, responseItem);
||可以从selector获取IMP,比如:- (IMP)methodForSelector:(SEL)aSelector;
||Swizzling应该总是在+load中执行
|在 iOS 中,几乎所有类都是 responder,比如 UIWindow、UIView、UIControl、UIControllers 等
|当手指去触摸屏幕上 UIView 的实例对象 aView。产生一个触摸事件 UIEventTypeTouches
|而接收触摸事件的对象 aView,就是一个 responder object。
|initial view –> super view –> …..–> view controller –> window –> Application –> AppDelegate
|
|/->沙盒(Sandbox)iOS沙盒机制【容芳志出品点击】
||每个应用程序都有自己的存储空间
||应用程序不能翻过自己的围墙去访问别的存储空间的内容
||应用程序请求的数据都要通过权限检测,假如不符合条件的话,不会被放行。
||->GroupiOS8+数据共享,例如扩展(Extension)共享数据【点击】
||->SpotlightiOS9+ 系统搜索。【官方demo点击】
||->GCD(Grand Central Dispatch)iOS开发有一个强有力的多线程工具【点击】
||多线程入门【raywenderlich出品点击】
||系统提供一个叫做 主队列(main queue)
||系统还提供一个叫做全局调度队列(Global Dispatch Queues)有四个优先级
||开发者自己创建队列(串行,或者并行)
||★至少有五个队列任你处置:主队列、四个全局调度队列,再加上任何你自己创建的队列。
||GCD 深入理解:第一部分【点击】
||GCD 深入理解:第二部分【点击】
||开发常见方法介绍
||dispatch_after 延后工作
||Dispatch Groups 会在整个组的任务都完成时通知你
||dispatch_semaphore_t 信号量,让你控制多个消费者对有限数量资源的访问。【点击】
||dispatch_semaphore_wait 使得信号量-1,当=0时候阻塞
||dispatch_semaphore_signal 释放信号量,即信号量+1
||
||->CoreData数据持久化,相比sqlite有下面优势【点击】
||数据库字段或者表有更改会导致crash,CoreData的版本管理和数据迁移变得非常有用,
||手动写sql语句操作还是麻烦一些。
||不光能操纵SQLite,CoreData和iCloud的结合也很好,如果有这方面需求的话优先考虑
||并不是直接操纵数据库,比如:使用CoreData时不能设置数据库的主键,目前仍需要手动操作。
||效率上其实跑程序时感觉不出来,毕竟手机上的数据不能跟网站的数据和访问量相提并论。
|/
|->特殊封装&平台特性
|
||->类别(Category)扩展(Extension)微小区别【点击】
||->KVC键值编码
||在IOS的中,没有绝对的私有,包括方法和变量,可以通过字符获取属性【点击】
||->KVO键值观察,依赖isa-swizzling技术【王中周出品点击】
||依赖Runtime 和KVC 一个新的类会动态被创建。详细原理【点击】另外一篇【点击】
||同时派生类还重写了 class 方法以“欺骗”外部调用者它就是起初的那个类。
||然后系统将这个对象的 isa 指针指向这个新诞生的派生类,因此这个对象就成为该派生类的对象了,
||因而在该对象上对 setter 的调用就会调用重写的 setter,从而激活键值通知机制
||★Person在建立KVO监听前和之后的打印输出 self->isa:Person [self class]:Person
||self->isa:NSKVONotifying_Person [self class]:Person
||★比如:Tableview上拉下拉动画检测offset;播放视频,获取视频时长时候等
||为什么KVO不成对儿调用,会崩溃?
||->多任务
||后台运行一段时间(不是地图,voip类app)【点击】
||->3D Touch,通过在plist中添加菜单,然后app实现下面的方法。进入APP
||- (void)application: performActionForShortcutItem: completionHandler:
||->spotlight通过系统搜索,进入APP【点击】
|->Touch ID如何使用iOS 8 指纹识别,代码、实例【点击】
|->HTTPs
|建立安全链接【点击】
|https进阶【点击】
|
->iOS工具
CrashHlytics Crash统计工具
AFNetworking 和 ASIHttp
SDImage
TMCache
AsyncDisplayKit 是 Facebook 推出的用于保持界面流畅性的框架
——————————————————————————————
参考文献:《Objective-C高级编程:iOS与OS X多线程和内存管理》日本人写的;
《Effective Objective C 2.0:编写高质量iOS与OS X代码的52个有效方法》;
《Objective-C基础教程(第2版)》
网络博客参考(无循序):念茜、南峰子、ibireme、容芳志、唐巧、王巍、董柏然、阮一峰、一片枫叶,王中周,颐和园等博主
参考的公众帐号:《iOSDevTips》
完整的UIKit和Foundation 结构 <点击>