NSOperation和NSOperationQueue验证笔记

NSOperation

准备工作
// 自定义一个类继承NSBlockOperation,
// 重写相关的方法,观察声明周期

@interface LKBlockOperation : NSBlockOperation
@end

@implementation LKBlockOperation

- (void)dealloc
{
    NSLog(@"LKBlockOperation dealloc");
}

- (void)start
{
    NSLog(@"Operation before execute super start");
    [super start];
    NSLog(@"Operation after execute super start");
}

- (void)main
{
    NSLog(@"Operation before execute super main");
    [super main];
    NSLog(@"Operation after execute super main");
}
@end

操作执行过程
// 通过下面的代码和打印可知: 
// 把Operation加入到Queue中并不是立马就执行
// 执行过程: start -> main -> CompletionBlock -> dealloc

- (void)testExecuteOperation
{
     LKBlockOperation *operation = [[LKBlockOperation alloc] init];
    [operation addExecutionBlock:^{
        
        NSLog(@"before execute operation");
        NSInteger i = 0;
        while (i < 3) {
            NSLog(@"%@_%zd", [NSThread currentThread], ++i);
        }
        
        NSLog(@"after execute operation");
    }];
    
    [operation setCompletionBlock:^{
        NSLog(@"completion");
    }];
    
    NSLog(@"before queue Add Operation");
    [_queue addOperation:operation];
    NSLog(@"after queue Add Operation");
}

/*
2018-03-01 13:42:57.494851+0800 OperationQueues[3797:129659] before queue Add Operation
2018-03-01 13:42:57.495266+0800 OperationQueues[3797:129659] after queue Add Operation
2018-03-01 13:42:57.495350+0800 OperationQueues[3797:129708] Operation before execute super start
2018-03-01 13:42:57.495594+0800 OperationQueues[3797:129708] Operation before execute super main
2018-03-01 13:42:57.495940+0800 OperationQueues[3797:129708] before execute operation
2018-03-01 13:42:57.496402+0800 OperationQueues[3797:129708] <NSThread: 0x600000267600>{number = 3, name = (null)}_1
2018-03-01 13:42:57.497014+0800 OperationQueues[3797:129708] <NSThread: 0x600000267600>{number = 3, name = (null)}_2
2018-03-01 13:42:57.497155+0800 OperationQueues[3797:129708] <NSThread: 0x600000267600>{number = 3, name = (null)}_3
2018-03-01 13:42:57.497389+0800 OperationQueues[3797:129708] after execute operation
2018-03-01 13:42:57.497501+0800 OperationQueues[3797:129708] Operation after execute super main
2018-03-01 13:42:57.497645+0800 OperationQueues[3797:129708] Operation after execute super start
2018-03-01 13:42:57.497648+0800 OperationQueues[3797:129700] completion
2018-03-01 13:42:57.556400+0800 OperationQueues[3797:129700] LKBlockOperation dealloc
*/
如果任务还没加入到队列中就取消
// 通过下面的代码和打印可知: 
// 如果任务在没有加入到队列之前就取消,还是会调用start和CompletionBlock。但是main函数是不会执行了。
// 还可以看到在start方法执行之前,isCancel已经为YES,
// 说明调用cancel方法之后,isCacel就会设置为YES,然后在根据是否取消来决定是否调用main方法

- (void)testCancelBeforeAddToOperation
{
    LKBlockOperation *operation = [[LKBlockOperation alloc] init];
    [operation addExecutionBlock:^{
        NSInteger i = 0;
        while (1) {
            NSLog(@"%@_%zd", [NSThread currentThread], i ++);
        }
    }];
    
    [operation setCompletionBlock:^{
        NSLog(@"completion");
    }];
    
    [operation cancel];
    
    NSLog(@"before queue Add Operation");
    [_queue addOperation:operation];
    NSLog(@"after queue Add Operation");
}

/*
2018-03-01 13:59:02.415212+0800 OperationQueues[4099:142821] before queue Add Operation
2018-03-01 13:59:02.415394+0800 OperationQueues[4099:142821] after queue Add Operation
2018-03-01 13:59:02.415428+0800 OperationQueues[4099:142863] self.cancel = 1
2018-03-01 13:59:02.415526+0800 OperationQueues[4099:142863] Operation before execute super start
2018-03-01 13:59:02.415739+0800 OperationQueues[4099:142866] completion
2018-03-01 13:59:02.415739+0800 OperationQueues[4099:142863] Operation after execute super start
2018-03-01 13:59:02.415907+0800 OperationQueues[4099:142863] LKBlockOperation dealloc
*/
任务的中途取消
// 通过下面的代码和打印可知: 
// 通过上面的测试我们知道当调用Cancel方法之后,isCancelled就会设置为YES,
// 所以我们应该在每一次执行耗时操作或者是循环都应该判断下isCancelled,这也是苹果推荐的。
// 我把if(weakOperation.isCancelled) {break;}去掉之后,这个任务是不能正常取消的,就算调用了cancel方法。因为这个任务并没有结束的标志(return)
    
- (void)testCancelOperationWhenExecuting
{
    NSBlockOperation *operation = [[NSBlockOperation alloc] init];
    __weak NSBlockOperation *weakOperation = operation;
    
    [operation addExecutionBlock:^{
        
        NSLog(@"before execute operation");

        NSInteger i = 0;
        
        while (1) {
            
            NSLog(@"%@_%zd", [NSThread currentThread], i ++);
            if(weakOperation.isCancelled) {
                break;
            }
            
            if(i == 3) {
                [weakOperation cancel];
            }
        }
        
        NSLog(@"after execute operation");
    }];
    
    [operation setCompletionBlock:^{
        NSLog(@"completion");
    }];
    
    NSLog(@"before queue Add Operation");
    [_queue addOperation:operation];
    NSLog(@"after queue Add Operation");
}

/*
2018-03-01 14:08:52.408137+0800 OperationQueues[4342:152805] before queue Add Operation
2018-03-01 14:08:52.408288+0800 OperationQueues[4342:152805] after queue Add Operation
2018-03-01 14:08:52.408332+0800 OperationQueues[4342:152842] before execute operation
2018-03-01 14:08:52.408608+0800 OperationQueues[4342:152842] <NSThread: 0x60c00026ee40>{number = 3, name = (null)}_1
2018-03-01 14:08:52.408840+0800 OperationQueues[4342:152842] <NSThread: 0x60c00026ee40>{number = 3, name = (null)}_2
2018-03-01 14:08:52.408988+0800 OperationQueues[4342:152842] <NSThread: 0x60c00026ee40>{number = 3, name = (null)}_3
2018-03-01 14:08:52.409110+0800 OperationQueues[4342:152842] <NSThread: 0x60c00026ee40>{number = 3, name = (null)}_4
2018-03-01 14:08:52.409197+0800 OperationQueues[4342:152842] after execute operation
2018-03-01 14:08:52.409339+0800 OperationQueues[4342:152852] completion
*/

NSOperationQueue

准备工作
- (NSBlockOperation *)createOperationRepeatCount:(NSInteger)count operationName:(NSString *)operationName
{
    NSBlockOperation *operation = [[NSBlockOperation alloc] init];
    operation.name = operationName;
    
    __weak NSBlockOperation *weakOperation = operation;
    [operation addExecutionBlock:^{
        
        NSLog(@"%@ before execute operation", weakOperation.name);
        
        NSInteger i = 0;
        
        while (i < count) {
            NSLog(@"%@-%zd", weakOperation.name, ++i);
            if(weakOperation.isCancelled) {
                break;
            }
        }
        
        NSLog(@"%@ after execute operation", weakOperation.name);
    }];

    return operation;
}
测试Queue的Cancel
// 通过下面的代码和打印可知:
// 调用cancelAllOperations和调用Operation的cancel一致。相当于里面遍历Operation。然后调用cancel

- (void)testQueueCancel
{
    NSBlockOperation *operation0 = [self createOperationRepeatCount:NSIntegerMax operationName:@"operation0"];
    NSBlockOperation *operation1 = [self createOperationRepeatCount:NSIntegerMax operationName:@"operation1"];
    NSBlockOperation *operation2 = [self createOperationRepeatCount:NSIntegerMax operationName:@"operation2"];
    
    [_queue addOperation:operation0];
    [_queue addOperation:operation1];
    [_queue addOperation:operation2];
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.01 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [_queue cancelAllOperations];
    });
}

/*
2018-03-01 15:26:50.856400+0800 OperationQueues[6275:233403] operation2 before execute operation
2018-03-01 15:26:50.856403+0800 OperationQueues[6275:233402] operation0 before execute operation
2018-03-01 15:26:50.856412+0800 OperationQueues[6275:233400] operation1 before execute operation
2018-03-01 15:26:50.856559+0800 OperationQueues[6275:233403] operation2-1
2018-03-01 15:26:50.856570+0800 OperationQueues[6275:233402] operation0-1
2018-03-01 15:26:50.856571+0800 OperationQueues[6275:233400] operation1-1
2018-03-01 15:26:50.856706+0800 OperationQueues[6275:233403] operation2-2
2018-03-01 15:26:50.856773+0800 OperationQueues[6275:233402] operation0-2
2018-03-01 15:26:50.856777+0800 OperationQueues[6275:233400] operation1-2
2018-03-01 15:26:50.856795+0800 OperationQueues[6275:233403] operation2-3
2018-03-01 15:26:50.856959+0800 OperationQueues[6275:233402] operation0-3
2018-03-01 15:26:50.857556+0800 OperationQueues[6275:233400] operation1-3
2018-03-01 15:26:50.857759+0800 OperationQueues[6275:233403] operation2-4
2018-03-01 15:26:50.858005+0800 OperationQueues[6275:233402] operation0-4
2018-03-01 15:26:50.858166+0800 OperationQueues[6275:233400] operation1-4
2018-03-01 15:26:50.858610+0800 OperationQueues[6275:233403] operation2-5
2018-03-01 15:26:50.858802+0800 OperationQueues[6275:233402] operation0-5
2018-03-01 15:26:50.858977+0800 OperationQueues[6275:233400] operation1-5
2018-03-01 15:26:50.859305+0800 OperationQueues[6275:233403] operation2-6
2018-03-01 15:26:50.920413+0800 OperationQueues[6275:233403] operation2 after execute operation
2018-03-01 15:26:50.920420+0800 OperationQueues[6275:233402] operation0 after execute operation
2018-03-01 15:26:50.920427+0800 OperationQueues[6275:233400] operation1 after execute operation
*/

测试Queue的Suspended
// 通过下面的代码和打印可知: 
// 先把setMaxConcurrentOperationCount设置为1, 让操作一个一个的执行
// 当operation0执行完的时候,就设置暂停,但是operation1还是会执行,
// 但是operation2是不会执行,需要等待取消暂停,operation2才会继续执行
// 可以把Queue当成水管,Suspended属性当成水龙头,当水龙头关了, 后面的水就出不来(也就是不执行)

- (void)testQueueSuspended
{
    [_queue setMaxConcurrentOperationCount:1];

    NSBlockOperation *operation0 = [self createOperationRepeatCount:3 operationName:@"operation0"];
    NSBlockOperation *operation1 = [self createOperationRepeatCount:3 operationName:@"operation1"];
    NSBlockOperation *operation2 = [self createOperationRepeatCount:3 operationName:@"operation2"];
    
    __weak NSOperationQueue *weakQueue = _queue;
    [operation0 setCompletionBlock:^{
        
        NSLog(@"queue setSuspended YES");
        [weakQueue setSuspended:YES];
        
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            NSLog(@"after 1 second queue setSuspended NO");
            [weakQueue setSuspended:NO];
        });
    }];
    
    [_queue addOperation:operation0];
    [_queue addOperation:operation1];
    [_queue addOperation:operation2];
}

/*
2018-03-01 15:28:45.414606+0800 OperationQueues[6346:235874] operation0 before execute operation
2018-03-01 15:28:45.414772+0800 OperationQueues[6346:235874] operation0-1
2018-03-01 15:28:45.414883+0800 OperationQueues[6346:235874] operation0-2
2018-03-01 15:28:45.414992+0800 OperationQueues[6346:235874] operation0-3
2018-03-01 15:28:45.415142+0800 OperationQueues[6346:235874] operation0 after execute operation
2018-03-01 15:28:45.415287+0800 OperationQueues[6346:235875] queue setSuspended YES
2018-03-01 15:28:45.415313+0800 OperationQueues[6346:235872] operation1 before execute operation
2018-03-01 15:28:45.415432+0800 OperationQueues[6346:235872] operation1-1
2018-03-01 15:28:45.415526+0800 OperationQueues[6346:235872] operation1-2
2018-03-01 15:28:45.415613+0800 OperationQueues[6346:235872] operation1-3
2018-03-01 15:28:45.415714+0800 OperationQueues[6346:235872] operation1 after execute operation
2018-03-01 15:28:46.508915+0800 OperationQueues[6346:235837] after 1 second queue setSuspended NO
2018-03-01 15:28:46.509152+0800 OperationQueues[6346:235872] operation2 before execute operation
2018-03-01 15:28:46.509272+0800 OperationQueues[6346:235872] operation2-1
2018-03-01 15:28:46.509368+0800 OperationQueues[6346:235872] operation2-2
2018-03-01 15:28:46.509455+0800 OperationQueues[6346:235872] operation2-3
2018-03-01 15:28:46.509564+0800 OperationQueues[6346:235872] operation2 after execute operation
*/
添加顺序对操作执行顺序的影响
// 通过下面的代码和打印可知: 
// 任务是按照FIFO的方式进行的, 也就是从先执行operation0 -> operation1 -> operation2
// 但是并不意味着operation0里面的任务代码会先执行

// 举个例子: 
// 假如往queue中加入10个operation, 分别是operation0~operation9,依次顺序到队列
// 设置OperationCoun = 2, 这个时候先执行operation0和operation1
// 等opertaion0或者operation1其中一个执行完,接下来的一个执行的就是operation3,等下一个任务执行完,接下来执行的就是operation4

- (void)testQueueConcurrentOperationCount
{
    [_queue setMaxConcurrentOperationCount:2];
    
    NSBlockOperation *operation0 = [self createOperationRepeatCount:2 operationName:@"operation0"];
    NSBlockOperation *operation1 = [self createOperationRepeatCount:2 operationName:@"operation1"];
    NSBlockOperation *operation2 = [self createOperationRepeatCount:2 operationName:@"operation2"];
    
    [_queue addOperation:operation0];
    [_queue addOperation:operation1];
    [_queue addOperation:operation2];
}

/*
maxConcurrentOperationCount = 1 的情况

2018-03-01 15:31:58.001022+0800 OperationQueues[6485:240841] operation0 before execute operation
2018-03-01 15:31:58.001177+0800 OperationQueues[6485:240841] operation0-1
2018-03-01 15:31:58.001363+0800 OperationQueues[6485:240841] operation0-2
2018-03-01 15:31:58.001473+0800 OperationQueues[6485:240841] operation0 after execute operation
2018-03-01 15:31:58.001622+0800 OperationQueues[6485:240830] operation1 before execute operation
2018-03-01 15:31:58.002013+0800 OperationQueues[6485:240830] operation1-1
2018-03-01 15:31:58.002187+0800 OperationQueues[6485:240830] operation1-2
2018-03-01 15:31:58.002287+0800 OperationQueues[6485:240830] operation1 after execute operation
2018-03-01 15:31:58.002428+0800 OperationQueues[6485:240829] operation2 before execute operation
2018-03-01 15:31:58.002528+0800 OperationQueues[6485:240829] operation2-1
2018-03-01 15:31:58.002623+0800 OperationQueues[6485:240829] operation2-2
2018-03-01 15:31:58.002709+0800 OperationQueues[6485:240829] operation2 after execute operation
*/

/*
maxConcurrentOperationCount = 2 的情况

2018-03-01 15:33:10.568711+0800 OperationQueues[6522:242194] operation1 before execute operation
2018-03-01 15:33:10.568722+0800 OperationQueues[6522:242197] operation0 before execute operation
2018-03-01 15:33:10.568877+0800 OperationQueues[6522:242194] operation1-1
2018-03-01 15:33:10.568882+0800 OperationQueues[6522:242197] operation0-1
2018-03-01 15:33:10.568990+0800 OperationQueues[6522:242194] operation1-2
2018-03-01 15:33:10.568997+0800 OperationQueues[6522:242197] operation0-2
2018-03-01 15:33:10.569093+0800 OperationQueues[6522:242194] operation1 after execute operation
2018-03-01 15:33:10.569105+0800 OperationQueues[6522:242197] operation0 after execute operation
2018-03-01 15:33:10.569271+0800 OperationQueues[6522:242237] operation2 before execute operation
2018-03-01 15:33:10.569530+0800 OperationQueues[6522:242237] operation2-1
2018-03-01 15:33:10.569750+0800 OperationQueues[6522:242237] operation2-2
2018-03-01 15:33:10.569982+0800 OperationQueues[6522:242237] operation2 after execute operation
*/
Operation间的Dependency对操作执行顺序的影响

// 注意: 添加依赖需要在加入到队列之前,并且不要相互依赖,否则就造成死锁

// 通过下面的代码和打印可知:
// 添加了依赖的操作打破了上面(添加顺序对操作执行的影响)的规律,但是没有添加依赖的操作还是遵守的


- (void)testQueueOperationDependency
{
    [_queue setMaxConcurrentOperationCount:2];
    
    NSBlockOperation *operation0 = [self createOperationRepeatCount:2 operationName:@"operation0"];
    NSBlockOperation *operation1 = [self createOperationRepeatCount:2 operationName:@"operation1"];
    NSBlockOperation *operation2 = [self createOperationRepeatCount:2 operationName:@"operation2"];
    NSBlockOperation *operation3 = [self createOperationRepeatCount:2 operationName:@"operation3"];
    NSBlockOperation *operation4 = [self createOperationRepeatCount:2 operationName:@"operation4"];

    [operation0 addDependency:operation1];
    
    [_queue addOperation:operation0];
    [_queue addOperation:operation1];
    [_queue addOperation:operation2];
    [_queue addOperation:operation3];
    [_queue addOperation:operation4];
}

/*
2018-03-01 16:02:47.435604+0800 OperationQueues[7178:269608] operation2 before execute operation
2018-03-01 16:02:47.435608+0800 OperationQueues[7178:269616] operation1 before execute operation
2018-03-01 16:02:47.435778+0800 OperationQueues[7178:269608] operation2-1
2018-03-01 16:02:47.435782+0800 OperationQueues[7178:269616] operation1-1
2018-03-01 16:02:47.435881+0800 OperationQueues[7178:269608] operation2-2
2018-03-01 16:02:47.435918+0800 OperationQueues[7178:269616] operation1-2
2018-03-01 16:02:47.435997+0800 OperationQueues[7178:269608] operation2 after execute operation
2018-03-01 16:02:47.436063+0800 OperationQueues[7178:269616] operation1 after execute operation
2018-03-01 16:02:47.436164+0800 OperationQueues[7178:269606] operation3 before execute operation
2018-03-01 16:02:47.436196+0800 OperationQueues[7178:269608] operation0 before execute operation
2018-03-01 16:02:47.436314+0800 OperationQueues[7178:269606] operation3-1
2018-03-01 16:02:47.436816+0800 OperationQueues[7178:269608] operation0-1
2018-03-01 16:02:47.437014+0800 OperationQueues[7178:269606] operation3-2
2018-03-01 16:02:47.437182+0800 OperationQueues[7178:269608] operation0-2
2018-03-01 16:02:47.437341+0800 OperationQueues[7178:269606] operation3 after execute operation
2018-03-01 16:02:47.437520+0800 OperationQueues[7178:269608] operation0 after execute operation
2018-03-01 16:02:47.480951+0800 OperationQueues[7178:269606] operation4 before execute operation
2018-03-01 16:02:47.481082+0800 OperationQueues[7178:269606] operation4-1
2018-03-01 16:02:47.481191+0800 OperationQueues[7178:269606] operation4-2
2018-03-01 16:02:47.481290+0800 OperationQueues[7178:269606] operation4 after execute operation
*/
Operation间的Dependency
// 注意: 添加依赖需要在加入到队列之前,并且不要相互依赖,否则就造成死锁

// 通过下面的代码和打印可知:
// 就算设置MaxConcurrentOperationCount = 2,操作还是一个一个的执行
// 添加了依赖,顺序页修改了,请参照上面(Operation间的Dependency对操作执行顺序的影响)

- (void)testQueueOperationDependency
{
    [_queue setMaxConcurrentOperationCount:2];
    
    NSBlockOperation *operation0 = [self createOperationRepeatCount:2 operationName:@"operation0"];
    NSBlockOperation *operation1 = [self createOperationRepeatCount:2 operationName:@"operation1"];
    NSBlockOperation *operation2 = [self createOperationRepeatCount:2 operationName:@"operation2"];
    
    [operation1 addDependency:operation2];
    [operation0 addDependency:operation1];
    
    [_queue addOperation:operation0];
    [_queue addOperation:operation1];
    [_queue addOperation:operation2];
}

/*
2018-03-01 15:56:03.770607+0800 OperationQueues[7053:264284] operation2 before execute operation
2018-03-01 15:56:03.770749+0800 OperationQueues[7053:264284] operation2-1
2018-03-01 15:56:03.770840+0800 OperationQueues[7053:264284] operation2-2
2018-03-01 15:56:03.770960+0800 OperationQueues[7053:264284] operation2 after execute operation
2018-03-01 15:56:03.771122+0800 OperationQueues[7053:264295] operation1 before execute operation
2018-03-01 15:56:03.771228+0800 OperationQueues[7053:264295] operation1-1
2018-03-01 15:56:03.771336+0800 OperationQueues[7053:264295] operation1-2
2018-03-01 15:56:03.771458+0800 OperationQueues[7053:264295] operation1 after execute operation
2018-03-01 15:56:03.771578+0800 OperationQueues[7053:264295] operation0 before execute operation
2018-03-01 15:56:03.771681+0800 OperationQueues[7053:264295] operation0-1
2018-03-01 15:56:03.771785+0800 OperationQueues[7053:264295] operation0-2
2018-03-01 15:56:03.771880+0800 OperationQueues[7053:264295] operation0 after execute operation
*/

补充
  • queuePriority属性只是在运行的时候,只是会资源的分配,并不会影响到上面所说的顺序
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容