随笔

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协议中预留给能够将连接改为管道方式的代理服务器。

你可能感兴趣的:(随笔)