深入简出-- iOS编程思想

栏目: IOS · 发布时间: 6年前

内容简介:编程思想的由来:在开发中我们会遇见各种各样的需求,经常会思考如何快速的完成这些需求,这样就会慢慢形成快速完成这些需求的思想。下面我来总结一下我们开发中常用的编程思想。处理事情以过程为核心,一步一步的实现。C语言编程就是面向过程编程。万物皆对象。典型代表OC等。

编程思想的由来:在开发中我们会遇见各种各样的需求,经常会思考如何快速的完成这些需求,这样就会慢慢形成快速完成这些需求的思想。下面我来总结一下我们开发中常用的编程思想。

1、面向过程编程思想

处理事情以过程为核心,一步一步的实现。C语言编程就是面向过程编程。

2、面向对象编程思想

万物皆对象。典型代表OC等。

3、链式编程思想

是将多个操作(多行代码)通过点号(.)链接在一起成为一句代码,使代码可读性好。a(1).b(2).c(3)

  • 链式编程特点:方法的返回值是block,block必须有返回值(本身对象),block参数(需要操作的值

  • 代表:masonry框架

    为了便于理解我们练习写一个计算器。首先创建一个计算机类CalculateManager。平常我们用masonry的时候设置它的属性都会返回一个block,并且block的返回值为make本事,所以我们定义一个方法。

@interface CalculateManager : NSObject

@property (nonatomic,assign) int result;

-(CalculateManager *(^)(int))add;

@end

实现为:

-(CalculateManager *(^)(int))add{
    return ^CalculateManager *(int value){
        _result+=value;
        return self;
    };
}

并把最后的计算结果存储的result变量里面用于返回获取结果。

我们是想让NSObject及其子类都拥有计算能力,所以写一个Category:NSObject (Calculate)。写一个方法参数为一个Block,该block的参数为上面的计算机类对象,所有的计算都在这个block里面执行。

+ (int)cyx_makeCalcuclate:(void(^)(CalculateManager *))block;

实现为:

//将所有的计算代码都放在这里
+ (int)cyx_makeCalcuclate:(void(^)(CalculateManager *))block{

    CalculateManager * manager = [CalculateManager new];
    block(manager);
    return manager.result;
}

用法为:

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    int result =  [NSObject cyx_makeCalcuclate:^(CalculateManager *manager) {
       //在这里面实现存放计算代码
        //5+5
        manager.add(5).add(3);
    }];
    NSLog(@"%d",result);
}

这样就实现了。 附demo: https://github.com/SionChen/ChainProgramming

4、响应式编程思想

不需要考虑调用顺序,只需要知道考虑结果,类似于蝴蝶效应,产生一个事件,会影响很多东西,这些事件像流一样的传播出去,然后影响结果,借用面向对象的一句话,万物皆是流。

  • 代表:KVO运用。

    我们自己模仿实现一下KVO。在写之前我们首先了解一下KVO的底层实现:

基本的原理:

当观察某对象 A 时,KVO 机制动态创建一个对象A当前类的子类,并为这个新的子类重写了被观察属性 keyPath 的 setter 方法。setter 方法随后负责通知观察对象属性的改变状况。

深入剖析:

Apple 使用了 isa 混写(isa-swizzling)来实现 KVO 。当观察对象A时,KVO机制动态创建一个新的名为:NSKVONotifying_A 的新类,该类继承自对象A的本类,且 KVO 为 NSKVONotifying_A 重写观察属性的 setter 方法,setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察对象属性值的更改情况。

参考文章: https://www.jianshu.com/p/e59bb8f59302

话不多说我们开始写,首先创建一个person类定义一个实例变量:

@interface Person : NSObject
{
    @public
    NSString * _name;
}

@property (nonatomic,copy) NSString *name;
@end

创建一个NSObject的Category用于给所有NSObject及其子类新增 添加监听方法:

@interface NSObject (KVO)
- (void)cyx_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
@end
NSString * const cyx_key = @"observer";

@implementation NSObject (KVO)
-(void)cyx_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context
{


    objc_setAssociatedObject(self, (__bridge const void *)(cyx_key), observer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);

    //修改isa 指针
    object_setClass(self, [SonPerson class]);
}

这里利用runtime修改isa指针,修改调用方法时寻找方法的类。这里我们修改到SonPerson类。并在SonPerson类里面实现监听方法。

extern NSString * const cyx_key;
@implementation SonPerson

-(void)setName:(NSString *)name{
    [super setName:name];

    NSObject * observer = objc_getAssociatedObject(self, cyx_key);

    [observer observeValueForKeyPath:@"name" ofObject:self change:nil context:nil];
}

这里也用到了runtime 里面 objc_getAssociatedObject 和objc_setAssociatedObject动态存储方法。

好了那我们来用一下试一下效果吧。

#import "ViewController.h"
#import "Person.h"
#import "NSObject+KVO.h"
@interface ViewController ()

@property (nonatomic,strong) Person *p;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    Person * p = [[Person alloc] init];
    [p cyx_addObserver:self forKeyPath:@"name" options:0 context:nil];
    _p = p;
}
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
    NSLog(@"%@",_p.name);
}

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{

    static int i=0;
    i++;
    _p.name = [NSString stringWithFormat:@"%d",i];
    //_p -> _name = [NSString stringWithFormat:@"%d",i];

}

输出:

2018-04-21 11:15:17.974785+0800 04-响应式编程思想[1882:274712] 1
2018-04-21 11:15:18.293700+0800 04-响应式编程思想[1882:274712] 2
2018-04-21 11:15:18.687331+0800 04-响应式编程思想[1882:274712] 3
2018-04-21 11:15:19.036166+0800 04-响应式编程思想[1882:274712] 4
2018-04-21 11:15:19.396075+0800 04-响应式编程思想[1882:274712] 5
2018-04-21 11:15:19.699907+0800 04-响应式编程思想[1882:274712] 6
2018-04-21 11:15:19.981256+0800 04-响应式编程思想[1882:274712] 7

demo: https://github.com/SionChen/ReactiveProgramming

5、函数式编程思想

是把操作尽量写成一系列嵌套的函数或者方法调用。

  • 函数式编程本质:就是往方法中传入Block,方法中嵌套Block调用,把代码聚合起来管理

  • 函数式编程特点:每个方法必须有返回值(本身对象),把函数或者Block当做参数,block参数(需要操作的值)block返回值(操作结果)

  • 代表:ReactiveCocoa。

      我们用函数式编程实现,写一个加法计算器,创建一个计算机类CaluculateManager。

@interface CaluculateManager : NSObject
@property (nonatomic,assign) int result;
//计算
-(instancetype)caluculate:(int(^)(int))caluculateBlock;
@end
@implementation CaluculateManager

-(instancetype)caluculate:(int (^)(int))caluculateBlock{
    _result = caluculateBlock(_result);
    return self;
}
@end

使用:

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.


    CaluculateManager * manager = [CaluculateManager new];
    int result = [[manager caluculate:^int(int results){
        results+=5;
        results*=5;
        return results;
    }] result];
    NSLog(@"%d",result);
}

附demo: https://github.com/SionChen/FunctionalProgramming 。欢迎指教。


以上所述就是小编给大家介绍的《深入简出-- iOS编程思想》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

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

自己动手做iOS App

自己动手做iOS App

张子怡 / 电子工业出版社 / 2017-8 / 69.00

《自己动手做iOS App:从设计开发到上架App Store》为想要接触iOS 应用设计、开发的读者提供了由浅入深的详细指导。从iOS 应用制作的步骤是什么,应该使用什么软件,如何发布应用到App Store,到iOS 的设计理念是什么,如何正确书写Swift 语言,再到后端和客户端是如何交互运作的等,本书配合图示,精辟、直观地阐明了iOS 应用制作中的种种疑问。 如果你是一位第一次接触i......一起来看看 《自己动手做iOS App》 这本书的介绍吧!

CSS 压缩/解压工具
CSS 压缩/解压工具

在线压缩/解压 CSS 代码

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

在线 XML 格式化压缩工具