学习内容
欢迎关注我的iOS学习总结——每天学一点iOS:https://github.com/practiceqian/one-day-one-iOS-summary
实现轮播图需要注意的地方
需要几张轮播图就设置UIScrollView的contentSize的宽度为自身的几倍
//设置高度为0是为了限制UIScrollView只能在x轴方向滚动,如果设置x为0那么只能在y轴方向滚动 CGSizeMake(自身宽度*n,0)
使用NSTImer定时器定时滚动时避免循环引用
- 这里一般使用系统的BLock方法加上__weak和__strong的使用
给UISCrollView加上图片时,每一张图片的x方向位移是初始偏移量加上图片下标*UISCrollView的宽度
CGRectMake(初始偏移量+i*(bounds.size.width),0,bounds.size.width-2*初始偏移量,bounds.size.height)
定时器的使用需要及时的删除/添加,同时注意创建定时器的方式
使用timerwithInterval创建的定时器不会自动添加到RunLoop中,需要手动开启RunLoop
[[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode]; [[NSRunLoop mainRunLoop] addTimer:timer forMode: UITrackingRunLoopMode]; //或者直接添加到NSRunLoopCommonModes中 [[NSRunLoop mainRunLoop] addTimer:timer forMode: NSRunLoopCommonModes];
通过timerWithTimeInterval和initWithFireDate方法创建出来的定时器,都需要手动加入到RunLoop中才会执行,否则不会执行;但是通过scheduledTimerWithTimeInterval创建出来的定时器是自动加入到RunLoop,而且会自动执行
当用户拖动UIScrollView时需要及时的移除定时器,停止拖动时需要及时的加上定时器
//这里可以设置两个函数,addTimer,removeTimer -(void)addTimer{ __weak typeof(self) weakSelf = self; self.bannerTimer = [NSTimer timerWithTimeInterval:3.0 repeats:YES block:^(NSTimer * _Nonnull timer) { __strong typeof(weakSelf) strongSelf = weakSelf; [strongSelf changePage]; }]; [[NSRunLoop mainRunLoop]addTimer:self.bannerTimer forMode:NSDefaultRunLoopMode]; } -(void)removeTimer{ [self.bannerTimer invalidate]; self.bannerTimer = nil; } //scrollview即将开始被拖动 - (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{ [self removeTimer]; } //scrollview即将停止拖动 - (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate{ [self addTimer]; }
iOS中的系统目录
获取当前应用程序的根目录
NSString* homePath = NSHomeDirectory()
获取应用程序的几个主要目录
Documents
保存用户创建的文档文件的目录,用户可以通过文件分享分享该目录下的文件,在iTunes和iCloud备份时会自动备份该目录
//NSSearchPathForDirectoriesInDomains返回的是一个字符串数组,但是数组里面只有一个元素 NSString* docPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
Library
- 不建议在该目录下保存任何用户相关数据,而是保存APP运行需要的修改数据,用户也可以根据自己的需要进行保存
- Caches
- 建议保存数据缓存使用,在用户的磁盘空间已经使用完毕时可能删除该目录中的文件,在APP试用期间不会被删除,但是在APP没有运行的时候系统可能会删除,需要持久化的数据不建议保存在该目录下,防止被系统删除
- preferences
- 用户偏好存储目录,使用NSUserDefaults或者CFPreferences接口保存的数据都保存在该目录下,编程人员不需要对该目录进行管理,iTunes和iCloud会自动对该目录进行备份
- Application State
- 用来保存与用户数据以外的所有文件,如游戏的新关卡,iTunes和iCloud会自动备份该目录
tmp
- 该目录用来保存临时使用的数据,编程人员应该在数据长时间不使用时主动删除该目录下的文件,在APP不运行期间,系统可能删除该目录下的文件,iTunes和iCloud不会备份该目录的文件
SystemData(暂无介绍)
综上所述,我们保存数据可以保存在Documents、Application State目录下,而preferences用于保存系统接口数据不建议使用,而tmp目录和cache目录中的内容可能会被系统磁盘压力紧张时删除
关于Runloop的一些知识点
RunLoop是什么?
RunLoop是一种事件循环,一般的while循环会导致CPU进入忙等状态,而RunLoop则是一种闲等待,当没有事件时,RunLoop会进入休眠状态,当有事件时,Runloop回去寻找相应的Handler处理事件,Runloop可以让线程在需要做事的时候忙起来,在不需要的时候进入休眠状态
do{ //获取消息 //处理消息 }while(消息!=退出)
一个Runloop对象,包含了一个线程,若干个mode,还有当前运行的mode
Runloop和线程是绑定在一起的,每个线程都有一个对应的Runloop对象,我们不能自己创建Runloop对象,但是可以获取到系统提供的Runloop对象
主线程的Runloop会在系统启动时完成启动,其他线程的Runloop默认并不会启动,需要我们手动启动
RunLoop Mode
- Mode可以视为事件的管家,一个Mode管理着各种事件
- Mode实际上是source,observer,timer的集合,不同的Mode把不同组的source,observer,timer隔绝开,Runloop在某个时刻只能运行在一个Mode下,处理这一个Mode下的source,observer,timer
- 苹果文档中提到的Mode共有五个
- NSDefaultRunLoopMode
- NSConnectionReplyMode
- NSModePanelRunLoopMode
- NSEventTrackingRunLoopMode
- NSRunLoopCommonModes
- 但是公开暴露出来接口的只有NSDefaultRunLoopMode和NSRunLoopCommonModes
- NSRunLoopCommonModes是一个Mode的集合,包含了NSDefaultRunLoopMode和NSEventTrackingRunLoopMode,并不是说RunLoop会运行在这两种模式下,而是相当于分别注册了default和eventtracking两种模式。
RunLoop Source
- RunLoopSource分为source,timer,observer三种,它们被称作ModeItem
获取RunLoop
//获取当前线程的RunLoop,子线程的RunLoop要在子线程获取 +(NSRunLoop*)currentRunLoop; //获取主线程的RunLoop +(NSRunLoop*)mainRunLoop;