copy与mutablecopy使用对比

杂七杂八

copy和mutableCopy都是浅拷贝!!!
iOS中的copy,mutableCopy,深拷贝和浅拷贝
关于NSString的copy和mutableCopy
iOS中关键字copy与mutableCopy的详解,看我你就都懂了
copy和mutableCopy到底是浅拷贝还是深拷贝?
青玉伏案:Objective-C中的深拷贝和浅拷贝

Stackoverflow

Documents

目录

  • 一、copy、mutableCopy
  • 二、系统的非容器类对象:这里指的是NSString、NSNumber等对象。
  • 三、系统的容器类对象:指NSArray,NSSet,NSDictionary等。
  • 四、另一个角度来看
    • 4.1、浅拷贝
    • 4.2、单层深copy
    • 4.3、双层深拷贝
    • 4.4、完全深拷贝
    • 4.5、自定义类对象之间的深浅拷贝问题
  • 五、@property中的copy关键字
  • 六、copy和block

一、copy、mutableCopy

  • copy:不可变拷贝,遵循NSCopying协议,需要对应实现copyWithZone方法;
  • mutableCopy:可变拷贝,遵循NSMutableCopying协议,需要对应实现mutableCopyWithZone:方法;
NSCopying、NSMutableCopying

二、系统的非容器类对象:这里指的是NSString、NSNumber等对象。

 // const是常量字符串,存在常量区 
 // constStr指针存在栈区, 指针指向常量区 
NSString * constStr = @"const";
NSString * constStrCopy = [constStr copy];
NSMutableString * constStrMutableCopy = [constStr mutableCopy];
NSLog(@"constStr = %p = %@",constStr,constStr);
NSLog(@"constStrCopy = %p = %@",constStrCopy,constStrCopy);
NSLog(@"constStrMutableCopy = %p = %@",constStrMutableCopy,constStrMutableCopy);

// originStr在栈中,指向堆区的地址 
NSString * originStr = [NSString stringWithFormat:@"origin"];
NSString * originStrCopy = [originStr copy];
NSMutableString * originStrMutableCopy = [originStr mutableCopy];
NSLog(@"originStr = %p = %@",originStr,originStr);
NSLog(@"originStrCopy = %p = %@",originStrCopy,originStrCopy);
NSLog(@"originStrMutableCopy = %p = %@",originStrMutableCopy,originStrMutableCopy);
    
NSMutableString *mutableOriginStr = [NSMutableString stringWithFormat:@"mutableOrigin"];
NSMutableString *mutableOriginStrCopy = [mutableOriginStr copy];
NSMutableString *mutableOriginStrMutableCopy = [mutableOriginStr mutableCopy];
NSLog(@"mutableOriginStr = %p = %@",mutableOriginStr,mutableOriginStr);
NSLog(@"mutableOriginStrCopy = %p = %@",mutableOriginStrCopy,mutableOriginStrCopy);
NSLog(@"mutableOriginStrMutableCopy = %p = %@",mutableOriginStrMutableCopy,mutableOriginStrMutableCopy);


// 返回值测试对象是否为mutable
[constStrMutableCopy appendString:@"const"];
[originStrMutableCopy appendString:@"origin"];
#pragma warnning - ERROR
[mutableOriginStrCopy appendString:@"mm"];   // ERROR

输出

constStr = 0x109a32558
constStrCopy = 0x109a32558
constStrMutableCopy = 0x60000307ad00

originStr = 0xc117d077374719f4
originStrCopy = 0xc117d077374719f4
originStrMutableCopy = 0x60000307ac40

mutableOriginStr = 0x60000307ac70
mutableOriginStrCopy = 0x600003e7d960
mutableOriginStrMutableCopy = 0x60000307adc0
总结图
总结:对于系统的非容器类对象,我们可以认为,如果对不可变对象复制,copy是指针复制(浅拷贝)和mutableCopy就是对象复制(深拷贝)。如果是对可变对象复制,都是深拷贝,但是copy返回的对象是不可变的。
copy返回的对象是不可变的,mutableCopy是可变的。
NSString内存分配结论:

iOS的关于堆区和栈区

@"" 和 initWithString:方法生成的字符串分配在常量区,系统自动管理内存;

initWithFormat:和 stringWithFormat: 方法生成的字符串分配在堆区,autorelease;

三、系统的容器类对象:指NSArray,NSSet,NSDictionary等。

copy和mutableCopy都是浅拷贝!!!
iOS深浅拷贝(纠错)

浅拷贝与深拷贝
观点一: 所有系统容器类的copy或mutableCopy方法,都是浅拷贝!!!

验证一:NSArray进行copy与mutableCopy,并改变NSArray内元素值,查看拷贝对象内部值的变化。

    NSArray * arr = [NSArray arrayWithObjects:
                     [NSMutableString stringWithString:@"one"],
                     [NSMutableString stringWithString:@"two"],
                     [NSMutableString stringWithString:@"three"],
                     [NSMutableString stringWithString:@"four"],nil];
    NSArray * arrcopy = [arr copy];
    NSMutableArray * arrmutablecopy = [arr mutableCopy];
    NSLog(@"arr = %p = %p",arr,arr[0]);
    NSLog(@"arrcopy = %p = %p",arrcopy,arrcopy[0]);
    NSLog(@"arrmutablecopy = %p = %p",arrmutablecopy,arrmutablecopy[0]);

    NSMutableString * mStr;
    mStr = arr[0];
    [mStr appendString:@"--array"];

    NSLog(@"改变内部元素后 arr:%@ = %p",arr,arr[0]);
    NSLog(@"改变内部元素后 arrcopy:%@ = %p",arrcopy,arrcopy[0]);
    NSLog(@"改变内部元素后 arrmutablecopy:%@ = %p",arrmutablecopy,arrmutablecopy[0]);

输出

arr = 0x604000446390 = 0x604000445d90
arrcopy = 0x604000446390 = 0x604000445d90
arrmutablecopy = 0x604000445520 = 0x604000445d90
改变内部元素后 arr:(
    "one--array",
    two,
    three,
    four
) = 0x604000445d90
改变内部元素后 arrcopy:(
    "one--array",
    two,
    three,
    four
) = 0x604000445d90
改变内部元素后 arrmutablecopy:(
    "one--array",
    two,
    three,
    four
) = 0x604000445d90

验证二:NSMutableArray进行copy与mutableCopy,并改变NSMutableArray内元素值,查看拷贝对象内部值的变化。

    NSMutableArray *mutableArr = [NSMutableArray arrayWithObjects:
                                  [NSMutableString stringWithString:@"abc"],
                                  [NSMutableString stringWithString:@"def"],
                                  [NSMutableString stringWithString:@"ghi"],
                                  [NSMutableString stringWithString:@"jkl"], nil];
    NSArray * mutableArrcopy = [mutableArr copy];
    NSMutableArray * mutableArrmutablecopy = [mutableArr mutableCopy];
    NSLog(@"mutableArr = %p = %p",mutableArr,mutableArr[0]);
    NSLog(@"mutableArrcopy = %p = %p",mutableArrcopy,mutableArrcopy[0]);
    NSLog(@"mutableArrmutablecopy = %p = %p",mutableArrmutablecopy,mutableArrmutablecopy[0]);
    
    NSMutableString * mStr1;
    mStr1 = mutableArr[0];
    [mStr1 appendString:@"--mutablearray"];
    
    [mutableArrmutablecopy addObject:@"FFF"];
    
    NSLog(@"改变内部元素后 mutableArr:%@ = %p",mutableArr,mutableArr[0]);
    NSLog(@"改变内部元素后 mutableArrcopy:%@ = %p",mutableArrcopy,mutableArrcopy[0]);
    NSLog(@"改变内部元素后 mutableArrmutablecopy:%@ = %p",mutableArrmutablecopy,mutableArrmutablecopy[0]);

输出

mutableArr = 0x604000452b10 = 0x604000452a80
mutableArrcopy = 0x604000452b70 = 0x604000452a80
mutableArrmutablecopy = 0x60400025fe90 = 0x604000452a80
改变内部元素后 mutableArr:(
    "abc--mutablearray",
    def,
    ghi,
    jkl
) = 0x604000452a80
改变内部元素后 mutableArrcopy:(
    "abc--mutablearray",
    def,
    ghi,
    jkl
) = 0x604000452a80
改变内部元素后 mutableArrmutablecopy:(
    "abc--mutablearray",
    def,
    ghi,
    jkl,
    FFF
) = 0x604000452a80
总结图
总结:copy操作返回的必然是一个不可变对象,无论源对象是可变对象还是不可变对象。如果源对象是一个不可变对象,那么它们(源对象和新生成的对象)指向同一个对象,如果源对象是可变对象,它们指向不同对象。
mutableCopy返回的必然是一个可变对象,无论源对象是可变对象还是不可变对象,它们(源对象和新生成的对象)仍指向不同地址,是两个对象。
特别注意的是:对于集合类的可变对象来说,深拷贝并非严格意义上的深复制,只能算是单层深复制,即虽然新开辟了内存地址,但是存放在内存上的值(也就是数组里的元素仍然之原数组元素值,并没有另外复制一份),这就叫做单层深复制。

四、另一个角度来看

4.1、浅拷贝

iOS 图文并茂的带你了解深拷贝与浅拷贝
Objective-C copy,看我就够了

NSArray *arr = [NSArray arrayWithObjects:@"1", nil];
NSArray *copyArr = [arr copy];
NSLog(@"%p", arr);
NSLog(@"%p", copyArr);

输出:浅拷贝

2018-10-24 10:00:17.256591+0800 TodayNews[2229:70407] 0x60000043d3c0
2018-10-24 10:00:17.256705+0800 TodayNews[2229:70407] 0x60000043d3c0

4.2、单层深copy

这里的单层指的是完成了NSArray对象的深copy,而未对其容器内对象进行处理。

NSArray *arr = [NSArray arrayWithObjects:@"1", nil];
NSArray *copyArr = [arr mutableCopy];
    
NSLog(@"%p", arr);
NSLog(@"%p", copyArr);
    
// 打印arr、copyArr内部元素进行对比
NSLog(@"%p", arr[0]);
NSLog(@"%p", copyArr[0]);

输出:

2018-10-24 10:06:10.985032+0800 TodayNews[2330:73697] 0x60000043a200
2018-10-24 10:06:10.985224+0800 TodayNews[2330:73697] 0x600000642a60
2018-10-24 10:06:10.985347+0800 TodayNews[2330:73697] 0x102bf00d8
2018-10-24 10:06:10.985438+0800 TodayNews[2330:73697] 0x102bf00d8

4.3、双层深拷贝

    // 随意创建一个NSMutableString对象
    NSMutableString *mutableString = [NSMutableString stringWithString:@"1"];
    // 随意创建一个包涵NSMutableString的NSMutableArray对象
    NSMutableString *mutalbeString1 = [NSMutableString stringWithString:@"1"];
    NSMutableArray *mutableArr = [NSMutableArray arrayWithObjects:mutalbeString1, nil];
    // 将mutableString和mutableArr放入一个新的NSArray中
    NSArray *testArr = [NSArray arrayWithObjects:mutableString, mutableArr, nil];
    // 通过官方文档提供的方式创建copy
    NSArray *testArrCopy = [[NSArray alloc] initWithArray:testArr copyItems:YES];
    
    // testArr和testArrCopy指针对比
    NSLog(@"%p", testArr);
    NSLog(@"%p", testArrCopy);
    
    // testArr和testArrCopy中元素指针对比
    // mutableString对比
    NSLog(@"%p", testArr[0]);
    NSLog(@"%p", testArrCopy[0]);
    // mutableArr对比
    NSLog(@"%p", testArr[1]);
    NSLog(@"%p", testArrCopy[1]);
    
    // mutableArr中的元素对比,即mutalbeString1对比
    NSLog(@"%p", testArr[1][0]);
    NSLog(@"%p", testArrCopy[1][0]);

输出

2018-10-24 10:10:20.994041+0800 TodayNews[2442:76854] 0x600000426500
2018-10-24 10:10:20.994165+0800 TodayNews[2442:76854] 0x6000004264c0
2018-10-24 10:10:20.994280+0800 TodayNews[2442:76854] 0x600000652870
2018-10-24 10:10:20.994421+0800 TodayNews[2442:76854] 0xa000000000000311
2018-10-24 10:10:20.994512+0800 TodayNews[2442:76854] 0x600000652960
2018-10-24 10:10:20.994599+0800 TodayNews[2442:76854] 0x600000005ef0
2018-10-24 10:10:20.994701+0800 TodayNews[2442:76854] 0x6000006528a0
2018-10-24 10:10:20.994993+0800 TodayNews[2442:76854] 0x6000006528a0

4.4、完全深拷贝

方式一:如果想完美的解决NSArray嵌套NSArray这种情形,可以使用归档、解档的方式。

// 随意创建一个NSMutableString对象
NSMutableString *mutableString = [NSMutableString stringWithString:@"1"];
// 随意创建一个包涵NSMutableString的NSMutableArray对象
NSMutableString *mutalbeString1 = [NSMutableString stringWithString:@"1"];
NSMutableArray *mutableArr = [NSMutableArray arrayWithObjects:mutalbeString1, nil];
// 将mutableString和mutableArr放入一个新的NSArray中
NSArray *testArr = [NSArray arrayWithObjects:mutableString, mutableArr, nil];
// 通过归档、解档方式创建copy
NSArray *testArrCopy = [NSKeyedUnarchiver unarchiveObjectWithData:
                            [NSKeyedArchiver archivedDataWithRootObject:testArr]];;
    
// testArr和testArrCopy指针对比
NSLog(@"%p", testArr);
NSLog(@"%p", testArrCopy);
    
// testArr和testArrCopy中元素指针对比
// mutableString对比
NSLog(@"%p", testArr[0]);
NSLog(@"%p", testArrCopy[0]);
// mutableArr对比
NSLog(@"%p", testArr[1]);
NSLog(@"%p", testArrCopy[1]);
    
// mutableArr中的元素对比,即mutalbeString1对比
NSLog(@"%p", testArr[1][0]);
NSLog(@"%p", testArrCopy[1][0]);

输出

2018-10-24 10:15:11.448311+0800 TodayNews[2549:80583] 0x600000430640
2018-10-24 10:15:11.448435+0800 TodayNews[2549:80583] 0x6000004309e0
2018-10-24 10:15:11.448528+0800 TodayNews[2549:80583] 0x60000045e2a0
2018-10-24 10:15:11.448684+0800 TodayNews[2549:80583] 0x60000045e3c0
2018-10-24 10:15:11.448773+0800 TodayNews[2549:80583] 0x60000045d0d0
2018-10-24 10:15:11.448925+0800 TodayNews[2549:80583] 0x60000045e450
2018-10-24 10:15:11.449012+0800 TodayNews[2549:80583] 0x60000045e1b0
2018-10-24 10:15:11.449161+0800 TodayNews[2549:80583] 0x60000045e7e0

方式二:- (instancetype)initWithArray:(NSArray<ObjectType> *)array copyItems:(BOOL)flag;

NSMutableArray *marry1 = [[NSMutableArray alloc] init];
    
NSMutableString *mstr1 = [[NSMutableString alloc]initWithString:@"value1"];
NSMutableString *mstr2 = [[NSMutableString alloc]initWithString:@"value2"];
    
[marry1 addObject:mstr1];
[marry1 addObject:mstr2];
    
NSArray *marray2 = [[NSArray alloc] initWithArray:marry1 copyItems:YES];
    
NSLog(@"marry1:%p - %@ \r\n",marry1,marry1);
NSLog(@"marry2:%p - %@ \r\n",marray2,marray2);
NSLog(@"数组元素地址:value1:%p - value2:%p \r\n",marry1[0],marry1[1]);
NSLog(@"数组元素地址:value1:%p - value2:%p \r\n",marray2[0],marray2[1]);

4.5、自定义类对象之间的深浅拷贝问题

在Objective-C中并不是所有的类都支持拷贝;只有遵循NSCopying协议的类,才支持copy拷贝,只有遵循NSMutableCopying协议的类,才支持mutableCopy拷贝。如果没有遵循拷贝协议,拷贝时会出错。

如果我们想再我们自定义的类中支持copy和mutableCopy那么我们就需要使我们定义的类遵循NSCopying和NSMutableCopying协议,代码如下:

@interface Study_CustomObject_copy_mutableCopy : NSObject <NSCopying, NSMutableCopying>  // 协议
@property (copy,nonatomic) NSString *name;
@property (copy,nonatomic) NSString *age;
@end

@implementation Study_CustomObject_copy_mutableCopy
- (id)copyWithZone:(NSZone *)zone
{
    Study_CustomObject_copy_mutableCopy *customobject = [[Study_CustomObject_copy_mutableCopy allocWithZone:zone] init];
    customobject.age = self.age;
    customobject.name = self.name;
    return customobject;
}
- (id)mutableCopyWithZone:(NSZone *)zone {
    Study_CustomObject_copy_mutableCopy *customobject = [[Study_CustomObject_copy_mutableCopy allocWithZone:zone] init];
    customobject.age = self.age;
    customobject.name = self.name;
    return customobject;
}
@end
调用 、 输出
// 调用
{
    Study_CustomObject_copy_mutableCopy *object = [[Study_CustomObject_copy_mutableCopy alloc]init];
    object.age = @"99";
    object.name = @"lionsom";
    
    Study_CustomObject_copy_mutableCopy *objectCopy = [object copy];
    Study_CustomObject_copy_mutableCopy *objectMutableCopy = [object mutableCopy];
    
    NSLog(@"object === %p , name === %p , age === %p",object, object.name, object.age);
    NSLog(@"objectCopy === %p , name === %p , age === %p",objectCopy, objectCopy.name, objectCopy.age);
    NSLog(@"objectMutableCopy === %p , name === %p , age === %p",objectMutableCopy, objectMutableCopy.name, objectMutableCopy.age);
}

// 输出
object === 0x60400023d7a0 , name === 0x100744ed8 , age === 0x100744eb8
objectCopy === 0x60400023bd80 , name === 0x100744ed8 , age === 0x100744eb8
objectMutableCopy === 0x60400023d180 , name === 0x100744ed8 , age === 0x100744eb8

五、@property中的copy关键字

iOS内存管理(6)--NSArray与NSMutableArray用copy修饰还是strong
OC的深拷贝与浅拷贝--NSArray与NSMutableArray应该使用copy还是strong?

  • NSString
    • strong 关键词:两个string指向相同的内存地址,修改一个,另一个也会改变;
    • (推荐)copy 关键词:两个string指向不同的内存地址,互不影响;
  • NSMutableString
    • (推荐)strong 关键词:两个string指向相同的内存地址,修改一个,另一个也会改变;
    • (崩溃)copy 关键词:copy之后,就把变量string变成了不可变的NSString类型,对不可变的NSString使用了NSMutableString的方法appendString。
  • NSArray
    • strong 关键词:两个string指向相同的内存地址,修改一个,另一个也会改变;
    • (推荐)copy 关键词:此时内存地址都是不同的,修改一个,互不影响;
  • NSMutableArray
    • (推荐)strong 关键词:两个string指向相同的内存地址,修改一个,另一个也会改变;
    • (崩溃)copy 关键词:copy之后,就把变量array变成了不可变的NSArray类型,对不可变的NSArray使用了NSMutableArray的方法addObject。

当修饰可变类型的属性时,如NSMutableArray、NSMutableDictionary、NSMutableString,用strong。

当修饰不可变类型的属性时,如NSArray、NSDictionary、NSString,用copy。

六、copy和block

block使用copy原理

简单来说,block就像一个函数指针,指向我们要使用的函数。

就和函数调用一样的,不管你在哪里写了这个block,只要你把它放在了内存中(通过调用存在这个block的方法或者是函数),不管放在栈中还是在堆中,还是在静态区。只要他没有被销毁,你都可以通过你声明的block调用他。

说到在类中声明一个block为什么要用copy修饰的话,那就要先说block的三种类型。

1._NSConcreteGlobalBlock,全局的静态block,不会访问外部的变量。就是说如果你的block没有调用其他的外部变量,那你的block类型就是这种。例如:你仅仅在你的block里面写一个NSLog("hello world");

2._NSConcreteStackBlock 保存在栈中的 block,当函数返回时会被销毁。这个block就是你声明的时候不用c opy修饰,并且你的block访问了外部变量。

3._NSConcreteMallocBlock 保存在堆中的 block,当引用计数为 0 时会被销毁。好了,这个就是今天的主角 ,用copy修饰的block。

我们知道,函数的声明周期是随着函数调用的结束就终止了。我们的block是写在函数中的。

如果是全局静态block的话,他直到程序结束的时候,才会被被释放。但是我们实际操作中基本上不会使用到不访问外部变量的block。【但是在测试三种区别的时候,因为没有很好的理解这种block,(用没有copy修饰和没有访问外部变量的block)试了好多次,以为是放在静态区里面的block没有随函数结束被释放。这是个小坑】

如果是保存在栈中的block,他会随着函数调用结束被销毁。从而导致我们在执行一个包含block的函数之后,就无法再访问这个block。因为(函数结束,函数栈就销毁了,存在函数里面的block也就没有了),我们再使用block时,就会产生空指针异常。

如果是堆中的block,也就是copy修饰的block。他的生命 周期就是随着对象的销毁而结束的。只要对象不销毁,我们就可以调用的到在堆中的block。

这就是为什么我们要用copy来修饰block。因为不用copy修饰的访问外部变量的block,只在他所在的函数被调用的那一瞬间可以使用。之后就消失了。

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

推荐阅读更多精彩内容