Qt隐式共享理解

 

Qt的隐式共享, 简单的说就是:赋值时不直接拷贝而是用共享指针使用同一块数据, 当其中一个被修改时,被修改的值才会拷贝一份出来修改.
如:
QFont f1;
QFont f2 = f1, // 并没有发生值拷贝(严格意义上来说拷贝了指针),f2和f1指向同一数据块.
f2.setPixelSize(5) // 此时先拷贝一份f1值给f2,然后再修改f2的PixelSize();
所以在Qt中使用共享类函数传参,返回值都可以不用引用, 也不影响效率.因为共享的是同一数据块.
而使用Qt提供的容器存储这些类也不会发生值拷贝, 这是很高效的.

下面我们来简单的看看Qt是怎么实现隐式共享的(转他人:(http://blog.csdn.net/zhu_xz/article/details/6061201)

 

为了最大化资源使用,和最小化数据拷贝,Qt在很多类中用到了隐式数据共享,以便数据仅在被写入时才被拷贝。该机制也被称为flyweight模式

 

让我们以QByteArray为例,看看其是如何实现的。其内部使用一个名为Data的私有结构体来追踪共享的数据:

  1. struct Data {  
  2.   QBasicAtomicInt ref; // 引用计数器,对其的操作是原子的  
  3.   int alloc; // 已分配的空间大小  
  4.   int size; // 数据的实际大小  
  5.   char *data; // 指向数据的指针  
  6.   char array[1]; // 数据有可能存于此位置  
  7. };  

 

 

这里,如果数据保存在其他位置,则需要用到data来指向实际的数据位置;如果保存在自身,则是array指向的位置。当对象被拷贝时(比如通过赋值运算符),则仅仅拷贝指针,而不拷贝数据本身:

  1. QByteArray &QByteArray::operator=(const QByteArray & other)  
  2. {  
  3.   // 增加要使用的共享数据的引用计数器的值  
  4.   other.d->ref.ref();  
  5.   // 减少当前共享数据的引用计数器的值  
  6.   if (!d->ref.deref())  
  7.     qFree(d);  
  8.   // 指向要使用的共享数据  
  9.   d = other.d;  
  10.   return *this;  
  11. }  

 

 

另一方面,如果共享的数据要被修改(比如通过resize()函数),则会自动拷贝之:

  1. void QByteArray::resize(int size)  
  2. {  
  3.   if (size <= 0) {  
  4.     // 如果目标大小不为正,则指向一个空的数据块  
  5.     Data *x = &shared_empty;  
  6.     x->ref.ref();  
  7.     if (!d->ref.deref())  
  8.       qFree(d);  
  9.     d = x;  
  10.   } else if (d == &shared_null) {  
  11.     // 如果当前是一个null块,则直接创建一个新的共享数据块  
  12.     Data *x = static_cast<data *>(qMalloc(sizeof(Data)+size));  
  13.     Q_CHECK_PTR(x);  
  14.     x->ref = 1;  
  15.     x->alloc = x->size = size;  
  16.     x->data = x->array;  
  17.     x->array[size] = '/0';  
  18.     (void) d->ref.deref();  
  19.     d = x;  
  20.   } else {  
  21.     // 如果有其他对象也在使用该共享数据,或者当前分配的空间过大或过小  
  22.     // 则重新分配空间,并拷贝数据  
  23.     // 注意:该操作在共享的数据块较大时可能会消耗一定的时间  
  24.     if (d->ref != 1 || size > d->alloc || (size < d->size && size < d->alloc >> 1))  
  25.       realloc(qAllocMore(size, sizeof(Data)));  
  26.     if (d->alloc >= size) {  
  27.       d->size = size;  
  28.       if (d->data == d->array) {  
  29.         d->array[size] = '/0';  
  30.       }  
  31.     }  
  32.   }  
  33. }  

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值