总结一次iOS面试

作者: Lucifron | 来源:发表于2016-09-01 23:51 被阅读346次

    昨天去了网易参加了次iOS面试,参观了下网易的办公环境,也认识到了一些不足。分享下部分问到的问题。(有问题欢迎评论)

    weak 和 assign 的区别?

    assign 可以用作对象和基本数据类型,使用之后如果没有置为nil,可能就会产生野指针。
    weak 只可以用于对象, 一旦不进行使用后,默认会被置为nil,就不会产生野指针。

    weak是如何将不用的指针置为nil?

    runtime 对注册的类, 会进行布局,对于 weak 对象会放入一个 hash 表中。 用 weak 指向的对象内存地址作为 key,当此对象的引用计数为0的时候会 dealloc,假如 weak 指向的对象内存地址是a,那么就会以a为键, 在这个 weak 表中搜索,找到所有以a为键的 weak 对象,从而设置为 nil。

    Post 和 Put 请求区别?

    POST
    用于提交请求,可以更新或者创建资源,是非幂等的。如果次发出同样的POST请求后,其结果是创建出了若干的资源。

    PUT
    用于向指定的URI传送更新资源,是幂等的。比如用PUT修改一篇文章,然后在做同样的操作,每次操作后的结果并没有不同。

    创建操作可以使用POST,也可以使用PUT,区别在于POST 是作用在一个集合资源之上的(/uri),而PUT操作是作用在一个具体资源之上的(/uri/xxx),再通俗点说,如果URL可以在客户端确定,那么就使用PUT,如果是在服务端确定,那么就使用POST,比如说很多资源使用数据库自增主键作为标识信息,而创建的资源的标识信息到底是什么只能由服务端提供,这个时候就必须使用POST。

    CoreData 是否线程安全?

    “In Core Data, the managed object context can be used with two concurrency patterns, defined by NSMainQueueConcurrencyType and NSPrivateQueueConcurrencyType.”
    (Developer Library Core Data)

    关于其他类的线程安全可参考 :
    https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/Multithreading/ThreadSafetySummary/ThreadSafetySummary.html

    Swift中map和flatmap区别?

    flatMap很像map函数,但是它摒弃了那些值为nil的元素。而且flapMap函数能够将可选类型(optional)转换为非可选类型(non-optionals),且对多维数组使用会降维。

    编译后可不可以为类增加属性?

    可以,甚至可以给在分类中(category)添加

    static char *propertyCharkey = "propertyKey";
    
    - (void)setPropertyWithName:(NSString *)name{
        objc_setAssociatedObject(self, &propertyCharkey, name, OBJC_ASSOCIATION_COPY);
    }
    
    - (NSString *)getProperty{
        return objc_getAssociatedObject(self, &propertyCharkey);
    }
    

    方法调用原理?

    在Objective-C中,消息直到运行时才绑定到方法实现上。编译器会将消息表达式[receiver message]转化为一个消息函数的调用,即objc_msgSend。这个函数将消息接收者和方法名作为其基础参数,如以下所示:

    objc_msgSend(receiver, selector)
    如果消息中还有其它参数,则该方法的形式如下所示:
    objc_msgSend(receiver, selector, arg1, arg2, …)

    这个函数完成了动态绑定的所有事情:

    1. 首先它找到selector对应的方法实现。因为同一个方法可能在不同的类中有不同的实现,所以我们需要依赖于接收者的类来找到的确切的实现。
    2. 它调用方法实现,并将接收者对象及方法的所有参数传给它。
    3. 最后,它将实现返回的值作为它自己的返回值。

    block部分

    何时需要使用weakSelf?

    weakSelf通常为了消除循环引用,如果block没有直接或者间接被self存储,就不会产生循环引用, 这时使用self是没问题的。

    比如常见的动画,虽然block retain了self,但self并没有retain block

     [UIView animateWithDuration:0.5 animations:^{
             self.view.backgroundColor = [UIColor grayColor];
     }];
    

    下面这样会造成循环引用,需要使用weak self

    @property (nonatomic,  strong) void (^testBlock)();
    
     self.testBlock = ^(){
           self.view.backgroundColor = [UIColor blackColor];
     };
    

    下面这种情况也是很常见的:

    @property (nonatomic, strong) Object *object;
    
    self.object = [[Object alloc]init];
     [self.object blockFunction:^{
            self.view.backgroundColor = [UIColor blueColor];
     }];
    

    但是光看这些是不能决定是否使用weak self的,因为根本不知道object这个实例是否在方法blockFunction内存储了block并间接引用了self

    这里就需要使用weak self来打破self-self.object-block这个引用环

    - (void)blockFunction:(ObjectBlock)block{
        self.testBlock = block;
        
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            block();
        });
    }
    
    下面俩个问题了解比较少,得清楚block实现原理,抽空好好研究下
    __block 是如何实现的?

    http://blog.csdn.net/abc649395594/article/details/47086751

    block有哪些类型?

    https://segmentfault.com/a/1190000006479320

    相关文章

      网友评论

      • 32d94172a689:∑(°口°๑)❢❢秒跪
        32d94172a689:@Lucifron (•̀ω•́)✧
        Lucifron:@ChitandaEru 所以知道了怎么用后,最好能理解为什么,一块加油

      本文标题:总结一次iOS面试

      本文链接:https://www.haomeiwen.com/subject/xnshettx.html