【iOS面试总结】疫情隔离中,线上面试的问题集合(第一部分) ...

宁睿  金牌会员 | 2022-6-21 10:08:34 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 847|帖子 847|积分 2551

 1、内存管理部分

  1.1 介绍下内存管理机制

  在iOS中,使用引用计数来管理OC对象的内存   
  一个新创建的OC对象引用计数默认是1,当引用计数减为0,OC对象就会销毁,释放其占用的内存空间,调用retain会让OC对象的引用计数+1,调用release会让OC对象的引用计数-1   
  内存管理的经验总结   
  MRC下 :
    当调用alloc、new、copy、mutableCopy方法返回了一个对象,在不需要这个对象时,要调用release或者autorelease来释放它     
    想拥有某个对象,就让它的引用计数+1;不想再拥有某个对象,就让它的引用计数-1     
    可以通过以下私有函数来查看自动释放池的情况     
    extern void _objc_autoreleasePoolPrint(void);   
  ARC下
     LLVM + Runtime 会为我们代码自动插入 retain 和 release 以及 autorelease等代码,不需要我们手动管理
 
  1.2 AutoreleadPool底层结构

    AutoreleasePool并没有单独的结构,而是由若干个AutoreleasePoolPage以双向链表的形式组合而成的栈结构(分别对应结构中的parent指针和child指针)
  1.3 常见的内存泄漏问题

    泄露的内存主要有以下两种:  
      Leak Memory 这种是忘记 Release 操作所泄露的内存。
      Abandon Memory 这种是循环引用,无法释放掉的内存。
    当时我只列出了循环引用引起的内存泄漏问题,欢迎其他补充。
    1.3.1 NSTimer 

      NSTimer会默认对当前self有个强引用,所有在self使用完成打算是否的时候,一定要先使用NSTimer的invalidate来停止是否时间控制对self的引用
    1.3.2 Block

      Block也是比较常见的循环引用问题,在Block中使用了self容易出现循环引用,因此很多人在使用block的时候,加入里面有用到self的操作都会声明一个__weak来修饰self。其实便不是这样的,不是所有使用了Block都会出现Self循环引用问题,只有self拥有Block的强引用才会出现这种情况。
    1.3.3 delegate

      Delegate是ios中开发中最常遇到的循环引用,一般在声明delegate的时候都要使用弱引用weak或者assign  2、优化部分

   2.1 性能优化

     1、在正确的地方使用 reuseIdentifier :  
        正确使用identifier可以有效复用cell。
     2、避免过于庞大的XIB: 
        当你加载一个引用了图片或者声音资源的nib时,nib加载代码会把图片和声音文件写进内存。
     3、不要阻塞主线程:  
        永远不要使主线程承担过多。因为UIKit在主线程上做所有工作,渲染,管理触摸反应,回应输入等都需要在它上面完成。一直使用主线程的风险就是如果你的代码真的block了主线程,你的app会失去反应,大部分阻碍主进程的情形是你的app在做一些牵涉到读写外部资源的I/O操作,比如存储或者网络。 
     4、在Image Views中调整图片大小: 
        如果要在UIImageView中显示一个来自bundle的图片,你应保证图片的大小和UIImageView的大小相同。在运行中缩放图片是很耗费资源的,特别是UIImageView嵌套在UIScrollView中的情况下。
        如果图片是从远端服务加载的你不能控制图片大小,比如在下载前调整到合适大小的话,你可以在下载完成后,最好是用background thread,缩放一次,然后在UIImageView中使用缩放后的图片。
   2.2 耗电优化

     耗电大户:CPU、网络请求、定位、GPU、多媒体、相机等等。
     2.2.1 网络请求

       1、减少、压缩网络数据。可以降低上传或下载的多媒体内容质量和尺寸等。
       2、使用缓存,不要重复下载相同的数据。
       3、网络不可用时不要尝试执行网络请求。
     2.2.2 定位

       1、除非是在导航的时候,app大部分时间不需要实时更新,降低位置的更新频率。
       2、尽量降低定位精度。iOS设备默认采用最高精度定位,如果你的app不是确实需要米级的位置信息,不要用最高精度
     2.2.3 CPU

        1、尽量减少计时器使用。使用计时器时,设置一个合适的超时时,不再需要时及时关闭重复性定时器。用事件通知代替定时器。有些app用定时器监控文件内容、网络或者其他状态的变化,这会导致CPU无法进入闲置状态而增加功耗。
     2.2.4 优化I/O访问

       1、app每次执行I/O任务,比如写文件,会导致系统退出闲置模式。而且写入缓存格外耗电。
     2.2.5 优化通知

       1、尽量用本地通知(local notification),如果你的app不依赖外部数据,而是需要基于时间的通知,应该用本地通知,可以让设备的网络硬件休息一下。
       2、远程推送有两个级别,一个是立即推送,另一个是针对功耗优化过的延时推送。如果不是真的需要即时推送,尽量使用延时推送。
    2.3 启动优化

      App启动时间可以通过xcode提供的工具来度量,在Xcode的Product->Scheme-->Edit Scheme->Run->Auguments中,将环境变量DYLD_PRINT_STATISTICS设为YES,优化需以下方面入手
      1、核心思想是减少dylibs的引用
      2、合并现有的dylibs(最好是6个以内)
      3、使用静态库
      4、多使用Swift结构体
   2.4 瘦身优化

      降低包大小需要从两方面着手
      2.4.1 编译器优化:

        Strip Linked Product、Make Strings Read-Only、Symbols Hidden by Default 设置为 YES,去掉异常支持,Enable C++ Exceptions、Enable Objective-C Exceptions 设置为 NO, Other C Flags 添加 -fno-exceptions 利用 AppCode 检测未使用的代码:菜单栏 -> Code -> Inspect Code编写LLVM插件检测出重复代码、未被调用的代码
      2.4.2 资源优化:

        1、可以对资源进行无损的压缩
        2、去除没有用到的资源
   3、HTTP / HTTPS

     3.1、TCP/IP中文名字是啥?

        Transmission Control Protocol/Internet Protocol,传输控制协议/网际协议  
     3.2、TCP/IP 四层模型 和OSI七层模型

         3.3、HTTP是作用在哪一层?

        应用层
     3.4、HTTPS三次握手

        1、客户端发送SYN包到服务端,等待服务端确认;
        2、服务端确认接收SYN包,并发送回来一个SYN+ACK包给客户端;
        3、客户端确认接收,并向服务端发送确认包ACK,连接建立。
       4、Swift部分

     4.1 swift 和 objective-c 区别?

       1、swift是强类型(静态)语言,有类型判断,objective-c弱类型(动态)语言    
       2、swift面向协议编程,objective-c面向对象编程 
       3、swift比objective-c代码简洁
       4、swift注重值类型,objective-c注重引用类型
       5、swift支持静态派发、动态派发方式,objective-c仅支持动态派发方式 
     4.2 什么是可选型(optional)?

       可选型是为了表达一个变量为空的情况,当一个变量为空,它的值就是nil
       在类型名称后面加个?来定义一个可选型
       值类型或引用类型都可以是可选型变量
  1. var name: String? // 默认为 nil
  2. var age: Int?     // 默认为nil
  3. print(name, age) // 打印 nil, nil
复制代码
     4.3 什么是泛型?        

       泛型是为了增加代码的灵活性而生的,它可以是满足对应代码类型的任意变量或方法;可以将类型参数化,提高代码复用率,减少代码量
  1. func swap<T>(a: inout T, b: inout T) {
  2.     (a, b) = (b, a)
  3. }
复制代码
     4.4 访问控制关键字

       Swift 中有个5个级别的访问控制权限,从高到低依次是 open, public, internal, fileprivate, private
       它们遵循的基本规则: 高级别的变量不允许被定义为低级别变量的成员变量,比如一个 private 的 class 内部允许包含 public的 String值,反之低级变量可以定义在高级别变量中;
       1、open: 具备最高访问权限,其修饰的类可以和方法,可以在任意 模块中被访问和重写.
       2、public: 权限仅次于 open,和 open 唯一的区别是: 不允许其他模块进行继承、重写
       3、internal: 默认权限, 只允许在当前的模块中访问,可以继承和重写,不允许在其他模块中访问
       4、fileprivate: 修饰的对象只允许在当前的文件中访问;
       5、private: 最低级别访问权限,只允许在定义的作用域内访问
  5、多线程

     5.1 什么是多线程?

        多线程是指实现多个线程并发执行的技术,进而提升整体处理性能。
        同一时间,CPU 只能处理一条线程,多线程并发执行,其实是 CPU 快速的在多条线程之间调度(切换)如果 CPU 调度线程的时间足够快, 就造成了多线程并发执行的假象。
      5.2 使用多线程的优势和弊端?

        优势:充分发挥多核处理器的优势,将不同线程任务分配给不同的处理器,真正进入“并行计算”状态
        弊端:新线程会消耗内存控件和cpu时间,线程太多会降低系统运行性能。
     5.3 进程和多线程的区别?

        进程:正在运行的程序,负责程序的内存分配,每一个进程都有自己独立的虚拟内存空间。(一个程序运行的动态过程)
        线程:线程是进程中一个独立执行的路径(控制单元)一个进程至少包含一条线程,即主线程可以将耗时的执行路径(如网络请求)放在其他线程中执行。
        比较:
          1、线程是 CPU 调用的最小单位
          2、进程是 CPU 分配资源和调度的单位
            3、一个程序可以对应多个进程,一个进程中可有多个线程,但至少要有一条线程
          4、同一个进程内的线程共享进程资源
     5.4 GCD的调度队列有哪些?

        1、主队列(main queue)
          主队列是串行队列。和其它串行队列一样,这个队列中的任务一次只能执行一个。然而,它能保证所有的任务都在主线程执行,而主线程是唯一可用于更新 UI 的线程。这个队列就是用于发生消息给 UIView 或发送通知的。        2、全局调度队列(Global Dispatch Queues) 
          全局调度队列是并发队列。目前的四个全局队列有着不同的优先级:background、low、default 以及 high。 
        3、自定义队列
          你也可以创建自己的串行队列或并发队列。 
 

   接下文 【iOS面试总结】疫情隔离中,线上面试的问题集合(第二部分)


来源:https://www.cnblogs.com/xjf125/p/16179908.html
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

宁睿

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表