总结一次iOS面试

昨天去了网易参加了次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

你可能感兴趣的:(总结一次iOS面试)