iOS多线程之NSOperation

1. NSOperation相关概念

NSOperation、NSOperationQueue 是基于 GCD 更高一层的封装,完全面向对象,是另外一套多线程解决方案。但是比 GCD 更简单易用、代码可读性也更高。在 NSOperation、NSOperationQueue 中也有类似的任务(操作)队列(操作队列)的概念。

1. 操作(Operation):执行操作的意思,换句话说就是你在线程中执行的那段代码。

2. 操作队列(Operation Queues):这里的队列指操作队列,即用来存放操作的队列。

2. NSOperation 的使用步骤

1. 创建操作:先将需要执行的操作封装到一个 NSOperation 对象中。
2. 创建队列:创建 NSOperationQueue 对象。
3. 将操作加入到队列中:将 NSOperation 对象添加到 NSOperationQueue 对象中。

3. 创建操作 (不使用 NSOperationQueue 的情况下)

NSOperation 是个抽象类,不能用来封装操作。我们只有使用它的子类来封装操作。我们有三种方式来封装操作。

  1. 使用子类 NSInvocationOperation
  2. 使用子类 NSBlockOperation
  3. 自定义继承自 NSOperation 的子类,通过实现内部相应的方法(main)来封装操作。

在不使用 NSOperationQueue,单独使用 NSOperation 的情况下系统同步执行操作(不会开启子线程)。

3.1 单独使用子类 NSInvocationOperation

// (不会开启子线程,在当前线程中)
-(void)startNSOperationAction{
    
    // 1.0 创建 NSOperation 的子类 NSInvocationOperation
    NSInvocationOperation *invocaOperation=[[NSInvocationOperation alloc]initWithTarget:self selector:@selector(operationFunction) object:nil];
    
    // 开始操作(执行操作)
    [invocaOperation start];
}
-(void)operationFunction{
    
    [NSThread sleepForTimeInterval:2]; // 模拟耗时操作
    
    NSLog(@"执行任务,当前线程是:%@",[NSThread currentThread]);
}

最终运行结果:

TestModel[86359:13360769] 执行任务,当前线程是:<NSThread: 0x280a21380>{number = 1, name = main}

3.2 单独使用子类 NSBlockOperation

// 场景一 (不会开启子线程,在当前线程中),不使用 addExecutionBlock: 的情况下
-(void)startNSOperationAction{
    
    // 1.0 创建 NSOperation 的子类 NSBlockOperation
    
    NSBlockOperation *blockOperation=[NSBlockOperation blockOperationWithBlock:^{
        
        [NSThread sleepForTimeInterval:2]; // 模拟耗时操作
        
        NSLog(@"执行任务,当前线程是:%@",[NSThread currentThread]);
    }];
    
    // 2.0 开始操作(执行操作)
    [blockOperation start];
  
}

最终运行结果:

TestModel[86367:13362642] 执行任务,当前线程是:<NSThread: 0x280769a40>{number = 1, name = main}





// 场景二 会开启一个子线程(添加更多的操作 使用 addExecutionBlock: )
-(void)startNSOperationAction{
    
    // 1.0 创建 NSOperation 的子类 NSBlockOperation
    
    NSBlockOperation *blockOperation=[NSBlockOperation blockOperationWithBlock:^{
        
//        [NSThread sleepForTimeInterval:2]; // 模拟耗时操作
        NSLog(@"执行任务(1),当前线程是:%@",[NSThread currentThread]);
    }];
    
    // 添加更多的操作
    [blockOperation addExecutionBlock:^{
        NSLog(@"执行任务(2),当前线程是:%@",[NSThread currentThread]);
    }];
    
    [blockOperation addExecutionBlock:^{
        NSLog(@"执行任务(3),当前线程是:%@",[NSThread currentThread]);
    }];
    
    [blockOperation addExecutionBlock:^{
        NSLog(@"执行任务(4),当前线程是:%@",[NSThread currentThread]);
    }];
    
    [blockOperation addExecutionBlock:^{
        NSLog(@"执行任务(5),当前线程是:%@",[NSThread currentThread]);
    }];
    
    // 2.0 开始操作(执行操作)
    [blockOperation start];
 
    // 注意点: 不能将 [blockOperation start]; 放在 [blockOperation addExecutionBlock:^{}]; 前面。不然程序会被奔溃,因为 blockOperation 已经销毁了 你还执行 操作。
}
  
最终运行结果:

TestModel[86378:13364140] 执行任务(1),当前线程是:<NSThread: 0x28174aec0>{number = 1, name = main}
TestModel[86378:13364140] 执行任务(3),当前线程是:<NSThread: 0x28174aec0>{number = 1, name = main}
TestModel[86378:13364159] 执行任务(2),当前线程是:<NSThread: 0x28171cf00>{number = 3, name = (null)}
TestModel[86378:13364140] 执行任务(4),当前线程是:<NSThread: 0x28174aec0>{number = 1, name = main}
TestModel[86378:13364159] 执行任务(5),当前线程是:<NSThread: 0x28171cf00>{number = 3, name = (null)}


通过 addExecutionBlock: 就可以为 NSBlockOperation 添加额外的操作。如果添加的操作多的话,blockOperationWithBlock: 中的操作也可能会在其他线程(非当前线程)中执行,这是由系统决定的,并不是说添加到 blockOperationWithBlock: 中的操作一定会在当前线程中执行。
 

3.3 单独使用自定义继承自 NSOperation 的子类

使用自定义继承自 NSOperation 的子类。可以通过重写 main 或者 start 方法 来定义自己的 NSOperation 对象

// CustomOperationClass.h 文件

#import <Foundation/Foundation.h>

@interface CustomOperationClass : NSOperation

@end


// CustomOperationClass.m 文件

@implementation CustomOperationClass

// 重写 main 方法
-(void)main{
    
    [NSThread sleepForTimeInterval:2];
    
    NSLog(@"使用自定义NSOperation,当前线程是: %@",[NSThread currentThread]);
}

@end

使用的时候导入头文件 CustomOperationClass.h。
  
// 使用自定义继承自 NSOperation 的子类(不会开启子线程,在当前线程)
-(void)startNSOperationAction{
    
    // 1.0 初始化自定义的NSOperation的类
    CustomOperationClass *cusOperation = [[CustomOperationClass alloc]init];
    
    // 2.0 开始操作(执行操作)
    [cusOperation start];
}

最终运行结果:

TestModel[86394:13367448] 使用自定义NSOperation,当前线程是: <NSThread: 0x280cdae80>{number = 1, name = main}


4. 创建队列 ( NSOperationQueue )

NSOperationQueue 一共有两种队列:主队列、自定义队列。其中自定义队列同时包含了串行、并发功能

主队列: 凡是添加到主队列中的操作,都会放到主线程中执行
// 主队列获取方法
NSOperationQueue *queue = [NSOperationQueue mainQueue];


自定义队列(非主队列), 添加到这种队列中的操作,就会自动放到子线程中执行。同时包含了:串行、并发功能。
// 自定义队列创建方法
NSOperationQueue *queue = [[NSOperationQueue alloc] init];

5. 将操作加入到队列中

上边我们说到 NSOperation 需要配合 NSOperationQueue 来实现多线程。

我们需要将创建好的操作加入到队列中去。总共有两种方法:

5.1 通过 addOperation: 添加操作到队列中

// 使用 addOperation: 将操作加入到队列中
-(void)startNSOperationAction{
    
    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.0 创建操作
    // 使用 NSInvocationOperation 创建操作1
    NSInvocationOperation *invocaOperation = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(runFunOne) object:nil];
    
    // 使用 NSInvocationOperation 创建操作2
    NSInvocationOperation *invocaOperationWithNext = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(runFunTwo) object:nil];
    
    
    // 使用 NSBlockOperation 创建操作3
    NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
         NSLog(@"执行任务(3),当前线程是:%@",[NSThread currentThread]);
    }];
    // 使用 NSBlockOperation 创建操作4
    [blockOperation addExecutionBlock:^{
        NSLog(@"执行任务(4),当前线程是:%@",[NSThread currentThread]);
    }];
    
    // 3.0 使用 addOperation: 添加所有操作到队列中(默认会自己启动)
    [queueObj addOperation:invocaOperation];
    [queueObj addOperation:invocaOperationWithNext];
    [queueObj addOperation:blockOperation];
    
}

-(void)runFunOne{
    NSLog(@"执行任务(1),当前线程是:%@",[NSThread currentThread]);
}

-(void)runFunTwo{
    NSLog(@"执行任务(2),当前线程是:%@",[NSThread currentThread]);
}

最终运行结果:

TestModel[86686:13417741] 执行任务(2),当前线程是:<NSThread: 0x2833db400>{number = 4, name = (null)}
TestModel[86686:13417737] 执行任务(1),当前线程是:<NSThread: 0x2833e5c40>{number = 3, name = (null)}
TestModel[86686:13417737] 执行任务(3),当前线程是:<NSThread: 0x2833e5c40>{number = 3, name = (null)}
TestModel[86686:13417737] 执行任务(4),当前线程是:<NSThread: 0x2833e5c40>{number = 3, name = (null)}

可以看出:使用 NSOperation 子类创建操作,并使用 addOperation: 将操作加入到自定义队列后 能够开启新线程,进行并发执行。

5.2 通过 addOperationWithBlock: 直接创建操作

// 使用 addOperationWithBlock: 添加操作到队列中
-(void)startNSOperationAction{
    
    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.使用 addOperationWithBlock: 添加操作到队列中
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(1),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(2),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(3),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(4),当前线程是:%@",[NSThread currentThread]);
    }];
    
}

最终运行结果:

TestModel[86693:13419925] 执行任务(1),当前线程是:<NSThread: 0x281a9b580>{number = 3, name = (null)}
TestModel[86693:13419925] 执行任务(3),当前线程是:<NSThread: 0x281a9b580>{number = 3, name = (null)}
TestModel[86693:13419925] 执行任务(4),当前线程是:<NSThread: 0x281a9b580>{number = 3, name = (null)}
TestModel[86693:13419927] 执行任务(2),当前线程是:<NSThread: 0x281aa9e40>{number = 4, name = (null)}

可以看出:使用 addOperationWithBlock: 将操作加入到操作队列后能够开启新线程,进行并发执行。

6. NSOperationQueue 控制串行执行、并发执行

NSOperationQueue 创建的自定义队列同时具有串行、并发功能,上边我演示了并发功能,那么他的串行功能是如何实现的?

通过设置属性 maxConcurrentOperationCount (最大并发操作数) 的个数 决定队列类型

maxConcurrentOperationCount 默认情况下为 -1,表示不进行限制,可进行并发执行。
maxConcurrentOperationCount 为 1 时,队列为串行队列。只能串行执行。
maxConcurrentOperationCount 大于 1 时,队列为并发队列。操作并发执行,当然这个值不应超过系统限制,即使自己设置一个很大的值,系统也会自动调整为 min

  
具体代码如下:

// 设置最大并发操作数为 1 , 串行队列(顺序执行)
-(void)startNSOperationAction{
    
    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.0 设置最大并发操作数
    queueObj.maxConcurrentOperationCount = 1;  // 串行队列(顺序执行)
    
    // 3.使用 addOperationWithBlock: 添加操作到队列中
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(1),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(2),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(3),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(4),当前线程是:%@",[NSThread currentThread]);
    }];
    
}

最终运行结果:

TestModel[86709:13422361] 执行任务(1),当前线程是:<NSThread: 0x280733e00>{number = 4, name = (null)}
TestModel[86709:13422362] 执行任务(2),当前线程是:<NSThread: 0x280737640>{number = 5, name = (null)}
TestModel[86709:13422362] 执行任务(3),当前线程是:<NSThread: 0x280737640>{number = 5, name = (null)}
TestModel[86709:13422362] 执行任务(4),当前线程是:<NSThread: 0x280737640>{number = 5, name = (null)}



// 设置最大并发操作数为 2 , 并发队列(无序执行)
-(void)startNSOperationAction{
    
    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.0 设置最大并发操作数
    queueObj.maxConcurrentOperationCount = 2;  // 并发队列(无序执行)
    
    // 3.使用 addOperationWithBlock: 添加操作到队列中
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(1),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(2),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(3),当前线程是:%@",[NSThread currentThread]);
    }];
    [queueObj addOperationWithBlock:^{
        NSLog(@"执行任务(4),当前线程是:%@",[NSThread currentThread]);
    }];
    
}

最终运行结果:

TestModel[86712:13422773] 执行任务(2),当前线程是:<NSThread: 0x28158d600>{number = 3, name = (null)}
TestModel[86712:13422773] 执行任务(3),当前线程是:<NSThread: 0x28158d600>{number = 3, name = (null)}
TestModel[86712:13422773] 执行任务(4),当前线程是:<NSThread: 0x28158d600>{number = 3, name = (null)}
TestModel[86712:13422772] 执行任务(1),当前线程是:<NSThread: 0x281598700>{number = 4, name = (null)}


可以看出:当最大并发操作数为 1 时,操作是按顺序串行执行的,并且一个操作完成之后,下一个操作才开始执行。
        当最大操作并发数为 2 时,操作是并发执行的,可以同时执行两个操作。而开启线程数量是由系统决定的,不需要我们来管理。

7. NSOperation其他操作

7.1 线程之间的通讯( 子线程回到主线程 )

当我们在其子线程中完成了耗时操作时,需要回到主线程进行 UI 刷新,那么就用到了线程之间的通讯。

// 线程间通讯(子线程回到主线程)
-(void)startCommunicationAction{

    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.0 添加操作(任务)
    [queueObj addOperationWithBlock:^{
        
         NSLog(@"在子线程中,执行任务(我好难呀),当前线程是:%@",[NSThread currentThread]);
        
        // 3.0 获取主线程
        NSOperationQueue *mainQueue = [NSOperationQueue mainQueue];
        
        // 4.0 回到主线程
        [mainQueue addOperationWithBlock:^{
            NSLog(@"回到主线程了,执行任务(😁),当前线程是:%@",[NSThread currentThread]);
        }];
        
    }];
    
}

最终运行结果:

TestModel[86757:13430970] 在子线程中,执行任务(我好难呀),当前线程是:<NSThread: 0x282596080>{number = 3, name = (null)}
TestModel[86757:13430948] 回到主线程了,执行任务(😁),当前线程是:<NSThread: 0x2825c2e40>{number = 1, name = main}

7.2 操作之间添加依赖 ( addDependency : )

使用场景:  比如说有A、B 、C 三个操作,其中 A执行完操作,C才能执行操作,B最后执行 (A-C-B的顺序)。
  
// 操作(任务)之间添加依赖
-(void)startNSOperationAction{
    
    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.0 创建三个操作(任务)
    NSBlockOperation *blockOperationA = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"执行任务(A),当前线程是:%@",[NSThread currentThread]);
    }];
    
    NSBlockOperation *blockOperationB = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"执行任务(B),当前线程是:%@",[NSThread currentThread]);
    }];
    
    NSInvocationOperation *invocaOperationC = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(runFunOne) object:nil];
    
    
    // 3.0 添加操作(任务)之间的依赖(不能相互依赖)
    [invocaOperationC addDependency:blockOperationA];
    [blockOperationB addDependency:invocaOperationC];
    
    // 4.0 使用 addOperation: 添加三个操作到队列中(启动)
    [queueObj addOperation:blockOperationA];
    [queueObj addOperation:blockOperationB];
    [queueObj addOperation:invocaOperationC];

}

-(void)runFunOne{
     NSLog(@"执行任务(C),当前线程是:%@",[NSThread currentThread]);
}

最终运行结果:

TestModel[86752:13429044] 执行任务(A),当前线程是:<NSThread: 0x281a28040>{number = 3, name = (null)}
TestModel[86752:13429042] 执行任务(C),当前线程是:<NSThread: 0x281a34280>{number = 4, name = (null)}
TestModel[86752:13429042] 执行任务(B),当前线程是:<NSThread: 0x281a34280>{number = 4, name = (null)}
  

7.3 操作(任务)完成时回调

// 操作(任务)完成时的通知 (通过 completionBlock )
-(void)startNSOperationFinish{
    
    // 1.0 创建自定义队列
    NSOperationQueue *queueObj = [[NSOperationQueue alloc]init];
    
    // 2.0 添加操作(任务)
    NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
         NSLog(@"在子线程中,执行任务(❤️),当前线程是:%@",[NSThread currentThread]);
    }];
    
    // 3.0 添加操作完成后的通知
    blockOperation.completionBlock = ^{
        NSLog(@"执行任务完毕了(😁)!可以下课放学回家了! 当前线程是:%@",[NSThread currentThread]);
    };
    
    // 4.0 将操作添加到队列中(默认开启)
    [queueObj addOperation:blockOperation];
    
}

最终运行结果:

TestModel[86766:13433031] 在子线程中,执行任务(❤️),当前线程是:<NSThread: 0x283ab8d40>{number = 4, name = (null)}
TestModel[86766:13433028] 执行任务完毕了(😁)!可以下课放学回家了! 当前线程是:<NSThread: 0x283a80e00>{number = 5, name = (null)}

7.4 其他函数(方法)和属性

1. - (void)cancel; 可取消操作,实质是标记 isCancelled 状态。
  
2. 判断操作状态方法
  - (BOOL)isFinished; 判断操作是否已经结束。
  - (BOOL)isCancelled; 判断操作是否已经标记为取消。
  - (BOOL)isExecuting; 判断操作是否正在在运行。
  - (BOOL)isReady; 判断操作是否处于准备就绪状态,这个值和操作的依赖关系相关。
  
3. 操作同步
  + (id)mainQueue; 获取主队列。
  + (id)currentQueue; 获取当前队列,如果当前线程不是在 NSOperationQueue 上运行则返回 nil。
  - (void)addOperations:(NSArray *)ops waitUntilFinished:(BOOL)wait; 向队列中添加操作数组,wait 标志是否阻塞当前线程直到所有操作结束
  - (NSArray *)operations; 当前在队列中的操作数组(某个操作执行结束后会自动从这个数组清除)。
  - (NSUInteger)operationCount; 当前队列中的操作数。
  - (void)addDependency:(NSOperation *)op; 添加依赖
  - (void)removeDependency:(NSOperation *)op; 移除依赖
  - (void)cancelAllOperations;  可以取消队列的所有操作。
  - (BOOL)isSuspended; 判断队列是否处于暂停状态。 YES 为暂停状态,NO 为恢复状态。
  - (void)setSuspended:(BOOL)b; 可设置操作的暂停和恢复,YES 代表暂停队列,NO 代表恢复队列。
  
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,607评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,047评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,496评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,405评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,400评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,479评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,883评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,535评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,743评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,544评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,612评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,309评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,881评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,891评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,136评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,783评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,316评论 2 342

推荐阅读更多精彩内容