来一次有侧重点的区分Swift与Objective-C

栏目: Objective-C · 发布时间: 5年前

内容简介:[TOC]@(swift)[温故而知新]面试中经常被问到

[TOC]

@(swift)[温故而知新]

面试中经常被问到 Objective-CSwift 的区别,其实区别还是很多的,重点整理一下个人觉得很重要的: 面向协议编程

一、Objective-C与Swift的异同

1.1、swift和OC的共同点:

  • OC 出现过的绝大多数概念,比如 引用计数ARC (自动引用计数)、 属性协议接口初始化扩展类命名参数匿名函数 等,在 Swift 中继续有效(可能最多换个术语)。
  • SwiftObjective-C 共用一套运行时环境, Swift 的类型可以桥接到 Objective-C (下面我简称OC),反之亦然

1.2、swift的优点:

  • swift注重安全, OC 注重灵活
  • swift注重面向协议编程、函数式编程、面向对象编程, OC 注重面向对象编程
  • swift注重值类型, OC 注重指针和引用
  • swift是静态类型语言, OC 是动态类型语言
  • swift容易阅读,文件结构和大部分语法简易化,只有.swift文件,结尾不需要分号
  • swift中的可选类型,是用于所有数据类型,而不仅仅局限于类。相比于 OC 中的 nil 更加安全和简明
  • swift中的泛型类型更加方便和通用,而非 OC 中只能为集合类型添加泛型
  • swift中各种方便快捷的高阶函数( 函数式编程 ) ( Swift的标准数组支持三个高阶函数: mapfilterreduce ,以及map的扩展 flatMap )
  • swift新增了两种权限,细化权限。 open > public > internal(默认) > fileprivate > private
  • swift中独有的元组类型( tuples ),把多个值组合成复合值。元组内的值可以是任何类型,并不要求是相同类型的。

1.3、swift的不足:

  • 版本不稳定
  • 公司使用比例不高,使用人数比例偏低
  • 有些语法其实可以只规定一种格式,不必这样也行,那样也行。像 Go 一样禁止一切(Go有点偏激)耍花枪的东西,同一个规范,方便团队合作和阅读他人代码。

Swift 跟 JavaScript 有什么相同和不同点?

从数据结构角度,Golang和Swift对比,有何优缺点?

iOS——Objective-C与Swift优缺点对比

二、Objective-C中的protocol与Swift中的protocol的区别

相比于 OCSwift 可以做到 protocol 协议方法的具体 默认实现 (通过 extension )相比 多态 更好的实现了 代码复用 ,而 OC 则不行。

三、 面向协议面向接口 )与 面向对象 的区别

面向对象面向协议 的的最明显区别是 对抽象数据的使用方式 ,面向对象采用的是 继承 ,而 面向协议 采用的是 遵守协议 。在 面向协议 设计中, Apple 建议我们更多的使用 值类型struct )而非 引用类型class )。这篇文章中有一个很好的例子说明了 面向协议面向对象 更符合 某些业务需求 。其中有飞机、汽车、自行车三种交通工具(均继承自父类交通工具);老虎、马三种动物(均继承父类自动物);在古代马其实也是一种交通工具,但是父类是动物,如果马也有交通 工具 的功能,则:

如果采用 面向对象编程 ,则需要既要继承动物,还要继承交通工具,但是父类交通工具有些功能马是不需要的。由此可见 继承 ,作为 代码复用 的一种方式, 耦合性 还是太强。 事物往往是一系列特质的组合,而不单单是以一脉相承并逐渐扩展的方式构建的。 以后慢慢会发现面向对象很多时候其实不能很好地对事物进行抽象。

来一次有侧重点的区分Swift与Objective-C

如果采用 面向协议编程 ,马只需要实现出行协议就可以拥有交通工具的功能了。 面向协议 就是这样的抽离方式,更好的职责划分,更加具象化,职责更加单一。 很明显 面向协议 的目的是为了降低代码的 耦合性

来一次有侧重点的区分Swift与Objective-C

总结:

面向协议 相对于 面向对象 来说更具有 可伸缩性可重用性 ,并且在编程的过程中更加 模块化 ,通过协议以及协议扩展替代一个 庞大的基类 ,这在 大规模系统编程 中会有很大的便捷之处。

3.1、 协议协议扩展基类 有三个明显的优点:

  • 1、类型可以遵守多个协议但是只有一个基类。 这意味着类型可以随意遵守任何想要特性的协议,而不需要一个巨大的基类。
  • 2、不需要知道源码就可以使用协议扩展添加功能。 这意味着我们可以任意扩展协议,包括swift内置的协议,而不需要修改 基类 源码 。一般情况下我们会给 特定的类 而非 类的层级继承体系 )添加扩展;但是如果必要,我们依然可以给 基类 添加扩展,使所有的子类 继承 添加的功能。使用协议,所有的 属性方法构造函数 都被定义在遵守协议的类型自身中。这让我们很容易地查看到所有东西是 怎么被定义初始化 的。 我们不需要在类的层级之间来回穿梭以查看所有东西是如何初始化的。 忘记设置超类可能没有什么大问题,但是在 更复杂 的类型中,忘记合理地设置某个属性可能会导致 意想不到 的行为。
  • 3、协议可以被类、结构体和枚举遵守,而类层级约束为类类型。 协议协议扩展 可以让我们在更合理的地方使用 值类型引用类型值类型 的一个主要的区别就是类型是如何传递的。当我们传递引用类型(class)的实例时,我们传递的对原实例的引用。这意味着所做的任何更改都会 反射回原实例中 。当我们传递值类型的实例时,我们传递的是对 原实例的一份拷贝 。这意味着所做的任何更改都不会反射回原实例中。使用值类型确保了我们总是得到一个唯一的实例因为我们传递了一份对原实例的拷贝而非对原实例的引用。因此,我们能相信没有代码中的其它部分会意外地修改我们的实例。这在 多线程 环境中尤其有用,其中不同的线程可以 修改数据 并创建 意外地行为

3.2、面向对象的特点

优点:

  • 封装

数据封装、访问控制、隐藏实现细节、类型抽象为类;

代码以逻辑关系组织到一起,方便阅读;

高内聚、低耦合的系统结构

  • 继承

代码重用,继承关系,更符合人类思维

  • 多态

接口重用,父类指针能够指向子类对象

当父类的引用指向子类对象时,就发生了向上转型,即把子类类型对象转成了父类类型。

向上转型的好处是隐藏了子类类型,提高了代码的扩展性。

多态的不足:

  • 父类有部分public方法是子类不需要的,也不允许子类覆盖重写
  • 父类有一些方法是必须要子类去覆盖重写的,在父类的方法其实也是一个空方法
  • 父类的一些方法即便被子类覆盖重写,父类原方法还是要执行的
  • 父类的一些方法是可选覆盖重写的,一旦被覆盖重写,则以子类为准

较好的抽象类型应该:

  • 更多地支持值类型,同时也支持引用类型
  • 更多地支持静态类型关联(编译期),同时也支持动态派发(runtime)
  • 结构不庞大不复杂
  • 模型可扩展
  • 不给模型强制添加数据
  • 不给模型增加初始化任务的负担
  • 清楚哪些方法该实现哪些方法不需实现

3.3、OneV's Den提到的 面向对象 的三个困境:

1、动态派发的安全性(这应该是OC的困境,在Swift中Xcode是不可能让这种问题编译通过的)

Objective-C 中下面这段代码编译是不会报警告和错误的

NSObject *v1 = [NSObject new];
NSString *v2 = [NSString new];
NSNumber *v3 = [NSNumber new];
NSArray *array = @[v1, v2, v3];
for (id obj in array) {
    [obj boolValue];
}
复制代码

Objective-C 中可以借助泛型检查这种潜在的问题,Xocde会提示警告

@protocol SafeProtocol <NSObject>
- (void)func;
@end

@interface SafeObj : NSObject<SafeProtocol>
@end
@implementation SafeObj
- (void)func {
    
}
@end

@interface UnSafeObj : NSObject
@end
@implementation UnSafeObj
@end
复制代码

Objective-CXcode7 中,可以使用带 泛型容器 也可以解决这个问题,但是只是:warning:,程序运行期间仍可能由于此问题导致的 崩溃

SafeObj *v1 = [[SafeObj alloc] init];
UnSafeObj *v2 = [[UnSafeObj alloc] init];
// 由于v2没有实现协议SafeProtocol,所以此处Xcode会有警告
// Object of type 'UnSafeObj *' is not compatible with array element type 'id<SafeProtocol>'
NSArray<id<SafeProtocol>> *array = @[v1, v2];
for (id obj in array) {
    [obj func];
}
复制代码

使用 swift ,必须指定类型,否则不是:warning:,而是:x:,所以 swift编译阶段 就可以检查出问题

// 直接报错,而不是警告
// Cannot convert value of type 'String' to expected argument type 'NSNumber'
var array: [NSNumber] = []
array.append(1)
array.append("a")
复制代码

2、横切关注点

我们很难在 不同的继承体系复用代码 ,用行话来讲就是 横切关注点Cross-Cutting Concerns )。比如下面的关注点 myMethod ,位于两条继承链 ( UIViewController -> ViewCotrollerUIViewController -> UITableViewController -> AnotherViewController ) 的横切面上。 面向对象 是一种不错的抽象方式,但是肯定不是最好的方式。它无法描述两个 不同事物 具有某个 相同特性 这一点。在这里, 特性的组合 要比继承更贴切事物的本质。

class ViewCotroller: UIViewController {   
    func myMethod() {
        
    }
}
复制代码
class AnotherViewController: UITableViewController {
    func myMethod() {
        
    }
}
复制代码

面向对象编程 中,针对这种问题的几种解决方案:

  • 1、 Copy & Paste

快速,但是这也是坏代码的开头。我们应该尽量避免这种做法。

  • 2、 引入 BaseViewController

看起来这是一个稍微靠谱的做法,但是如果不断这么做,会让所谓的 Base 很快变成垃圾堆。职责不明确,任何东西都能扔进 Base ,你完全不知道哪些类走了 Base ,而这个“超级类”对代码的影响也会不可预估。

  • 3、 依赖注入

通过外界传入一个带有 myMethod 的对象,用新的类型来提供这个功能。这是一个稍好的方式,但是引入额外的依赖关系,可能也是我们不太愿意看到的。

  • 4、 多继承

当然, Swift 是不支持多继承的。不过如果有多继承的话,我们确实可以从多个父类进行继承,并将 myMethod 添加到合适的地方。有一些语言选择了支持多继承 (比如 C++ ),但是它会带来 OOP 中另一个著名的问题: 菱形缺陷

Swift面向协议编程 中,针对这种问题的解决方案 ( 使用协议扩展添加默认实现 ):

protocol P {
    func myMethod()
}
extension P {
    func myMethod() {
        doWork()
    }
}
复制代码
extension ViewController: P { }
extension AnotherViewController: P { }

viewController.myMethod()
anotherViewController.myMethod()
复制代码

3、菱形问题

多继承中,两个父类实现了相同的方法, 子类无法确定继承哪个父类的此方法 ,由于多继承的拓扑结构是一个菱形,所以这个问题有被叫做菱形缺陷( Diamond Problem )。

上面的例子中,如果我们有多继承,那么 ViewControllerAnotherViewController 的关系可能会是这样的:

来一次有侧重点的区分Swift与Objective-C

如果 ViewControllerUITableViewController 都实现了 myMethod 方法,则在 AnotherViewController 中就会出现菱形缺陷问题。

我们应该着眼于写干净并安全的代码,干净的代码是非常易读和易理解的代码。

四、编程实践:基于 protocol 的链表实现

import UIKit

protocol ChainListAble {
    associatedtype T: Equatable
    // 打印
    var description: String{get}
    // 数量
    var count: Int{get}
    
    /// 插入
    func insertToHead(node: Node<T>)
    func insertToHead(value: T)
    func insertToTail(node: Node<T>)
    func insertToTail(value: T)
    func insert(node: Node<T>, afterNode: Node<T>) -> Bool
    func insert(value: T, afterNode: Node<T>) -> Bool
    func insert(node: Node<T>, beforNode: Node<T>) -> Bool
    func insert(value: T, beforNode: Node<T>) -> Bool
    
    /// 删除(默认第一个符合条件的)
    @discardableResult func delete(node: Node<T>) -> Bool
    @discardableResult func delete(value: T) -> Bool
    @discardableResult func delete(index: Int) -> Bool
    //func delete(fromIndex: Int, toIndex: Int) -> Bool
    //func deleteAll()
    
    /// 查找(默认第一个符合条件的)
    func find(value: T) -> Node<T>?
    func find(index: Int) -> Node<T>?
    
    /// 判断结点是否在链表中
    func isContain(node: Node<T>) -> Bool
}

/// [值类型不能在递归里调用](https://www.codetd.com/article/40263),因此Node类型只能是class而不是struct
// 有些时候你只能使用类而不能使用结构体,那就是递归里
// struct报错:Value type 'Node' cannot have a stored property that recursively contains it
class Node<T: Equatable> {
    var value: T
    var next: Node?
    
    /// 便利构造方法
    ///
    /// - Parameter value: value
    convenience init(value: T) {
        self.init(value: value, next: nil)
    }
    
    /// 默认指定初始化方法
    ///
    /// - Parameters:
    ///   - value: value
    ///   - next: next
    init(value: T, next: Node?) {
        self.value = value
    }
    
    // 销毁函数
    deinit {
        print("\(self.value) 释放")
    }
}

extension Node {
    /// 返回当前结点到链表尾的长度
    var count: Int {
        var idx: Int = 1
        var node: Node? = self
        while node?.value != nil {
            node = node?.next
            idx = idx + 1
        }
        return idx
    }
}

class SingleChainList: ChainListAble {
    typealias T = String
    // 哨兵结点,不存储数据
    private var dummy: Node = Node.init(value: "")
}
extension SingleChainList {
    var description: String {
        var description: String = ""
        var tempNode = self.dummy
        while let nextNode = tempNode.next {
            description = description + " " + nextNode.value
            tempNode = nextNode
        }
        return description
    }
    var count: Int {
        var count: Int = 0
        var tempNode = self.dummy
        while let nextNode = tempNode.next {
            count = count + 1
            tempNode = nextNode
        }
        return count
    }
    
    /// 在头部插入值
    ///
    /// - Parameter value: value
    func insertToHead(value: T) {
        let node: Node = Node.init(value: value)
        self.insertToHead(node: node)
    }
    /// 在头部插入结点
    ///
    /// - Parameter node: node
    func insertToHead(node: Node<T>) {
        node.next = self.dummy.next
        self.dummy.next = node
    }
    /// 在尾部插入值
    ///
    /// - Parameter value: value
    func insertToTail(value: T) {
        let node: Node = Node.init(value: value)
        self.insertToTail(node: node)
    }
    /// 在尾部插入结点
    ///
    /// - Parameter node: node
    func insertToTail(node: Node<T>) {
        var tailNode: Node = self.dummy
        while let nextNode = tailNode.next {
            tailNode = nextNode
        }
        tailNode.next = node
    }
    /// 在指定结点的后面插入新value
    ///
    /// - Parameters:
    ///   - value: 新值
    ///   - afterNode: 指定结点
    /// - Returns: true or false
    func insert(value: T, afterNode: Node<T>) -> Bool {
        let node: Node = Node.init(value: value)
        return self.insert(node: node, afterNode: afterNode)
    }
    /// 在指定结点的后面插入新结点
    ///
    /// - Parameters:
    ///   - value: 新结点
    ///   - afterNode: 指定结点
    /// - Returns: true or false
    func insert(node: Node<T>, afterNode: Node<T>) -> Bool {
        guard self.isContain(node: afterNode) else {
            return false
        }
        node.next = afterNode.next
        afterNode.next = node
        return true
    }
    /// 在指定结点的前面插入新value(双向链表实现这种插入方式速度比单向链表快)
    ///
    /// - Parameters:
    ///   - value: 新值
    ///   - beforNode: 指定结点
    /// - Returns: true or false
    func insert(value: T, beforNode: Node<T>) -> Bool {
        let node: Node = Node.init(value: value)
        return self.insert(node: node, beforNode: beforNode)
    }
    /// 在指定结点的前面插入新结点(双向链表实现这种插入方式速度比单向链表快)
    ///
    /// - Parameters:
    ///   - node: 新结点
    ///   - beforNode: 指定结点
    /// - Returns: true or false
    func insert(node: Node<T>, beforNode: Node<T>) -> Bool {
        var tempNode: Node = self.dummy
        while let nextNode = tempNode.next {
            if nextNode === beforNode {
                node.next = beforNode
                tempNode.next = node
                return true
            }
            tempNode = nextNode
        }
        return false
    }
    /// 删除指定value的结点
    ///
    /// - Parameter value: value
    /// - Returns: true or false
    func delete(value: T) -> Bool {
        var tempNode: Node = self.dummy
        while let nextNode = tempNode.next {
            // 此处判断 == 是否合理
            if nextNode.value == value {
                tempNode.next = nextNode.next
                return true
            }
            tempNode = nextNode
        }
        return false
    }
    /// 删除指定的结点
    ///
    /// - Parameter node: node
    /// - Returns: true or false
    func delete(node: Node<T>) -> Bool {
        var tempNode = self.dummy
        while let nextNode = tempNode.next {
            if nextNode === node {
                tempNode.next = nextNode.next
                return true
            }
            tempNode = nextNode
        }
        return false
    }
    /// 删除指定下标的结点
    ///
    /// - Parameter index: index
    /// - Returns: true or false
    func delete(index: Int) -> Bool {
        var idx: Int = 0
        var tempNode: Node = self.dummy
        while let nextNode = tempNode.next {
            if index == idx {
                tempNode.next = nextNode.next
                return true
            }
            tempNode = nextNode
            idx = idx + 1
        }
        return false
    }
    
    /// 查找指定值的node
    ///
    /// - Parameter value: value
    /// - Returns: node
    func find(value: T) -> Node<T>? {
        var tempNode = self.dummy
        while let nextNode = tempNode.next {
            if nextNode.value == value {
                return nextNode
            }
            tempNode = nextNode
        }
        return nil
    }
    /// 查找指定下标的结点
    ///
    /// - Parameter index: index
    /// - Returns: node
    func find(index: Int) -> Node<T>? {
        var idx: Int = 0
        var tempNode: Node = self.dummy
        while let nextNode = tempNode.next {
            if index == idx {
                return nextNode
            }
            tempNode = nextNode
            idx = idx + 1
        }
        return nil
    }
    /// 判断给定的链表是否在链表中
    ///
    /// - Parameter node: node
    /// - Returns: true or false
    func isContain(node: Node<T>) -> Bool {
        var tempNode = self.dummy.next
        while tempNode != nil {
            if tempNode === node {
                return true
            }
            tempNode = tempNode?.next
        }
        return false
    }
    /// 单向链表反转:方式一非递归实现
    ///
    /// - Parameter chainList: 源链表
    /// - Returns: 反转后的链表
    func reverseList() {
        var prevNode: Node<String>? = self.dummy.next
        var curNode: Node<String>? = prevNode?.next
        var tempNode: Node<String>? = curNode?.next
        prevNode?.next = nil
        while curNode != nil {            
            tempNode = curNode?.next
            curNode?.next = prevNode
            prevNode = curNode
            curNode = tempNode
        }
        self.dummy.next = prevNode
    }
    /// 单向链表反转:方式二递归实现
    ///
    /// - Parameter chainList: 源链表
    /// - Returns: 反转后的链表
    func reverseListUseRecursion(head: Node<T>?, isFirst: Bool) {
        var tHead = head
        if isFirst {
            tHead = self.dummy.next
        }
        guard let rHead = tHead else { return }
        if rHead.next == nil {
            self.dummy.next = rHead
            return
        }
        else {
            self.reverseListUseRecursion(head:rHead.next, isFirst: false)
            rHead.next?.next = rHead
            rHead.next = nil
        }
    }
}

class LinkedListVC: UIViewController {
    var chainList: SingleChainList = SingleChainList.init()
    override func viewDidLoad() {
        super.viewDidLoad()
        // 初始化链表
        for i in 0..<10 {
            let node: Node = Node.init(value: String(i))
            chainList.insertToTail(node: node)
        }
        // 查找结点
        for i in 0..<12 {
            if let find: Node = chainList.find(index: i) {
                debugPrint("find = \(find.value)")
            }
            else {
                debugPrint("not find idx = \(i)")
            }
        }
        // 删除结点
        if chainList.delete(index: 10) {
            debugPrint("删除 index = \(index)成功")
        }
        else {
            debugPrint("删除 index = \(index)失败")
        }
        // 打印结点value信息
        debugPrint(chainList.description)
        // 打印结点个数
        debugPrint(chainList.count)
        // 单向链表反转
        chainList.reverseList()
        // 打印结点value信息
        debugPrint(chainList.description)
        // 单向链表反转
        chainList.reverseListUseRecursion(head: nil, isFirst: true)
        // 打印结点value信息
        debugPrint(chainList.description)
    }
}
复制代码

以上所述就是小编给大家介绍的《来一次有侧重点的区分Swift与Objective-C》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

The Lean Startup

The Lean Startup

Eric Ries / Crown Business / 2011-9-13 / USD 26.00

更多中文介绍:http://huing.com Most startups fail. But many of those failures are preventable. The Lean Startup is a new approach being adopted across the globe, chan ging the way companies are built and ......一起来看看 《The Lean Startup》 这本书的介绍吧!

JSON 在线解析
JSON 在线解析

在线 JSON 格式化工具

XML、JSON 在线转换
XML、JSON 在线转换

在线XML、JSON转换工具

XML 在线格式化
XML 在线格式化

在线 XML 格式化压缩工具