1.appearance
用appearance拿到对象,然后用拿到的对象设置被appearance注释()的属性,就会使整个项目的这种控件都变为设置的样子,例如:
UISwitch *switch = [UISwitch appearance];
switch.onTintColor = [UIColor redColor];
这样设置就会使整个项目的switch控件开启颜色变为红色
注意:如果该控件已经显示出来,则设置无效,要想设置有效需要将整个父控件移除,然后再加回去.
2.隐藏宏
在Build Settings中搜索DEBUG,在Debug那一栏中添加宏,这样就可以在项目中使用该宏,但无法找到宏的详细内容
注意:宏命名的时候不能全是小写字母,至少有一个大写字母或至少有一个数字
3.只执行一次的操作
如果代码块想在程序中只执行一次,可以用dispatch_once或者懒加载
4.tabBar
需求:在tabBar中央加一个按钮
解决思想:重写tabBar,在layoutSubview方法中,重新布局系统自带的按钮(UITabBarButton,系统私有类),在中央空出一块地方,然后添加按钮
1.cell之间分割线(重写setFrame方法)
思路:
在cell原有的高度下减1,这样就可以实现不用添加View就有分割线,且分割线和背景色一样
具体实现:
由于cell的重复利用机制,所以在设置的时候改是无法实现的,但是每次重用cell的时候都会调用cell的setFrame方法,所以在自定义cell中重写setFrame方法,并且在调用[super setFrame]之前,将高度减1,这样就可以实现不添加View就有分割线这一需求.不光如此,还可以设置左右边距,将x加1,将width减2.
扩展:
我们自己封装的控件,不想在外界被人修改size,可以在[super setFrame]之前设置一个固定的size,但是bounds也会修改size,所以也要重写setBounds方法
2.网络请求的成功和失败
当域名错误时,请求就会失败;当域名正确但参数错误时,请求返回的数据依然是成功的,只是返回一个空数据,所以在请求成功的block块中也要进行一次判断,判断返回数据是否为空
3.网络请求延时
在网络请求时,为提高用户体验,在屏幕中央添加一个蒙版,提示用户正在网络请求.但有时网络请求延时很高,以防引起用户不耐,所以在请求时,用户随时可以点击返回按钮退出界面,并且在dealloc中移除蒙版.但是block中强引用控制器,造成控制器无法销毁,造成在退出界面后,蒙版依然存在这种情况
解决方案:不要让block强引用控制器(在block块中不使用self),或者将self包装成弱指针:
__weak typeof(self) weakSelf = self;
但是还有问题,以AFNetworking为例,由于网络管理者在内部被强引用,所以在控制器销毁的时候,依然在进行网络请求.
解决方法1:用weak类型的属性引用网络管理者,在销毁控制器的时候调用:
[self.manager.tasks makeObjectsPerformSelector:@selector(cancel)];
这个方法的意思:取消所有正在网络请求的任务,执行这个方法之后,管理者依然可用.
解决方法2:依然是用weak类型属性医用网络管理者,在销毁控制器的时候调用:
[self.manager invalidateSessionCancelingTasks:YES];
这个方法的意思:使session停止请求,执行这个方法之后,session销毁,管理者和session是一对一的,导致管理者不能用.
但是还有问题!取消任务也被认为是请求失败,会执行请求失败的block代码块,所以在退出界面后弹出一个请求失败的弹框,所以要在请求失败的block代码块中判断error的错误代码,取消请求的错误代码是:NSURLErrorCancelled(-999)
4.weak和IBOutlet
被weak引用的属性,当没有强指针指向属性时,属性会立即销毁;如果被weak引用的属性有IBOutlet修饰时,并不会被马上销毁,而是在代码块过后销毁,因为在IBOutlet内部有个隐藏的强指针引用着.
5.layer(图层)
layer是一个相对耗时的操作,会造成卡顿,尽量少使用layer
6.const和宏
宏相对比较消耗性能
const只是拿常量的值使用,性能较好,但是const只能修饰常量,不能修饰方法和编译之前不确定的值.const修饰右边的内容,例如:
NSString *const name = @"wbw"; 表示不能修改指针的指向
NSString const *name = @"wbw"; 表示不能修改指针指向的内容地址的值
7.常量书写规范
仅限文本件使用的不可变全局变量,要加上:static和const,例如:
static NSString *const name = @"wbw";
整个项目都可以使用的不可变全局变量,创建一个.h和.m文件,在.m文件中创建被const修饰的变量(例如: NSString *const name = @"wbw";),在.h文件中声明(例如: UIKIT_EXTERN NSString *const name;),在.pch文件中导入.h文件
注:UIKIT_EXTERN就是extern,苹果内部包装了一层,可能内部有所优化
8.控件不能响应点击事件,原因可能有
1.userInteractionEnabled = NO;
2.enabled = NO;
3.父控件的userInteractionEnabled = NO;
4.父控件的enabled = NO;
5.控件已超出父控件的边框范围
9.自定义tableView的footerView
先将tableView的footerView替换成自定义的footerView,然后设置自定义footerView的高度时,发现设置高度无效,而自定义footerView的高度又必须根据子控件而变化,解决方法:
1.在计算完子控件高度后,将自定义footerView重新设置为tableView的footerView:
UITableView *tableView = (UITableView *)self.superView;
tableView.tableFooterView = self;
2.根据公式计算出行数,行数*子控件的高度,设置tableView的contentSize
计算行数的公式: 总行数 = (总个数 + 每行的个数 - 1) / 每行的个数
10.webView加载网页
加载网页需要从模型中拿URL时,不能在set方法中加载网页,因为此时的webView是空的,set方法先调用,viewDidLoad后调用,必须在viewDidLoad里面加载网页
11.清除缓存
步骤1.拿到手机缓存的数据大小:
方法1:使用SDWebImage框架中的SDImageCache.h
[SDImageCache sharedImageCache].getSize
注意:只能计算用SDWebImage下载的图片所在文件夹的大小
方法2:用文件管理者遍历拿到所有子文件
//总大小
NSInteger size = 0;
//文件路径
NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
NSString *file = [cache stringByAppendingPathComponent:@"default"];
//路径管理者
NSFileManager *manager = [NSFileManager defaultManager];
//遍历拿到所有的子路径(包括子路径的子路径)
NSArray *subpaths = [manager subpathsAtPath:file];
//遍历子路径
for (NSString *subpath in subpaths) {
//根据子路径拼接出全路径
NSString *fullSubpath = [file stringByAppendingPathComponent:subpath];
//文件的所有属性
NSDictionary *attrs = [manager attributesOfItemAtPath:fullSubpath error:nil];
//将文件的大小加到总大小上
size += [attrs[NSFileSize] integerValue];
}
方法3:NSDirectoryEnumerator(迭代器)遍历路径的所有子文件
方法3:
//总大小
NSInteger size = 0;
//文件路径
NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
NSString *file = [cache stringByAppendingPathComponent:@"default"];
//路径管理者
NSFileManager *manager = [NSFileManager defaultManager];
//遍历路径下的所有子文件
NSDirectoryEnumerator *enumerator = [manager enumeratorAtPath:file];
//遍历enumerator(类似于数组)
for (NSString *subpath in enumerator) {
//根据子路径拼接出全路径
NSString *fullSubpath = [file stringByAppendingPathComponent:subpath];
//文件的所有属性
NSDictionary *attrs = [manager attributesOfItemAtPath:fullSubpath error:nil];
//将文件的大小加到总大小上
size += [attrs[NSFileSize] integerValue];
}
步骤2:删除文件夹
[[NSFileManager defaultManager] removeItemAtPath:@"路径" error:nil];
*注意点
1.由于遍历比较耗时,所以要开启新的线程用来计算文件的大小
2.为严谨起见,要判断路径是否存在,而且要判断路径是否是文件夹
//判断是否是文件夹
BOOL isDirectory = NO;
//判断路径是否存在
BOOL exists = [manager fileExistsAtPath:@"路径" isDirectory:&isDirectory];
//如果是文件,就直接计算大小
if (isDirectory == NO) {
[manager attributesOfItemAtPath:@"路径" error:nil].fileSize;
}
3.在计算的时候应该禁止用户点击cell,待计算完毕之后,恢复点击
cell.userInteractionEnabled;
12.tableView的行被选中后,马上弹起(其实就是取消选中)
调用选中时的代理方法(didSelectRowAtIndexPath),在方法内部调用:
[tableView deselectRowAtIndexPath:indexPath animated:YES];
13.不同标识的cell
当众多cell中有个别特殊的cell时,在循环利用的时候会引起混乱
解决的办法:可以设置多个标识,注册不同的cell,每次循环利用的时候,对应行只加载对应标识的cell
14.正在执行动画的控件移出再回来后,恢复动画
当正在执行动画的cell被移出界面,再重新利用回来后,动画不再执行
解决的办法:拿到执行动画的控件,再次开启动画
15.layoutIfNeeded,setNeedsLayout,setNeedsDisplay的区别**
//重新刷新自己和子控件的所有内容(状态,尺寸)
[testView layoutIfNeeded];
//重新调用testView的layoutSubviews(重新排布子控件的frame)
[testView setNeedsLayout];
//重新调用testView的drawRect:方法(重新绘制testView里面的内容,一般不包括子控件)
[testView setNeedsDisplay];
- HTTP/1.1协议中共定义了八种方法(有时也叫“动作”)来表明Request-URI指定的资源的不同操作
方式:
OPTIONS 返回服务器针对特定资源所支持的HTTP请求方法。也可以利用向Web服务器发送'*'的请求来测试服务器的功能性。
HEAD 向服务器索要与GET请求相一致的响应,只不过响应体将不会被返回。这一方法可以在不必传输整个响应内容的情况下,就可以获取包含在响应消息头中的元信息。
GET 向特定的资源发出请求。注意:GET方法不应当被用于产生“副作用”的操作中,例如在web app.中。其中一个原因是GET可能会被网络蜘蛛等随意访问。
POST 向指定资源提交数据进行处理请求(例如提交表单或者上传文件)。数据被包含在请求体中。POST请求可能会导致新的资源的建立和/或已有资源的修改。
PUT 向指定资源位置上传其最新内容。
DELETE 请求服务器删除Request-URI所标识的资源。
TRACE 回显服务器收到的请求,主要用于测试或诊断。
CONNECT HTTP/1.1协议中预留给能够将连接改为管道方式的代理服务器。