Swift -POP( 面向协议编程)与OOP(面向对象编程)
面向协议编程(Protocol Oriented Programming,简称POP),是Swift的一种编程范式,Apple于2015年WWDC提出的,如果大家看Swift的标准库,就会看到大量POP的影子。
同时Swift也是一门面向对象的编程语言(Object Oriented Programming,简称OOP),在Swift开发中,OOP和POP是相辅相成的,任何一方并不能取代另一方。
今天我们重点讲解下面向协议编程(POP)在Swift下的使用
回顾OOP
OOP的三大特性:封装、继承、多态
继承的经典使用场合
当多个类(比如A、B、C类)具有很多共性时,可以将这些共性抽取到一个父类(比如D类),最后A、B、C类继承D类
OOP的不足
但有些问题,使用OOP并不能很好的解决问题,比如如何将BVC、DVC的公共方法run抽取出来?
class BVC: UIViewController{
func run() {
print("run")
}
} class DVC: UITableViewController{
func run() {
print("fun")
}
}
基于OOP想到的一些解决方案?
- 将run方法放到另一个对象A中,然后BVC、DVC拥有A属性--多了一些额外的依赖关系
- 将run方法增加到UIViewController分类中--UIViewController会越来越臃肿,而且会影响它的其它所有子类
- 将run方法抽取到新的父类,采用多继承?Swift不支持多继承-(C++支持多继承)
POP的解决方案
protocol Runnable {
func run()
} extension Runnable {
func run() {
print("run")
}
} class BVC: UIViewController, Runnable{}
class DVC: UITableViewController, Runnable{}
POP的注意点
- 优先考虑创建协议,而不是父类(基类)
- 优先考虑值类型(struct, enum),而不是引用类型(class)
- 巧妙利用协议拓展功能
- 不要为了面向协议而实用协议
实例-使用协议实现前缀效果
统计字符串有几个数字出现,例如1234dafdaf1234,应该返回数字8?
1、如果在平常一处用到,可能大家直接会写一个对象方法,然后调用一下,即可计算出,如下:
func numberCount(_ str: String) -> Int {
var count =
for c in str where (""..."").contains(c) {
count +=
}
return count
}
通过调用self.numberCount("1234dafdaf1234")得出结论
2、通过上面需求,可以得出是字符串的拓展功能,因此想到另外一种对字符串进行拓展extension String,计算属性相当于方法
extension String {
///计算属性===方法,下面两种完全等价
//方法
func numberCount() -> Int {
var count =
for c in self where(""..."").contains(c) {
count +=
}
return count
} //计算属性
var numberCount1: Int {
var count =
for c in self where (""..."").contains(c) {
count +=
}
return count
} }
print("1234dafdaf1234".numberCount())
3、通过上面的已经可以很好的解决问题啦,对于要求很高的团队,上面还是有一些问题的,比如,上面是对系统的String进行拓展,如果定义的计算属性和系统的属性名字相同咋办,针对这个问题,可以有,在numberCount前面加入项目简称----(计算属性的本质是方法,要与存储属性分别开,不能为extension添加存储属性,如果要想添加,建立关联添加)
extension String {
///计算属性===方法,下面两种完全等价
//方法
func hcc_NumberCount() -> Int {
var count =
for c in self where(""..."").contains(c) {
count +=
}
return count
} //计算属性
var hccNumberCount1: Int {
var count =
for c in self where (""..."").contains(c) {
count +=
}
return count
} }
print("1234dafdaf1234".hccNumberCount1)
可以解决上面遇到的问题,但是不够优雅,也不符合Swift的风格,Swift中有大量协议的使用,如果有协议的加入,可以比较优雅的实现
是否可以实现"1234dafdaf1234".hcc.numberCount?
(因为是“”.hcc,所以对字符串拓展了一个属性hcc,而hcc.numberCount又是hcc类的一个属性,所以如下)
struct HCC {
var string: String
init(_ str: String) {
self.string = str
}
var numberCount: Int {
var count =
for c in string where (""..."").contains(c) {
count +=
}
return count
}
} extension String {
var hcc: HCC {return HCC(self)}//传值self字符串
} print("1234dafdaf1234".hcc.numberCount)
4、上面已经完成对字符串的拓展功能的系列,也已经很好的很优雅的解决了问题,但是如果相对字符串拓展一个功能的话,这就OK啦
如果想对数组进行拓展一个类似的方法,还要在HCC里面增加array属性和初始化以及拓展Array功能,就会发现冗余代码太多,且不够封装,不够通用
这时候泛型的作用就来啦,如下
struct HCC<Base> {
var base: Base
init(_ base: Base) {
self.base = base
}
} extension String {
var hcc: HCC<String> {HCC(self)}
} class Person{}
extension Person {
var hcc: HCC<Person> {HCC(self)}
} extension HCC where Base == String {
var numberCount: Int {
var count =
for c in base where(""..."").contains(c){
count +=
}
return count
}
} extension HCC where Base == Person {
func run() {
print("run")
}
} "1234dafdaf1234".hcc.numberCount
Person().hcc.run()
效果图如下
5、上面实现了通过类的对象调用,可不可以实现通过类本身来调用呢,因为我在使用类.hcc的时候,并不想出现类对象的属性,只想出现类型本身的方法和属性,这就需要用到static来修饰
struct HCC<Base> {
var base: Base
init(_ base: Base) {
self.base = base
}
} extension String {
var hcc: HCC<String> {HCC(self)}
static var hcc: HCC<String>.Type {HCC<String>.self}
} class Person{}
extension Person {
var hcc: HCC<Person> {HCC(self)}
} extension HCC where Base == String {
var numberCount: Int {
var count =
for c in base where(""..."").contains(c){
count +=
}
return count
} static func test() {
print("test")
}
} extension HCC where Base == Person {
func run() {
print("run")
}
} "1234dafdaf1234".hcc.numberCount
Person().hcc.run()
String.hcc.test()
完成上面需求!
6、但是如果要再次增加一个Dog类,也要在Dog类中有
var hcc: HCC<String> {HCC(self)}
static var hcc: HCC<String>.Type {HCC<String>.self}
这些代码,增加其他,会导致代码还是会有点冗余,这样就发现了POP的好处-是面向协议编程,将公共的地方抽出来(协议只能声明一些东西,想扩充一些东西,就是在extension加入)
///前缀类型
struct HCC<Base> {
var base: Base
init(_ base: Base) {
self.base = base
}
}
///利用协议扩展前缀属性
protocol HCCCompatible {}
extension HCCCompatible {
var hcc: HCC<Self> {HCC(self)}
static var hcc: HCC<Self>.Type {HCC<Self>.self}
}
///给字符串扩展功能
//让String拥有前缀属性
extension String: HCCCompatible {}
//给string.hcc以及String().hcc前缀扩展功能
extension HCC where Base == String {
var numberCount: Int {
var count = 0
for c in base where("0"..."9").contains(c){
count += 1
}
return count
}
static func test() {
print("test")
}
}
class Person{}
extension Person: HCCCompatible{}
class Dog{}
extension Dog: HCCCompatible{}
extension HCC where Base == Person {
func run() {
print("run")
}
}
这样就比较顺利较完美的解决了问题,又很好的拓展了功能!!!
总结
以后要给某一个类扩展功能,可采取下面步骤
- 定义一个前缀类型(如上面的hcc等)
- 定义一个协议(protocol)
- 遵守该协议即可
Swift -POP( 面向协议编程)与OOP(面向对象编程)的更多相关文章
- 为什么说swift是面向协议编程--草稿
为什么说swift是面向协议编程 public protocol ReactiveCompatible { /// Extended type associatedtype CompatibleTyp ...
- Python黑帽编程2.9 面向对象编程
Python黑帽编程2.9 面向对象编程 我个人认为,计算机语言的发展,有两个方向,一个是从低到高的发展过程,在这个过程中,语言的思考和解决问题的方式是面向硬件的.硬件本质上处理的是信号,在此基础上, ...
- Dart编程实例 - Dart 面向对象编程
Dart编程实例 - Dart 面向对象编程 class TestClass { void disp() { print("Hello World"); } } void main ...
- Swift中面向协议的编程
什么是面向协议的编程? 面向协议的编程,是一种编程范式. 编程范式,是一个计算机科学用语.维基百科中的解释是,计算机编程的基本风格或典型模式.通俗来说,就是解决某一个问题的方法不同方法和思路. 像大家 ...
- Java实现OOP(面向对象编程)
一.对象的综述 面向对象编程(OOP)具有多方面的吸引力.对管理人员,它实现了更快和更廉价的开发与维护过程.对分析与设计人员,建模处理变得更加简单,能生成清晰.易于维护的设计方案.对程序员,对象模型显 ...
- 用C实现OOP面向对象编程(1)
如摘要所说,C语言不支持OOP(面向对象的编程).并这不意味着我们就不能对C进行面向对象的开发,只是过程要复杂许多.原来以C++的许多工作,在C语言中需我们手动去完成. 博主将与大家一起研究一下如下用 ...
- 一百零六、SAP的OOP面向对象编程,OO-ALV的简介
面向对象编程,如图 基本概念: 1.对象(Object)是一个现实实体的抽象.一个对象可被认为是一个把数据(属性)和程序(方法)封装在一起的实体,这个程序产生该对象的动作或对它接受到的外界信号的反应. ...
- 为什么swift是面向协议的编程--对面向对象机制的改进
主要目标是提供抽象能力和解决值类型的多态问题 Actually, Abrahams says, those are all attributes of types, and classes are j ...
- (转)OOP(面向对象编程)的几大原则
文章转载自:http://blog.csdn.net/anders_zhuo/article/details/8949566 设计模式遵循的一般原则: 1.开-闭原则(Open-Closed Prin ...
随机推荐
- File流与IO流 看这一篇就够了
主要内容 File类 递归 IO流 字节流 字符流 异常处理 Properties 缓冲流 转换流 序列化流 打印流 学习目标 [ ] 能够说出File对象的创建方式 [ ] 能够说出File类获取名 ...
- Spring Boot 2.x基础教程:默认数据源Hikari的配置详解
通过上一节的学习,我们已经学会如何应用Spring中的JdbcTemplate来完成对MySQL的数据库读写操作.接下来通过本篇文章,重点说说在访问数据库过程中的一个重要概念:数据源(Data Sou ...
- shell命令之一天一见
一.在统计行数时常要用的到命令包括 w.c.l, 在这里做个简单的介绍. 语法:wc [选项] 文件… 说明:该命令统计给定文件中的字节数.字数.行数.如果没有给出文件名,则从标准输入读取.wc同时也 ...
- 第一次安装android studio遇到的问题
安装android studio一点都不顺利,最后总是成功安装,但是忘了把问题记录下来,下一次肯定还肯能出现问题 忘了把问题记录下来了,好像是sync failed 第一次安装3.1.4遇到的问题,好 ...
- Comet OJ Contest #0 解方程(暴力)
题意: 给定自然数n,求满足$\displaystyle \sqrt{x-\sqrt{n}}=\sqrt{z}-\sqrt{y}$的x,y,z,输出解的个数以及所有解 xyz的和 n<=1e9, ...
- num07---工厂方法模式
一.简单工厂模式 [之所以叫简单,说明没有完全做到 设计模式的要求] 前言:活字印刷术,面向对象思想 复用 维护 扩展 灵活 高内聚低耦合 以 实现 一个计算器 为例: 1.创建 抽象类count, ...
- Spring——管理Bean的生命周期
我们可以自定义bean的初始化和销毁方法,这里所指的的初始化和bean的构造不同,初始化是在bean构造完成后,对bean内部的属性或一些逻辑进行初始化. 首先要弄清一些概念: 构造(对象创建) 单实 ...
- finished with exit code -1073740791 (0xC0000409)解决方案
1.在用keras框架跑NER的train时,而且只是在用了keras_contrib.layers的CRF时出现问题: 遇到无错跳出finished with exit code -10737407 ...
- [Effective Java 读书笔记] 第二章 创建和销毁对象 第三 四条
第三条 用私有构造器或者枚举类型强化singleton属性 singleton指只能被实例化一次的类,即将构造器设置为私有,使用公有静态成员来实例化,且只实例化一次对象 第四条 通过私有构造器强化不可 ...
- centos7安装bind(DNS服务)
环境介绍 公网IP:149.129.92.239 内网IP:172.17.56.249 系统:CentOS 7.4 一.安装 yum install bind bind-utils -y 二.修改bi ...