IOS面试题object-c 11-20

这篇具有很好参考价值的文章主要介绍了IOS面试题object-c 11-20。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

11、解释self = [super init]方法?

容错处理, 当父类初始化失败,会返回一个nil, 表示初始化失败。

由于继承的关系, 子类是需要拥有父类的实例和行为, 因此, 我们必须先初始化父类,然后再初始化子类

12、简述使用block有什么优点?
代码紧凑,传值、回调都很方便,省去了写代理的很多代码。

NSTimer封装成的block,具体实现实现方法:

NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0
        repeats:YES
        callback:^() {
                weakSelf.secondsLabel.text = ...
        }
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

13、 Object-C中是否有二维数组,如何实现二维数组?

在实际的项目开发中,二维数组也是常常用到的数据结构

OC中的二维数组也是通过一维数组来建立的,今天我们来详解一下如何在OC中使用二维数组。
使用NSArray初始化的一维数组和二维数组都是不可变数组

#import
int main(int argc, const char * argv[]) {
        @autoreleasepool {

        //定义2个一维数组;
        NSArray *firstRow = [[NSArray alloc] initWithObjects:@"1",@"2",@"3", nil];
        NSArray *secondRow = [[NSArray alloc] initWithObjects:@"4",@"5",@"6", nil];

        //使用一维数组来初始化二维数组;
        NSArray *my2DArray = [[NSArray alloc] initWithObjects:firstRow,secondRow, nil];
        //输出二维数组对象;
        NSLog(@"二维数组:%@",my2DArray);

        //遍历二维数组;
        for (int i = 0; i < [my2DArray count]; i++) {
                        for (int j = 0; j < firstRow.count; j++) {
                                NSLog(@"二维数组元素:%@",[[my2DArray objectAtIndex:i] objectAtIndex :j]);
                        }
                }
        }
        return 0;
}
14、简述property属性的修饰符的作用?
getter=getName、setter=setName:设置setter与getter的方法名;
readwritereadonly设置可供访问级别;
assign:方法直接赋值,不进行任何retain操作,为了解决原类型与环循引用问题;
retain:其setter方法对参数进行release旧值再retain新值,所有实现都是这个顺序;

copy:其setter方法进行copy操作,与retain处理流程一样,先对旧值release,再copy出新的对象,retainCount为1。

这是为了减少对上下文的依赖而引入的机制。

nonatomic:非原子性访问,不加同步, 多线程并发访问会提高性能。

注意,如果不加此属性,则默认是两个访问方法都为原子型事务访问

15、 将字符串“2015-04-10”格式化日期转为NSDate类型

NSString *timeStr = @"2015-04-10";


NSDateFormatter *formatter = [[NSDateFormatter alloc] init];


formatter.dateFormat = @"yyyy-MM-dd";


formatter.timeZone = [NSTimeZone defaultTimeZone];


NSDate *date = [formatter dateFromString:timeStr];


// 2015-04-09 16:00:00 +0000
NSLog(@"%@", date);

16、简述SDWebImage内部实现过程?
入口 setImageWithURL:placeholderImage:options: 会先把 placeholderImage 显示,然后 SDWebImageManager 根据 URL 开始处理图片。
进入 SDWebImageManager-downloadWithURL: delegate:options:userInfo:,交给 SDImageCache 从缓存查找图片是否已经下载 queryDiskCacheForKey:delegate:userInfo:.
先从内存图片缓存查找是否有图片,如果内存中已经有图片缓存,SDImageCacheDelegate 回调 imageCache:didFindImage:forKey:userInfo: 到 SDWebImageManager。
SDWebImageManagerDelegate 回调 webImageManager:didFinishWithImage: 到 UIImageView+WebCache 等前端展示图片。
如果内存缓存中没有,生成 NSInvocationOperation 添加到队列开始从硬盘查找图片是否已经缓存

根据 URLKey 在硬盘缓存目录下尝试读取图片文件。

这一步是在 NSOperation 进行的操作,所以回主线程进行结果回调 notifyDelegate:。

如果从硬盘缓存目录读取不到图片,说明所有缓存都不存在该图片,需要下载图片,回调 imageCache:didNotFindImageForKey:userInfo: 。
共享或重新生成一个下载器 SDWebImageDownloader 开始下载图片。

图片下载由 NSURLConnection 来做,实现相关 delegate 来判断图片下载中、下载完成和下载失败。
connection:didReceiveData: 中利用 ImageIO 做了按图片下载进度加载效果。
connectionDidFinishLoading: 数据下载完成后交给 SDWebImageDecoder 做图片解码处理
图片解码处理在一个 NSOperationQueue 完成,不会拖慢主线程 UI。如果有需要对下载的图片进行二次处理,最好也在这里完成,效率会好很多。
在主线程 notifyDelegateOnMainThreadWithInfo: 宣告解码完成,imageDecoder:didFinishDecodingImage:userInfo: 回调给 SDWebImageDownloader。
imageDownloader:didFinishWithImage: 回调给 SDWebImageManager 告知图片下载完成。
通知所有的 downloadDelegates 下载完成,回调给需要的地方展示图片。

将图片保存到 SDImageCache 中,内存缓存和硬盘缓存同时保存

写文件到硬盘也在以单独 NSInvocationOperation 完成,避免拖慢主线程。

SDImageCache 在初始化的时候会注册一些消息通知,在内存警告或退到后台的时候清理内存图片缓存,应用结束的时候清理过期图片。

SDWI 也提供了 UIButton+WebCache 和 MKAnnotationView+WebCache,方便使用。

SDWebImagePrefetcher 可以预先下载图片,方便后续使用

17、 为什么 Objective-C 的方法调用要用方括号?

首先要说的是,Objective-C 的历史相当久远,如果你查 wiki 的话,你会发现:Objective-C 和 C++ 这两种语言的发行年份都是 1983 年。在设计之初,二者都是作为 C 语言的面向对象的接班人,希望成为事实上的标准。

最后结果大家都知道了,C++ 最终胜利了,而 Objective-C 在之后的几十年中,基本上变成了苹果自己家玩的玩具。不过最终,由于 iPhone 的出现,Objective-C 迎来了第二春,在 TOBIE 语言排行榜上,从 20 名开外一路上升,排名曾经超越过 C++,达到了第三名,但是随着 swift 的出现,Objective-C 的排名则一路下滑

Objective-C 在设计之初参考了不少 Smalltalk 的设计,而消息发送则是向 Smalltalk 学来的。Objective-C 当时采用了方括号的形式来表示发送消息,为什么没有选择用点呢?

我个人觉得是,当时市面上并没有别的面向对象语言的设计参考,而 Objective-C 「发明」了方括号的形式来给对象发消息,而 C++ 则「发明」了用点的方式来 “发消息”。

有人可能会争论说 C++ 的「点」并不是真正的发消息,但是其实二者都是表示「调用对象所属的成员函数」。

另外,有读者评论说使用方括号的形式是为了向下兼容 C 语言,我并不觉得中括号是唯一选择,C++ 不也兼容了 C 语言么?Swift 不也可以调用 C 函数么?
最终,其实是 C++ 的「发明」显得更舒服一些,所以后来的各种语言都借鉴了 C++ 的这种设计,也包括 Objective-C 在内。Objective-C 2.0 版本中,引入了 dot syntax,即:
a = obj.foo 等价于 a = [obj foo]
obj.foo = 1 则等价于 [obj setFoo:1]
Objective-C 其实在设计之中确实是比较特立独行的,除了方括号的函数调用方式外,还包括比较长的,可读性很强的函数命名风格。
我个人并不讨厌 Objective-C 的这种设计,但是从 Swift 语言的设计来看,苹果也开始放弃一些 Objective-C 的特点了,比如就去掉了方括号这种函数调用方式。
所以,回到我们的问题,我个人认为,答案就是:Objective-C 在 1983 年设计的时候,并没有什么有效的效仿对象,于是就发明了一种有特点的函数调用方式,现在看起来,这种方式比点操作符还是略逊一筹。

大多数语言一旦被设计好,就很难被再次修改,应该说 Objective-C 发明在 30 年前,还是非常优秀的,它的面向对象化设计得非常纯粹,比 C++ 要全面得多,也比 C++ 要简单得多。
 18、如何创建一个可以被取消执行的 block?
// 方法一:创建一个类,将要执行的 block 封装起来,然后类的内部有一个 _isCanceled 变量,在执行的时候,检查这个变量,如果 _isCanceled 被设置成 YES 了,则退出执行。
        typedef void (^Block)();
        @interface CancelableObject : NSObject
                - (id)initWithBlock:(Block)block;
                - (void)start;
                - (void)cancel;
        @end
@implementation CancelableObject {
        BOOL _isCanceled;
        Block _block;
}
- (id) initWithBlock: (Block)block {
        self = [super init];
        if (self != nil) {
                _isCanceled = NO;
                _block = block;
        }
        return self;
}
- (void) start {
        __weak typeof(self) weakSelf = self;
        dispatch_async(dispatch_get_global_queue(0, 0),
        ^{

                if (weakSelf) {
                        typeof(self) strongSelf = weakSelf;
                        if (!strongSelf->_isCanceled) {
                                (strongSelf->_block)();
                        }
                }
        });
}

- (void) cancel {
        _isCanceled = YES;
}

@end

// 另外一种写法,将要执行的 block 直接放到执行队列中,但是让其在执行前检查另一个 isCanceled 的变量,然后把这个变量的修改实现在另一个 block 方法中,如下所示:


typedef void (^CancelableBlock)();
typedef void (^Block)();

+ (CancelableBlock) dispatch_async_with_cancelable:(Block)block {
        __block BOOL isCanceled = NO;
        CancelableBlock cb = ^() {
                isCanceled = YES;
        };

        dispatch_async(dispatch_get_global_queue(0, 0), ^{
                if (!isCanceled) {
                        block();
                }
        });

        return cb;
}

以上两种方法都只能在 block 执行前有效,如果要在 block 执行中有效,只能让 block 在执行中,有一个机制来定期检查外部的变量是否有变化,而要做到这一点,需要改 block 执行中的代码。

在本例中,如果 block 执行中的代码是通过参数传递进来的话,似乎并没有什么办法可以修改它了

19、简述IOS __block和__weak修饰符的区别?
1、__weak:
__weak只能在ARC下使用,且只能修饰对象,不能修饰基本数据类型(int等),弱引用,可以block解决循环引用问题。
2、__block:
__block既能再ARC下使用,也能在 MRC下使用。既能修饰对象又能修饰基本数据类型。
ARC、MRC区别:__block对象在ARC下可能会导致循环引用,非ARC下会避免循环引用。
ARC、MRC共同点:block内对于栈上的数据不能修改,使用__block修饰以后放到了堆上就可以修改了。
20、解释单例的写法在单例中使用数组要注意什么?
static PGSingleton *sharedSingleton;

+ (instancetype)sharedSingleton
{
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
                sharedSingleton = [[PGSingleton alloc] init];
        });
        return sharedSingleton;
}
其实上面的还不是标准的单例方法,标准的单例方法需要重写 copyWithZone,allocWithZone,init,确保以任何方式创建出来的对象只有一个,这里就不详细写了。

单例使用 NSMutableArray 的时候,防止多个地方对它同时遍历和修改的话,需要加原子属性

并且property用strong,并且写一个遍历和修改的方法。

加上锁. Lock,UnLock.(PS:考虑性能问题尽量避免使用锁,在苹果的文档张看到的不要问我为什么,我也忘了自己查去。。)文章来源地址https://www.toymoban.com/news/detail-839154.html

到了这里,关于IOS面试题object-c 11-20的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • objective-c 基础学习

    目录 第一节:OC 介绍 ​​第二节:Fundation 框架 ​第三节:NSLog 相对于print 的增强 ​第四节:NSString ​第五节:oc新增数据类型 第六节: 类和对象 ​类的方法的声明与实现 ​第七节:类和对象的存储 第八节:nil 与 NULL 第九节:分组导航标记#pragma mark ​第十节:方法与函

    2024年02月07日
    浏览(42)
  • 第一章 熟悉Objective-C

    Objective—C语言是由Smalltalk演化而来,后者是消息型语言的鼻祖,所以该语言使用的“消息结构”而非“函数调用”。 1. 消息和函数调用之间的区别 关键区别在于: 使用消息结构的语言,其运行所应执行的代码由运行环境来决定;而使用函数调用的语言,则由编译器决定。

    2024年01月18日
    浏览(40)
  • 【KVC补充 Objective-C语言】

    2024年02月15日
    浏览(42)
  • Effective Objective-C 学习(四)

    在执行后台任务时,GCD 并不一定是最佳方式。还有一种技术叫做 NSOperationQueue,它虽然与 GCD 不同,但是却与之相关,开发者可以把操作以 NSOperation 子类的形式放在队列中,而这些操作也能够并发执行。 GCD是纯C的API,而NSOperationQueue是Objective-C的对象。这意味着使用GCD时,任

    2024年02月19日
    浏览(40)
  • Effective Objective-C 学习(二)

    “属性”(property)是 Objective-C 的一项特性,用于封装对象中的数据。Objective-C 对象通常会把其所需的数据保存为各种实例变量。实例变量一般通过“存取方法”来访问。其中,“获取方法”(getter)用于读取变量值,而“设置方法”(setter)用于写入变量值。开发者可以令

    2024年02月20日
    浏览(48)
  • Objective-C日期NSDate使用

    2024年01月21日
    浏览(50)
  • Effective Objective-C 学习(三)

    Objective-C 使用引用计数来管理内存:每个对象都有个可以递增或递减的计数器。如果想使某个对象继续存活,那就递增其引用计数:用完了之后,就递减其计数。计数变为 0时,就可以把它销毁。 在ARC中,所有与引用计数有关的方法都无法编译(由于 ARC 会在编译时自动插入

    2024年02月22日
    浏览(49)
  • use gnustep objective-c

    专注于概念,而不是迷失在语言技术细节中 编程语言的目的是成为一个更好的程序员; 也就是说,在设计和实现新系统以及维护旧系统方面变得更加有效 header preprocess interface implementation method variable declare and expression comment basic integer set and float set enum type void type derive type incl

    2024年02月14日
    浏览(42)
  • Objective-C获取变量类型的方法

    在Objective-C中,要获取一个对象的类型,可以使用[object class]方法。这将返回一个Class对象,表示该对象的类型。 另外,typeid是C++中的,用于获取一个变量的类型信息。在Objective-C中,typeid并不适用于获取对象类型。相反,您应该使用[object class]方法来获取对象的类型。

    2024年02月13日
    浏览(43)
  • Objective-C之Class底层结构探索

    在讲 OC-Class 底层类结构之前,先看下下面这张图: 通过isa走位图 得出的结论是: 1,类,父类,元类都包含了 isa , superclass 2,对象isa指向类对象,类对象的isa指向了元类,元类的 isa 指向了根元类,根元类 isa 指向自己 3,类的 superclass 指向父类,父类的 superclass 指向的根类

    2024年03月24日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包