李洪强iOS经典面试题

p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 17.0px "Helvetica Neue"; color: #333333; min-height: 20.0px }
p.p2 { margin: 0.0px 0.0px 0.0px 0.0px; font: 17.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
p.p3 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px "Helvetica Neue"; color: #ffffff; background-color: #ea6f5a }
p.p4 { margin: 0.0px 0.0px 0.0px 0.0px; font: 17.0px "Helvetica Neue"; color: #333333; background-color: #ffffff; min-height: 20.0px }
p.p5 { margin: 0.0px 0.0px 0.0px 0.0px; font: 24.0px "Helvetica Neue"; color: #969696; background-color: #ffffff; min-height: 29.0px }
p.p7 { margin: 0.0px 0.0px 0.0px 0.0px; font: 34.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
p.p8 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "Helvetica Neue"; color: #969696; background-color: #ffffff }
p.p9 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 16.0px "Helvetica Neue"; color: #2f2f2f; background-color: #ffffff; min-height: 18.0px }
p.p10 { margin: 0.0px 0.0px 0.0px 0.0px; font: 16.0px "Helvetica Neue"; color: #2f2f2f; background-color: #ffffff; min-height: 18.0px }
p.p11 { margin: 0.0px 0.0px 0.0px 0.0px; font: 16.0px "Helvetica Neue"; color: #2f2f2f; background-color: #ffffff }
p.p12 { margin: 0.0px 0.0px 0.0px 0.0px; font: 22.0px "Helvetica Neue"; color: #2f2f2f; background-color: #ffffff }
p.p13 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px Menlo; color: #657b83; background-color: #f6f6f6 }
p.p14 { margin: 0.0px 0.0px 0.0px 0.0px; font: 16.0px "Helvetica Neue"; color: #2f2f2f; background-color: #f7f7f7 }
p.p15 { margin: 0.0px 0.0px 0.0px 0.0px; font: 20.0px "Helvetica Neue"; color: #2f2f2f; background-color: #ffffff }
p.p16 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 14.0px "Helvetica Neue"; color: #969696; background-color: #ffffff }
p.p17 { margin: 0.0px 0.0px 0.0px 0.0px; font: 16.0px "Helvetica Neue"; color: #3194d0; background-color: #ffffff }
p.p18 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "Helvetica Neue"; color: #9b9b9b; background-color: #ffffff }
p.p19 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "Helvetica Neue"; color: #333333; background-color: #ffffff; min-height: 14.0px }
p.p20 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 16.0px "Helvetica Neue"; color: #8c8c8c; background-color: rgba(181, 181, 181, 0.098) }
p.p21 { margin: 0.0px 0.0px 0.0px 0.0px; font: 17.0px "Helvetica Neue"; color: #333333; background-color: rgba(181, 181, 181, 0.098) }
p.p22 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "Helvetica Neue"; color: #969696; background-color: rgba(181, 181, 181, 0.098) }
p.p23 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 17.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
p.p24 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 16.0px "Helvetica Neue"; color: #ffffff; background-color: #ea6f5a }
p.p26 { margin: 0.0px 0.0px 0.0px 0.0px; font: 18.0px "Helvetica Neue"; color: #ea6f5a; background-color: #ffffff }
p.p27 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #9b9b9b; background-color: #ffffff }
p.p28 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
p.p29 { margin: 0.0px 0.0px 0.0px 0.0px; font: 16.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
p.p30 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #969696; background-color: #ffffff }
p.p31 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
p.p32 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #3194d0; background-color: #ffffff }
p.p34 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px "Helvetica Neue"; color: #333333; background-color: #f5f5f5 }
p.p35 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #333333; background-color: #f5f5f5 }
p.p36 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #a0a0a0; background-color: #f5f5f5 }
li.li2 { margin: 0.0px 0.0px 0.0px 0.0px; font: 17.0px "Helvetica Neue"; color: #333333; background-color: #ffffff }
li.li6 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 17.0px Helvetica; color: #969696; background-color: #ffffff }
li.li11 { margin: 0.0px 0.0px 0.0px 0.0px; font: 16.0px "Helvetica Neue"; color: #2f2f2f; background-color: #ffffff }
li.li25 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 17.0px "Helvetica Neue"; color: #333333; background-color: #ffffff; min-height: 20.0px }
li.li30 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px "Helvetica Neue"; color: #969696; background-color: #ffffff }
li.li33 { margin: 0.0px 0.0px 0.0px 0.0px; text-align: center; font: 20.0px Helvetica; color: #333333; background-color: #ffffff }
span.s1 { }
span.s2 { font: 15.0px "Helvetica Neue"; color: #ffffff }
span.s3 { }
span.s4 { font: 17.0px "Helvetica Neue" }
span.s5 { font: 12.0px "Helvetica Neue"; color: #ea6f5a }
span.s6 { font: 16.0px "Helvetica Neue" }
span.s7 { font: 12.0px "Helvetica Neue"; color: #8c8c8c }
span.s8 { font: 16.0px "Helvetica Neue"; color: #3194d0 }
span.s9 { color: #2f2f2f }
span.s10 { color: #3194d0 }
span.s11 { font: 12.0px "Helvetica Neue"; color: #9b9b9b }
span.s12 { font: 19.0px "Helvetica Neue" }
span.s13 { font: 12.0px "Helvetica Neue"; color: #333333 }
span.s14 { font: 14.0px "Helvetica Neue"; color: #9b9b9b }
span.s15 { font: 12.0px "Helvetica Neue"; color: #969696 }
span.s16 { font: 17.0px "Helvetica Neue" }
span.s17 { font: 15.0px "Helvetica Neue" }
span.s18 { font: 17.0px "Helvetica Neue"; color: #333333 }
span.s19 { font: 14.0px "Helvetica Neue"; color: #3194d0 }
span.s20 { color: #333333 }
span.s21 { }
span.s22 { font: 14.0px "Helvetica Neue" }
ol.ol1 { list-style-type: decimal }
ul.ul1 { list-style-type: none }
ul.ul2 { list-style-type: disc }

1. struct和class的区别

swift中,class是引用类型,struct是值类型。值类型在传递和赋值时将进行复制,而引用类型则只会使用引用对象的一个"指向"。所以他们两者之间的区别就是两个类型的区别。

class有这几个功能struct没有的:

  • class可以继承,这样子类可以使用父类的特性和方法
  • 类型转换可以在runtime的时候检查和解释一个实例的类型
  • 可以用deinit来释放资源
  • 一个类可以被多次引用

struct也有这样几个优势:

  • 结构较小,适用于复制操作,相比于一个class的实例被多次引用更加安全。
  • 无须担心内存memory leak或者多线程冲突问题

顺便提一下,array在swift中是用struct实现的。Apple重写过一次array,然后复制就是深度拷贝了。猜测复制是类似参照那样,通过栈上指向堆上位置的指针来实现的。而对于它的复制操作,也是在相对空间较为宽裕的堆上来完成的,所以性能上还是不错的。

下面引用猫神OneV的博客:

var arr = [0,0,0]

var newArr = arr

arr[0] = 1

//Check arr and newArr

arr //[1, 0, 0]

newArr // before beta3:[1, 0, 0], after beta3:[0, 0, 0]

所以可以猜测其实在背后 Array和 Dictionary的行为并不是像其他 struct 那样简单的在栈上分配,而是类似参照那样,通过栈上指向堆上位置的指针来实现的。而对于它的复制操作,也是在相对空间较为宽裕的堆上来完成的。当然,现在还无法(或者说很难)拿到最后的汇编码,所以这只是一个猜测而已。

补充:

C语言中,struct与的class的区别:

struct只是作为一种复杂数据类型定义,不能用于面向对象编程。

C++中,struct和class的区别:

对于成员访问权限以及继承方式,class中默认的是private的,而struct中则是public的。class还可以用于表示模板类型,struct则不行。

2. 介绍一下观察者模式

观察者模式(Observer Pattern):定义对象间的一种一对多依赖关系,使得每当一个对象状态发生改变时,其相关依赖对象皆得到通知并被自动更新。

在IOS中典型的推模型实现方式为NSNotificationCenter和KVO。

NSNotificationCenter

NSNotificationCenter

  1. 观察者Observer,通过NSNotificationCenter的addObserver:selector:name:object接口来注册对某一类型通知感兴趣。在注册时候一定要注意,NSNotificationCenter不会对观察者进行引用计数+1的操作,我们在程序中释放观察者的时候,一定要去报从center中将其注销了。
  2. 通知中心NSNotificationCenter,通知的枢纽。
  3. 被观察的对象,通过postNotificationName:object:userInfo:发送某一类型通知,广播改变。
  4. 通知对象NSNotification,当有通知来的时候,Center会调用观察者注册的接口来广播通知,同时传递存储着更改内容的NSNotification对象。

KVO

KVO的全称是Key-Value Observer,即键值观察。是一种没有中心枢纽的观察者模式的实现方式。一个主题对象管理所有依赖于它的观察者对象,并且在自身状态发生改变的时候主动通知观察者对象。

  1. 注册观察者
  2. [object addObserver:self forKeyPath:property options:NSKeyValueObservingOptionNew context:]。
  3. 更改主题对象属性的值,即触发发送更改的通知。
  4. 在制定的回调函数中,处理收到的更改通知。
  5. 注销观察者 [object removeObserver:self forKeyPath:property]。

3.在一个HTTPS连接的网站里,输入账号密码点击登录后,到服务器返回这个请求前,中间经历了什么

这个非常得深非常得广,我来大概说一下。

HTTPS加密流程

  1. 客户端会打包一个请求,包括url,端口啊,你的账号密码等等。账号密码登陆应该用的是Post方式,所以相关的用户信息会被加载到body里面。这个请求应该包含三个方面:网络地址,协议,资源路径。注意,这里是HTTPS,就是HTTP + SSL / TLS,在HTTP上又加了一层处理加密信息的模块(相当于是个)。
  2. 一般会先请求DNS服务器。DNS服务器负责将你的网络地址解析成IP地址,这个IP地址对应网上一台机器。这其中可能发生Hosts Hijack和ISP failure的问题。
  3. 协议是获取资源的方式HTTP,FTP,UDP,不同协议有不同的格式,有些是process-to-process的,有些是host-to-host的。
  4. 客户端会和服务器的端口之间建立一个socket连接,socket一般都是以file descriptor的方式解析请求。
  5. 服务器端接收到请求。服务器端会有一套数字证书(相当于是个钥匙),这个证书会先返回给客户端。客户端会解析证书,相当于用钥匙(证书)把锁(内容)锁上(生成私匙),接着再传送加密信息。
  6. 服务器端接收到加密信息(私匙)之后,会进行解密,并把要返回的数据进行对称加密返回到客户端。假如路径不对,会出现404的错误。
  7. 一般访问服务器之前可能会访问一下proxy。这玩意是个代理,有时候当防火墙用,有时候当cache使。如果后台是reverse-proxy结构,那么实际上有多个web服务器藏在proxy之后按需处理请求,而你访问的永远是proxy,这样可以解决过载问题。
  8. 有时候访问完web服务器后还要访问一下file服务器,主要是请求数据库里的一些信息。
  9. 服务器将相应打包,直接或通过proxy(大多数时候)返回给客户端。客户端会用刚刚生成的私匙进行解密,将内容显示在浏览器上。

HTTPS加密过程详解请去https原理:证书传递、验证和数据加密、解密过程解析

4.在一个app中间有一个button,在你手触摸屏幕点击后,到这个button收到点击事件,中间发生了什么

响应链大概有以下几个步骤

  1. 设备将touch到的UITouch和UIEvent对象打包, 放到当前活动的Application的事件队列中
  2. 单例的UIApplication会从事件队列中取出触摸事件并传递给单例UIWindow
  3. UIWindow使用hitTest:withEvent:方法查找touch操作的所在的视图view

RunLoop这边我大概讲一下

  1. 主线程的RunLoop被唤醒
  2. 通知Observer,处理Timer和Source 0
  3. Springboard接受touch event之后转给App进程
  4. RunLoop处理Source 1,Source1 就会触发回调,并调用_UIApplicationHandleEventQueue() 进行应用内部的分发。
  5. RunLoop处理完毕进入睡眠,此前会释放旧的autorelease pool并新建一个autorelease pool

深挖请去深入理解RunLoop

UIResponder是UIView的父类,UIView是UIControl的父类。

声明一下,第3题依然有很大缺陷,不过因为深挖的地方太多,本文不可能完全兼顾,只能抛砖引玉。另外文章的目的是以面试题为引进行学习,所以写得有点多,可能与面试技巧和时间有冲突。

时间: 2024-10-12 09:13:57

李洪强iOS经典面试题的相关文章

李洪强iOS经典面试题总结

iOS经典面试题总结--内存管理 我根据自己的情况做了一下总结,答案是我总结的,如有答的不好的地方,希望批评指正以及交流,谢谢! 内存管理 1.什么是ARC? ARC是automatic reference counting自动引用计数,在程序编译时自动加入retain/release.在对象被创建时retain count+1,在对象被release时count-1,当count=0时,销毁对象.程序中加入autoreleasepool对象会由系统自动加上autorelease方法,如果该对象

李洪强iOS经典面试题12

1.说说响应链 答: 事件响应链.包括点击事件,画面刷新事件等.在视图栈内从上至下,或者从下之上传播. 可以说点事件的分发,传递以及处理.具体可以去看下touch事件这块.因为问的太抽象化了 严重怀疑题目出到越后面就越笼统. 可以从责任链模式,来讲通过事件响应链处理,其拥有的扩展性 2.什么时候使用NSMutableArray,什么时候使用NSArray? 答:当数组在程序运行时,需要不断变化的,使用NSMutableArray,当数组在初始化后,便不再改变的,使用NSArray.需要指出的是,

李洪强IOS开发之-iOS经典面试题

李洪强IOS开发之-iOS经典面试题 写这篇文章的目的是因为前两天同学想应聘iOS开发,从网上找了iOS面试题和答案让我帮忙看看.我扫了一眼,倒吸了一口冷气,仔细一看,气的发抖.整篇题目30多个没有一个答案是对的,总结这篇面试题的作者对iOS机制根本就是一知半解就敢发到网上,不管有心还是无心都是坑害新人.所以在这里总结一下这几年面试别人和被别人面试遇到的一些我认为比较好的基础题目分享给大家,进阶题目在后续补充.我的理解如果有错漏请一定指出,非常感谢! 从12年开始做面试官到现在已经三个年头了,这

李洪强iOS开发之iOS学习方法收集

李洪强iOS开发之iOS学习方法收集 在这里收集一些iOS学习方法,会不断更新 项目 简述 日期 一年多iOS开发总结 作者总结了自己一年多的iOS学习经验,对于iOS初学者来说很多地方是可以借鉴的 2015.01.07 iOS面试基础知识36题 作者收集了iOS基础比较常见的面试题 2015.04.01

李洪强iOS开发之RunLoop的原理和核心机制

李洪强iOS开发之RunLoop的原理和核心机制 搞iOS之后一直没有深入研究过RunLoop,非常的惭愧.刚好前一阵子负责性能优化项目,需要利用RunLoop做性能优化和性能检测,趁着这个机会深入研究了RunLoop的原理和特性. RunLoop的定义 当有持续的异步任务需求时,我们会创建一个独立的生命周期可控的线程.RunLoop就是控制线程生命周期并接收事件进行处理的机制. RunLoop是iOS事件响应与任务处理最核心的机制,它贯穿iOS整个系统. Foundation: NSRunLo

李洪强iOS开发之性能优化技巧

李洪强iOS开发之性能优化技巧 通过静态 Analyze 工具,以及运行时 Profile 工具分析性能瓶颈,并进行性能优化.结合本人在开发中遇到的问题,可以从以下几个方面进行性能优化. 一.view优化 1.不透明的View 设置为opaque. 2.根据实际情况重用.延迟加载或预加载View. 3.减少subviews数量,定制复杂cell使用drawRect.尽量使用drawRect而不是layoutSubView. 4.不直接调用drawRect. layoutSubviews方法.万不

李洪强iOS开发之动态获取UILabel的bounds

李洪强iOS开发之动态获取UILabel的bounds 在使用UILabel存放字符串时,经常需要获取label的长宽数据,本文列出了部分常用的计算方法. 1.获取宽度,获取字符串不折行单行显示时所需要的长度  CGSize labelBounds = [str sizeWithFont:font constrainedToSize:CGSizeMake(MAXFLOAT, 30)]; 注:如果想得到宽度的话,size的width应该设为MAXFLOAT. 2.获取高度,获取字符串在指定的siz

李洪强iOS开发-网络新闻获取数据思路回顾

李洪强iOS开发-网络新闻获取数据思路回顾 01 创建一个继承自AFHTTPSessionManager的工具类:LHQNetworkTool 用来发送网络请求获取数据  1.1 定义类方法返回单例对象 1.2 单例的实现,设置baseURL,设置超时时长等 02 创建一个分类, 传入一个类的参数返回一个数组,返回当前类的所有属性 用在模型中处理模型数据 03 创建图片轮播模型(model)类发送请求获取轮播图片的数据  3.1 定义属性: 标题,图片 定义两个类方法: (1)传入一个字典的参数

李洪强iOS开发之数据存储

李洪强iOS开发之数据存储 iOS应用数据存储的常用方式 1.lXML属性列表(plist)归档 2.lPreference(偏好设置) 3.lNSKeyedArchiver归档(NSCoding) 4.lSQLite3 5.lCore Data Documents: 保存应用运行时生成的需要持久化的数据,iTunes同步设备时会备份该目录.例如,游戏应用可将游戏存档保存在该目录 tmp: 保存应用运行时所需的临时数据,使用完毕后再将相应的文件从该目录删除.应用没有运行时,系统也可能会清除该目录