zoukankan      html  css  js  c++  java
  • iOS面试题

    __block和__weak修饰符的区别其实是挺明显的: 

    1.__block不管是ARC还是MRC模式下都可以使用,可以修饰对象,还可以修饰基本数据类型。 
    2.__weak只能在ARC模式下使用,也只能修饰对象(NSString),不能修饰基本数据类型(int)。 
    3.__block对象可以在block中被重新赋值,__weak不可以。 

     

     tableView 滑动卡的问题主要是因为:从缓存中或者是从本地读取图片给UIImage的时候耗费的时间。需要把下面的两句话放到子线程里面:

    1. NSData *imgData = [NSData dataWithContentsOfURL:[NSURL URLWithString:app.icon]]; //得到图像数据  
    2.         UIImage *image = [UIImage imageWithData:imgData];  

     

    把UIImage赋值给图片的时候在主线程。

    子线程不能更新UI 所有的UI跟新都是主线程执行了。手指滑动屏幕了。或者屏幕的某个方法执行了。

     

    子线程里面加入NSTimer 的时候需要 手动添加NSRunloop   否则不能循环。

     

    单利里面添加 NSMutableArray 的时候,防止多个地方对它同时便利和修改的话,需要加原子属性。并且用strong,,,并且写一个遍历和修改的方法。加上锁。   Lock   UnLock    

     

        __weak ViewController*  weakSelf = self;

    GCD里面用 __weak 防止内存释放不了,循环引用。

     

    static 关键字的作用: 

    1)函数体内 static 变量的作用范围为该函数体,不同于 auto 变量,该变量的内存只被分配一次, 

    因此其值在下次调用时仍维持上次的值; 

    2)在模块内的 static 全局变量可以被模块内所用函数访问,但不能被模块外其它函数访问; 

    3)在模块内的 static 函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明 

    它的模块内; 

    4)在类中的 static 成员变量属于整个类所拥有,对类的所有对象只有一份拷贝; 

    5)在类中的 static 成员函数属于整个类所拥有,这个函数不接收 this 指针,因而只能访问类的static 成员变量。 

    UIButton 的父类是UIControl  UIControl的父类是UIView UIView的父类是 UIResponder

     

    http状态吗 :302 是请求重定向。500以上是服务器错误。400以上是请求链接错误或者找不到服务器。200以上是正确。100以上是请求接受成功。

     

    copy与retain:
    1、copy其实是建立了一个相同的对象,而retain不是;
    2、copy是内容拷贝,retain是指针拷贝  
    3、copy是内容的拷贝 ,对于像NSString,的确是这样,但是如果copy的是一个NSArray呢?这时只是copy了指向array中相对应元素的指针.这便是所谓的"浅复制".
    4、copy的情况:NSString *newPt = [pt copy];
    此时会在堆上重新开辟一段内存存放@"abc" 比如0X1122 内容为@"abc 同时会在栈上为newPt分配空间 比如地址:0Xaacc 内容为0X1122 因此retainCount增加1供newPt来管理0X1122这段内存;
    assign与retain:
    1、assign: 简单赋值,不更改索引计数;
    2、assign的情况:NSString *newPt = [pt assing]; 
    此时newPt和pt完全相同 地址都是0Xaaaa 内容为0X1111 即newPt只是pt的别名,对任何一个操作就等于对另一个操作, 因此retainCount不需要增加;
    3、assign就是直接赋值;
    4、retain使用了引用计数,retain引起引用计数加1, release引起引用计数减1,当引用计数为0时,dealloc函数被调用,内存被回收;    
    5、retain的情况:NSString *newPt = [pt retain]; 
    此时newPt的地址不再为0Xaaaa,可能为0Xaabb 但是内容依然为0X1111。 因此newPt 和 pt 都可以管理"abc"所在的内存,因此 retainCount需要增加1 ;
    readonly:
    1、属性是只读的,默认的标记是读写,如果你指定了只读,在@implementation中只需要一个读取器。或者如果你使用@synthesize关键字,也是有读取器方法被解析  
    readwrite:
    1、说明属性会被当成读写的,这也是默认属性。设置器和读取器都需要在@implementation中实现。如果使用@synthesize关键字,读取器和设置器都会被解析;
    nonatomic:
    1、非原子性访问,对属性赋值的时候不加锁,多线程并发访问会提高性能。如果不加此属性,则默认是两个访问方法都为原子型事务访问;
    weak and strong property (强引用和弱引用的区别):
    1、 weak 和 strong 属性只有在你打开ARC时才会被要求使用,这时你是不能使用retain release autorelease 操作的,因为ARC会自动为你做好这些操作,但是你需要在对象属性上使用weak 和strong,其中strong就相当于retain属性,而weak相当于assign。
    2、只有一种情况你需要使用weak(默认是strong),就是为了避免retain cycles(就是父类中含有子类{父类retain了子类},子类中又调用了父类{子类又retain了父类},这样都无法release)    
    3、声明为weak的指针,指针指向的地址一旦被释放,这些指针都将被赋值为nil。这样的好处能有效的防止野指针。   
     ARC(Automatic Reference Counting):
    1、就是代码中自动加入了retain/release,原先需要手动添加的用来处理内存管理的引用计数的代码可以自动地由编译器完成了。

    该机能在 iOS 5/ Mac OS X 10.7 开始导入,利用 Xcode4.2 以后可以使用该特性。
    strong,weak,copy 具体用法:
    1. 具体一点:IBOutlet可以为weak,NSString为copy,Delegate一般为weak,其他的看情况。一般来说,类“内部”的属性设 置为strong,类“外部”的属性设置为weak。说到底就是一个归属权的问题。小心出现循环引用导致内存无法释放。
    2.不用ARC的话就会看到很多retian。
    3.如果你写了@synthesize abc = _abc;的话,系统自动帮你声明了一个_abc的实例变量。
       使用assign: 对基础数据类型 (NSInteger)和C数据类型(int, float, double, char,等)
       使用copy: 对NSString

     

    代理的作用?
    答案:代理的目的是改变或传递控制链。允许一个类在某些特定时刻通知到其他类,而不需要获取到那些类的指针。可以减少框架复杂度。
    另外一点,代理可以理解为java中的回调监听机制的一种类似。

     

    4.XML与HTML的设计区别是:XML的核心是数据,其重点是数据的内容。而HTML 被设计用来显示数据,其重点是数据的显示。

    5.XML和HTML语法区别:HTML的标记不是所有的都需要成对出现,XML则要求所有的标记必须成对出现;HTML标记不区分大小写,XML则大小敏感,即区分大小写。

     

    static有什么用途?(请至少说明两种)
                1.限制变量的作用域
                2.设置变量的存储域
                7. 引用与指针有什么区别?
                1) 引用必须被初始化,指针不必。
                2) 引用初始化以后不能被改变,指针可以改变所指的对象。
                2) 不存在指向空值的引用,但是存在指向空值的指针。
                8. 描述实时系统的基本特性
                在特定时间内完成特定的任务,实时性与可靠性

    64全局变量和局部变量在内存中是否有区别?如果有,是什么区别?
                全局变量储存在静态数据库,局部变量在堆栈

     

    堆和栈上的指针 

    指针所指向的这块内存是在哪里分配的,在堆上称为堆上的指针,在栈上为栈上的指针

    在堆上的指针,可以保存在全局数据结构中,供不同函数使用访问同一块内存

    在栈上的指针,在函数退出后,该内存即不可访问

    59什么是指针的释放

    具体来说包括两个概念

    1 释放该指针指向的内存,只有堆上的内存才需要我们手工释放,栈上不需要

    2 将该指针重定向为NULL. 

    60数据结构中的指针

    其实就是指向一块内存的地址,通过指针传递,可实现复杂的内存访问

    7 函数指针? 

    指向一块函数的入口地址

     

    8 指针作为函数的参数

    比如指向一个复杂数据结构的指针作为函数变量 

    这种方法避免整个复杂数据类型内存的压栈出栈操作,提高效率

    注意:指针本身不可变,但指针指向的数据结构可以改变

     

    9 指向指针的指针

    指针指向的变量是一个指针,即具体内容为一个指针的值,是一个地址

    此时指针指向的变量长度也是4位

    61指针与地址的区别

    区别: 

    1指针意味着已经有一个指针变量存在,他的值是一个地址,指针变量本身也存放在一个长度为四个字节的地址当中,而地址概念本身并不代表有任何变量存在

    2 指针的值,如果没有限制,通常是可以变化的,也可以指向另外一个地址

       地址表示内存空间的一个位置点,他是用来赋给指针的,地址本身是没有大小概念,指针指向变量的大小,取决于地址后面存放的变量类型

    62指针与数组名的关系

      其值都是一个地址,但前者是可以移动的,后者是不可变的

     

    12 怎样防止指针的越界使用问题

      必须让指针指向一个有效的内存地址

    1 防止数组越界 

    2 防止向一块内存中拷贝过多的内容 

    3 防止使用空指针 

    4 防止改变const修改的指针 

    5 防止改变指向静态存储区的内容 

    6 防止两次释放一个指针 

    7 防止使用野指针.

     

    37What is lazy loading?
    答案:懒汉模式,(懒加载)只在用到的时候才去初始化。
    也可以理解成延时加载。
    我觉得最好也最简单的一个列子就是tableView中图片的加载显示了。
    一个延时载,避免内存过高,一个异步加载,避免线程堵塞。

     

    What are KVO and KVC?
    答案:kvc:键 - 值编码是一种间接访问对象的属性使用字符串来标识属性,而不是通过调用存取方法,直接或通过实例变量访问的机制。
    很多情况下可以简化程序代码。apple文档其实给了一个很好的例子。
    kvo:键值观察机制,他提供了观察某一属性变化的方法,极大的简化了代码。
    具体用看到嗯哼用到过的一个地方是对于按钮点击变化状态的的监控。
    比如我自定义的一个button
    [cpp] 
    [self addObserver:self forKeyPath:@"highlighted" options:0 context:nil]; 
    #pragma mark KVO 
    - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context 

        if ([keyPath isEqualToString:@"highlighted"] ) { 
            [self setNeedsDisplay]; 
        } 

    对于系统是根据keypath去取的到相应的值发生改变,理论上来说是和kvc机制的道理是一样的。
    对于kvc机制如何通过key寻找到value:
    “当通过KVC调用对象时,比如:[self valueForKey:@”someKey”]时,程序会自动试图通过几种不同的方式解析这个调用。首先查找对象是否带有 someKey 这个方法,如果没找到,会继续查找对象是否带有someKey这个实例变量(iVar),如果还没有找到,程序会继续试图调用 -(id) valueForUndefinedKey:这个方法。如果这个方法还是没有被实现的话,程序会抛出一个NSUndefinedKeyException异常错误。
    (cocoachina.com注:Key-Value Coding查找方法的时候,不仅仅会查找someKey这个方法,还会查找getsomeKey这个方法,前面加一个get,或者_someKey以及_getsomeKey这几种形式。同时,查找实例变量的时候也会不仅仅查找someKey这个变量,也会查找_someKey这个变量是否存在。)
    设计valueForUndefinedKey:方法的主要目的是当你使用-(id)valueForKey方法从对象中请求值时,对象能够在错误发生前,有最后的机会响应这个请求。这样做有很多好处,下面的两个例子说明了这样做的好处。
    来至cocoa,这个说法应该挺有道理。
    因为我们知道button却是存在一个highlighted实例变量.因此为何上面我们只是add一个相关的keypath就行了,

     

     

    Difference between shallow copy and deep copy?
    浅复制和深复制的区别?
    答案:浅层复制:只复制指向对象的指针,而不复制引用对象本身。
    深层复制:复制引用对象本身。
    意思就是说我有个A对象,复制一份后得到A_copy对象后,对于浅复制来说,A和A_copy指向的是同一个内存资源,复制的只不过是是一个指针,对象本身资源
    还是只有一份,那如果我们对A_copy执行了修改操作,那么发现A引用的对象同样被修改,这其实违背了我们复制拷贝的一个思想。深复制就好理解了,内存中存在了
    两份独立对象本身。
    用网上一哥们通俗的话将就是:
    浅复制好比你和你的影子,你完蛋,你的影子也完蛋
    深复制好比你和你的克隆人,你完蛋,你的克隆人还活着。

     

     

  • 相关阅读:
    ubuntu jdk安装
    paxos 算法略解
    java 异常问题 (望解答)
    jquery加载问题
    The Pragmatic Programmer Quick Reference Guide
    车辆调度管理系统开发(三)
    车辆调度管理系统开发(二)
    车辆调度管理系统开发(一)
    vue2过渡&动画
    VScode 之快速创建vue模板
  • 原文地址:https://www.cnblogs.com/leikun1113/p/5467451.html
Copyright © 2011-2022 走看看