objc与鸭子对象(上)
这是《objc与鸭子对象》的上半部分,《objc与鸭子对象(下)》中介绍了鸭子类型的进阶用法、依赖注入以及demo。
我是前言
鸭子类型
(Duck Type)即:“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子”,换成程序猿语言就是:“当调用者知道这个对象能调用什么方法时,管它这个对象到底是什么类的实例呢”。本文对objc中的鸭子类型对象进行简单探究,并用一个“只用一个类实现Json Entity”的小demo实践下这个思路的魔力。进阶篇请看下半部分。
objc与鸭子类型
id类型是个大鸭子
鸭子类型是动态语言的特性,编译时并不决定函数调用关系,说白了所有的类型声明都是给编译器看的。objc在动态和静态方面找到了不错的平衡,既保留了严格的静态检查也没破坏运行时的动态特性。
我们知道,向一个objc对象(或Class)发消息,实际上就是沿着它的isa
指针寻找真正函数地址,所以只要一个对象满足下面的结构,就可以对它发送消息:
struct objc_object { |
也就是熟知的id
类型,objc在语言层面先天就支持了这个基本的鸭子类型,我们可以将任意一个对象强转为id类型从而向它发送消息,就算它并不能响应这个消息,编译器也无从知晓。
正如这篇文章中对objc对象的简短定义:The best definition for a Smalltalk or Objective-C "object" is "something that can respond to messages.
object并非一定是某个特定类型的实例,只要它能响应需要的消息就可以了。
从@interface到@protocol
正如objc先天支持的动态的id
类型,@protocol
为鸭子类型提供了编译时的强类型检查,实现了Cocoa中经典的鸭子类型使用场景:
@property (nonatomic, assign) id <UITableViewDataSource> dataSource; |
利用鸭子类型设计的接口会给使用者更大的灵活度。同时@protocol
可以用来建立伪继承
关系
@protocol UIScrollViewDelegate<NSObject> |
<NSObject>
协议的存在一方面是给NSProxy
这样的其他根类使用,同时也给了鸭子协议类型一个根类型,正如给了大部分类一个NSObject根类一样。说个小插曲,由于objc中Class也是id
类型,形如id<UITableViewDataSource>
的鸭子类型是可以用Class对象来扮演的,只需要把实例方法替换成类方法,如:
@implementation DataSource |
设置table view的data source:
self.tableView.dataSource = (Class<UITableViewDataSource>)[DataSource class]; |
这种非主流写法合法且运行正常,归功于objc中加号和减号方法在@selector
中并未体现,在@protocol
中也是形同虚设,这种代码我相信没人真的写,但确实能体现鸭子类型的灵活性。
[Demo]一个类实现Json Entity
Entity
对象表示某个纯数据的结构,如:
@interface XXUserEntity : NSObject |
实际开发中这种类往往对应着server端返回的一个JSON串,如:
{"name": "sunnyxx", "sex": "boy", "age": 24, ...} |
解析这些映射是个纯重复工作,建类、写属性、解析…如今已经有JSONModel,Mantle等不错的框架帮忙。这个demo我们要用鸭子类型的思想去重新设计,把这些Entity类简化成一个鸭子类。
由于上面的UserEntity
类,只有属性的getter和setter,这正对应了NSMutableDictionary
的objectForKey:
和setObjectForKey:
,同时,JSON数据也会解析成字典,这就完成了巧妙的对接,下面去实现这个类。
真正干活的是一个字典,保证封装性和纯粹性,这个类直接使用NSProxy
作为纯代理类,只暴露一个初始化方法就好了:
// XXDuckEntity.h |
NSProxy
默认是没有初始化方法的,也省去了去规避其他初始化方法的麻烦,为了简单直接初始化时就把json串解开成字典(暂不考虑json是个array):
- (instancetype)initWithJSONString:(NSString *)json |
NSProxy
可以说除了重载消息转发机制外没有别的用法,这也是它被设计的初衷,自己什么都不干,转给代理对象就好。往这个proxy发消息是注定会走消息转发的,首先判断下是不是一个getter或setter的selector:
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector |
签名替换成字典的两个方法后开始走转发,在这里设置参数和对内部字典的真正调用:
- (void)forwardInvocation:(NSInvocation *)invocation |
当然还有这两个必不可少的从getter和setter中获取属性名的Helper:
- (NSString *)propertyNameScanFromGetterSelector:(SEL)selector |
一个简单的鸭子Entity就完成了,之后所有的Entity都可以使用@protocol
而非子类化的方式来定义,如:
@protocol XXUserEntity <NSObject> |
当数据从网络层回来时,鸭子类型让这个对象用起来和真有这么个类没什么两样:
- (void)requestFinished:(XXDuckEntity<XXStudentEntity> *)student { |
至此,所有的entity被表示成了N个<Protocol>
的.h
文件加一个XXDuckEntity
类,剩下的就靠想象力了。
这个demo的源码将在下半部分之后给出
http://blog.sunnyxx.com/2014/08/24/objc-duck/
objc与鸭子对象(上)的更多相关文章
- 通过jQuery Ajax使用FormData对象上传文件
FormData对象,是可以使用一系列的键值对来模拟一个完整的表单,然后使用XMLHttpRequest发送这个"表单". 在 Mozilla Developer 网站 使用For ...
- 在Function对象上扩展method方法
;(function() { /** * 在Function对象上扩展method方法 * @param {String} name 扩展的方法名称 * @param {Function} callb ...
- Javascript进阶篇——( JavaScript内置对象---上-Date,string,charAt,indexOf,split,substring,substr)笔记整理
什么是对象JavaScript 中的所有事物都是对象,如:字符串.数值.数组.函数等,每个对象带有属性和方法.对象的属性:反映该对象某些特定的性质的,如:字符串的长度.图像的长宽等:对象的方法:能够在 ...
- Effective JavaScript Item 36 实例状态仅仅保存在实例对象上
本系列作为EffectiveJavaScript的读书笔记. 一个类型的prototype和该类型的实例之间是"一对多"的关系.那么,须要确保实例相关的数据不会被错误地保存在pro ...
- IT轮子系列(四)——使用Jquery+formdata对象 上传 文件
前言 在MVC 中文件的上传,一般都采用控件: <h2>IT轮子四——文件上传</h2> <div> <input type="file" ...
- FormData 对象上传二进制文件
使用jQuery 利用 FormData 上传文件:http://harttle.com/2016/07/04/jquery-file-upload.html 通过FormData对象可以组装 ...
- WinForm 绑定到嵌套对象上的属性
WinFrom 绑定到嵌套对象上的属性 关键字: Windows Forms, DataBindings, Nested Class, 嵌套类 在 WinForm 中很早就已经支持数据绑定, 使用数据 ...
- 通过jQuery Ajax使用FormData对象上传文件 (转载)
XMLHttpRequest Level 2 添加了一个新的接口——FormData.与普通的 Ajax 相比,使用 FormData 的最大优点就是我们可以异步上传二进制文件.jQuery 2.0+ ...
- [转] 通过jQuery Ajax使用FormData对象上传文件
FormData对象,是可以使用一系列的键值对来模拟一个完整的表单,然后使用XMLHttpRequest发送这个"表单". 在 Mozilla Developer 网站 使用For ...
随机推荐
- 10-05 Java 内部类概述和讲解
内部类的概述 /* 内部类概述: 把类定义在其他类的内部,这个类就被称为内部类. 举例:在类A中定义了一个类B,类B就是内部类. 内部的访问特点: A:内部类可以直接访问外部类的成员,包括私有. B: ...
- odoo开发笔记--odoo可用小图标
odoo系统的小图标都采用了 fontawesome字体图标 官网: http://fontawesome.dashgame.com/ Github:http://fortawesome.github ...
- Python Web Server Gateway Interface -- WSGI
了解了HTTP协议和HTML文档,我们其实就明白了一个Web应用的本质就是: 浏览器发送一个HTTP请求: 服务器收到请求,生成一个HTML文档: 服务器把HTML文档作为HTTP响应的Body发送给 ...
- .NET跨平台实践:再谈用C#开发Linux守护进程 — 完整篇
Linux守护进程是Linux的后台服务进程,相当于Windows服务,对于为Linux开发服务程序的朋友来说,Linux守护进程相关技术是必不可少的,因为这个技术不仅仅是为了开发守护进程,还可以拓展 ...
- 安装Centos7 随手记
1.老笔记本安装Centos7 配置:酷睿I3 内存8G 2.原有系统Win7 将原来的硬盘空间,调整出60G 给Centos7 用. 3.安装Centos7 图形介面的,和windows安装过程类 ...
- 学会四招让你在linux下安装程序变得简单
一.背景 由于最近想自己摸索一些linux下的东西,开始玩起了Linux系统,在安装软件的过程中有诸多的不解和困惑,现在终于搞明白了具体是怎么样的安装步骤和过程,先分享给你们同时也方便自己复习查阅. ...
- 详解C#泛型(一)
一.C#中的泛型引入了类型参数的概念,类似于C++中的模板,类型参数可以使类型或方法中的一个或多个类型的指定推迟到实例化或调用时,使用泛型可以更大程度的重用代码.保护类型安全性并提高性能:可以创建自定 ...
- 使用ssh-keygen生成ssh公钥和私钥
默认是在用户目录下: windows一般为 c:\User\username\.ssh下 linux一般为 /home/username/.ssh 生成的命令如下: ssh-keygen -t rsa ...
- 【React 资料备份】React Hook
Hooks是React16.8一个新增项,是我们可以不用创建class组件就能使用状态和其他React特性 准备工作 升级react.react-dom npm i react react-dom - ...
- Docker 使用官方镜像
Docker 使用官方镜像 如何使用官方镜像 Docker 中国官方镜像加速可通过 registry.docker-cn.com 访问.目前该镜像库只包含流行的公有镜像,而私有镜像仍需要从美国镜像库中 ...