1. 序列帧动画实现
- 1方式1:
[UIImage animatedImageWithImages:**动画图片数组** duration:**持续时间**]; // 可以获取一个能做动画的UIImage对象"gif图片"
- 2 方式2:
self.imageView.animationImages = array; // 装图片的数组(需要做动画的图片数组)
self.imageView.animationDuration = 2; // 动画时间
self.imageView.animationRepeatCount = 1; // 重复次数 0 表示重复
[self.imageView startAnimating]; // 开始序列帧动画
2. 图片浏览器-两种加载plist的方式
- 1方式1,通过文件路径加载(本地)
NSString *path = [[NSBundle mainBundle] pathForResource:@"images.plist" ofType:nil];
NSArray *images = [NSArray arrayWithContentsOfFile:path];
- 2方法2,通过文件 URL 加载(本地/网络)统一资源定位符
- NSURL *url = [[NSBundle mainBundle] URLForResource:@"images.plist" withExtension:nil];
- NSArray *images = [NSArray arrayWithContentsOfURL:url];
3. 图片浏览器-内存问题
3.1 加载图片的两种方法
- 1.1 通过imageNamed方法
系统会进行缓存,程序员无法销毁.
加载图片时图片最好就直接放在Assets文件中
self.imageView.image = [UIImage imageNamed:@"image文件名"];
- 1.2通过imageWithContentsOfFile方法
建立的图片,使用完成之后,会自动被释放.
用路径方式去加载图片时图片不能放在Assets中,因为放在Assets文件中的图片编译后会被打包成.car文件
[UIImage imageWithContentsOfFile:<#(nonnull NSString *)#>];
3.2 如何选择加载图片方式:
常用的图片,(小的按钮/背景)素材,放在 Assets 中,使用 imageNamed 加载,性能高
临时使用的图片,放在项目的Supporting Files文件夹下,获取图片文件后,使用 imageWithContentsOfFile 加载,使用完成立即释放!
4 MVC简单介绍和类前缀
模型 : 数据
视图 : 负责显示
控制器 : 处理逻辑,如跳转界面
类前缀苹果推荐使用三个或三个以上字母,防止重名
5 应用管理-两种加载xib的方式
从 NSBundle加载XIB,只需要在第一个参数传入 XIB 的文件名,注意:没有扩展名
- 方法1,iOS 3.0 的方法
UIView *appView = [[NSBundle mainBundle] loadNibNamed:@"CZAppView" owner:nil options:nil].lastObject;
- 方法2,iOS 4.0 的方法,做了内存优化”如果内存紧张”内存警告,可以自动释放,如果有需要会重新自动加载””
//第一个参数:XIB的名字
//第二个参数:Bundle 名,如果传入 nil,会自动从 mainBundle 获取
UINib *nib = [UINib nibWithNibName:@"CZAppView" bundle:nil];
//第一个、第二个参数,老师没有讲,说自己从来没有用过。就写nil。
UIView *appView = [nib instantiateWithOwner:nil options:nil].lastObject;
6-应用管理-管理XIB的类
管理xib的类一般做两个事件
把加载xib的实现细节封装在此类中
把子控件设置数据的代码也封装在此类的内部,不要放在外面
#import <UIKit/UIKit.h>
@class HMApp;
@interface HMAppView : UIView
// 定义一个属性用来接收数据
@property (nonatomic, strong) HMApp *app;
// 创建appView
+ (instancetype)appView;
@end
在自定义View中,新建类方法返回自己的实例,然后加载xib.
MVC中V负责显示,C只需要拿到这个视图,至于这个视图是怎么来的,C不需要关心.
7 应用管理-九宫格布局
// 定义格子的宽和高
NSInteger appW = 100;
NSInteger appH = 120;
// 定义格子的总数量
NSInteger countOfApps = 80;
// 定义一行有几个格子
NSInteger numberOfCol = 3;
// 定义左边距、右边距、顶部间距
NSInteger marginOfLeft = 10;
NSInteger marginOfRight = 10;
NSInteger marginOfTop = 30;
// 格子之间的间距 视图总宽度-左边距-右边距-(格子宽*一行有几个) / (一行有几个 减 1)
NSInteger marginOfApp = (self.view.bounds.size.width - marginOfLeft - marginOfRight - (appW * numberOfCol)) / (numberOfCol - 1);
// 开始循环布局
for (NSInteger i = 0; i < countOfApps; i++) {
//计算列号 第几个格子的索引 % 每行的个数
NSUInteger col = i % numberOfCol;
// 计算X = 左边间距 + (格子的宽 + 格子之间的间距) * 列号
CGFloat appX = marginOfLeft + (appW + marginOfApp)*col;
//计算行号 行号 = i / 列数
NSUInteger row = i / numberOfCol;
// 计算格子的Y = 顶部间距 + (格子的高 + 格子之间间距) * 行号
CGFloat appY = marginOfTop + (appH + marginOfApp)*row;
// 建立UIView
GMAppView *xib = [GMAppView GMAppView];
xib.appInfo = _appData[i];
//设置frame
xib.frame = CGRectMake(appX, appY, appW, appH);
//把appView添加到控制器的view上
[self.view addSubview:xib];
}
8 字典转模型+MVC
8.1 字典转模型的原因
直接通过字典的键名获取plist中的数据信息,需要直接和数据打交道,如果需要多次使用可能会因为不小心把键名写错,而程序并不报错。鉴于此,可以考虑把字典数据转换成一个模型,把数据封装到一个模型中去,让viewController不再直接和数据打交道,而是和模型交互。
一般情况下,设置数据和取出数据都使用“字符串类型的key”,编写这些key时,编辑器没有智能提示,需要手敲。如:
dict[@"name"] = @"Jack";NSString *name = dict[@"name"];```
##8.2 字典转模型的流程
![](http://upload-images.jianshu.io/upload_images/2248583-c479385a9d8363d9.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
##8.3 注意点
模型应该提供一个可以传入字典参数的构造方法
**同时,还要提供读取Plist,并转为数组的方法**
-(instancetype)initWithDict:(NSDictionary *)dict;
+(instancetype)xxxWithDict:(NSDictionary *)dict;
- (NSArray *)loadPlist;
//方法实现
(instancetype)initWithDict:(NSDictionary *)dictionary{
if (self = [super init]) {
GMModel *appData = [GMModel new]; [appData setValuesForKeysWithDictionary:dictionary]; return appData;
}
return self;
}
- (instancetype)GModelWithDict:(NSDictionary *)dictionary{
return [[self alloc]initWithDict:dictionary];
}
/**
return a array from Plist
@return NSArray
*/
- (NSArray *)loadPlist{
// 获取plist文件的NSURL
NSURL * plistUrl = [[NSBundle mainBundle] URLForResource:@"apps.plist" withExtension:nil];
// 通过Plist 的URL读出文件中的array。
NSArray * array = [NSArray arrayWithContentsOfURL:plistUrl];
NSMutableArray * arrayM = [NSMutableArray array];
// plist中的数组中存放的是一个一个字典,因此遍历数组中的字典。把遍历出来的字典添加入可变数组。
for (NSDictionary * dict in array) {
GMModel * temp = [GMModel modelWithDict:dict];
[arrayM addObject:temp];
}
return arrayM;
}
提示:在模型中合理地使用只读属性,可以进一步降低代码的耦合度。
声明属性
@property (nonatomic, copy) NSString *name;@property (nonatomic, copy) NSString *icon;
// 图像,定义属性时,会生成getter&setter方法,还会生成一个带下划线的成员变量// 如果是readonly属性,只会生成getter方法,同时没有成员变量@property (nonatomic, strong, readonly) UIImage *image;
### 8.3.1 作为模型
- 有和字典一样的属性
- 有一个根据字典返回自己的方法 (类方法、对象方法两种)
- 加载plist并转为数组的方法
### 8.3.2 作为view
- 有一个返回自己的实例的方法(至于怎么创建的 无所谓)
- 在.h 里面有一个用来接收模型类型的属性
- 在.m 里面应该有需要显示值的子控件
- 重写set模型的方法,先模拟系统的_xx=xx,把数据放在子控件当中
### 8.3.3 作为controller
- 解析数据,注意转化成模型.步骤(路径,解析临时数组,创建可变数组,遍历获取字典,字典转模型,把模型添加到可变数组中,返回).
- 根据view提供的方法添加到视图上(c不关心view具体是如何创建的),并且,把解析的模型数据复制给view.h里面用来接收模型类型的属性