美文网首页iOS开发攻城狮的集散地
深入简出-- iOS编程思想

深入简出-- iOS编程思想

作者: Hedgehog___ | 来源:发表于2018-04-21 11:25 被阅读88次

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

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编程思想

    本文链接:https://www.haomeiwen.com/subject/vrntlftx.html