昨天去了网易参加了次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, …)
这个函数完成了动态绑定的所有事情:
- 首先它找到selector对应的方法实现。因为同一个方法可能在不同的类中有不同的实现,所以我们需要依赖于接收者的类来找到的确切的实现。
- 它调用方法实现,并将接收者对象及方法的所有参数传给它。
- 最后,它将实现返回的值作为它自己的返回值。
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