日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

c++ primer 5th第13章拷贝控制知识点和自编习题答案

發(fā)布時(shí)間:2024/4/18 c/c++ 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 c++ primer 5th第13章拷贝控制知识点和自编习题答案 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

????????首先,先給大家提個(gè)醒。在網(wǎng)上的隨書源代碼里關(guān)于hasptr類的類指針版本編寫的移動構(gòu)造函數(shù)、移動賦值運(yùn)算符、和析構(gòu)函數(shù)部分是有錯(cuò)誤的。大家可以把hasptr累指針版本(里面帶移動構(gòu)造函數(shù)和移動賦值運(yùn)算符部分)可以運(yùn)行下,g++編譯環(huán)境發(fā)現(xiàn)core dumped錯(cuò)誤。原因是:在代碼中途有的對象的use和ps所指的內(nèi)存已經(jīng)被釋放了,所以在析構(gòu)函數(shù)中再次直接釋放use和ps就是錯(cuò)的了。正確應(yīng)該在中途每個(gè)地方use和ps被接管后都置為nullptr,然后在析構(gòu)函數(shù)中必須首先判斷use和ps是否是nullptr,然后再釋放。

************************************************************************************************************************************************

對nullptr指針,執(zhí)行delete操作會引發(fā)錯(cuò)誤,g++提示:core dumped

************************************************************************************************************************************************

習(xí)題13.1拷貝、賦值和銷毀

1.拷貝初始化何時(shí)發(fā)生:

2.拷貝初始化的限制-----不能隱式使用explicit的構(gòu)造函數(shù)初始化一個(gè)類類型。?

?3.編譯器可以繞過拷貝構(gòu)造函數(shù)。

?習(xí)題13.1.1

練習(xí)13

答:拷貝構(gòu)造函數(shù)的名字和類名本身相同,參數(shù)類型是自身類型的引用,沒有返回值。像這樣的構(gòu)造函數(shù)叫做拷貝構(gòu)造函數(shù)。在用同類型的一個(gè)對象初始化另一個(gè)對象時(shí)候使用。

答:?在函數(shù)調(diào)用中,如果函數(shù)參數(shù)是非引用類型,要拷貝初始化,所以構(gòu)造函數(shù)的參數(shù)必須是引用。如果構(gòu)造函數(shù)自己的參數(shù)類型也是非引用,那么就會在拷貝參數(shù)的時(shí)候調(diào)用本身,如此反復(fù),則進(jìn)入無限循環(huán)。

練習(xí)13.2

答:StrBlob只有一個(gè)數(shù)據(jù)成員,類型是shared_ptr<vector<string>>,如果拷貝一個(gè)StrBlob,那么會拷貝shard_ptr對象,原來的shared_ptr的引用計(jì)數(shù)會增加1。新對象和原來對象會共享內(nèi)存。

?答:

第1處:local = arg。因?yàn)橛猛愋偷牧硪粋€(gè)對象初始化本對象。

第二處:*heap = local;同理

第三處:Point pa[4] = {local,*heap};

當(dāng)然,這是這段代碼所發(fā)生的拷貝初始化。

?答:

class HasPtr{public:HasPtr(const string & s = string()):ps(new string(s)),i(0){}HasPtr(const HasPtr&);private:string *ps;int i; }; HasPtr::HasPtr(const HasPtr & p):ps(new string(*p.ps)),i(p.i){}

13.1.2練習(xí)

練習(xí)13.6

答:

拷貝賦值運(yùn)算符是定義類類型如何用另一個(gè)同類型對象給本對象賦值的運(yùn)算符。有以下幾個(gè)特點(diǎn):

返回類型是自身類型的引用、函數(shù)名字是operator=,參數(shù)類型是自身引用。

合成的賦值運(yùn)算符完成用本類類型對象給自身類型對象的賦值任務(wù)。

如果編譯器沒有定義拷貝賦值運(yùn)算符,那么編譯器會合成一個(gè),叫做拷貝賦值運(yùn)算符。

練習(xí)13.7

答:

當(dāng)用一個(gè)StrBlob對象給另一個(gè)StrBlob對象賦值的時(shí)候,shared_ptr對象對調(diào)用自己的拷貝賦值運(yùn)算符,被賦值shared_ptr對象自身的引用計(jì)數(shù)會增加1,左側(cè)StrBlob對象的shared_ptr對象的引用計(jì)數(shù)減1。

????????????類StrBlobPtr對象有一個(gè)shared_ptr對象,一個(gè)week_ptr<vector<string>>對象和一個(gè)size_t對象。賦值的時(shí)候,右側(cè)shared_ptr對象的引用計(jì)數(shù)加1,左側(cè)的shared_ptr的引用計(jì)數(shù)減1,左側(cè)對象的shared_ptr對象指向右側(cè)的shared_ptr指向的內(nèi)存。對于weak_ptr對象,賦值后左側(cè)的weak_ptr的內(nèi)容換成右側(cè)類的weak_ptr的內(nèi)容。其它不變。curr的值也是賦值。

13.8

答:

class HasPtr{public:HasPtr(const string & s = string()):ps(new string(s)),i(0){}HasPtr(const HasPtr&);HasPtr& operator=(const HasPtr &);~HasPtr(){delete ps;}private:string *ps;int i; }; HasPtr::HasPtr(const HasPtr & p):ps(new string(*p.ps)),i(p.i){} HasPtr& HasPtr::operator=(const HasPtr &p){ //這里一定要爭取處理自賦值情況,也就是不能再拷貝之前釋放掉參數(shù)的成員ps指向的內(nèi)存 string *new_ps = new string(*p.ps); delete ps; ps = new_ps; i = p.i; return *this; }

13.1.3析構(gòu)函數(shù)

析構(gòu)函數(shù)執(zhí)行與構(gòu)造函數(shù)相反的工作:構(gòu)造函數(shù)初始化對象的非static數(shù)據(jù)成員,還可能做一些其它工作;析構(gòu)函數(shù)釋放對象使用的資源,并銷毀對象的非static數(shù)據(jù)成員。

析構(gòu)函數(shù)是類的一個(gè)成員函數(shù),名字由波浪號接類名構(gòu)成。它沒有返回值,也不接受參數(shù)。析構(gòu)函數(shù)不接受參數(shù),所以不可以重載,一個(gè)類只有一個(gè)唯一的析構(gòu)函數(shù)。

析構(gòu)函數(shù)完成什么工作

首先執(zhí)行函數(shù)體,然后銷毀成員,成員按照初始化的順序的逆序銷毀。

在一個(gè)析構(gòu)函數(shù)中,不存在類似構(gòu)造函數(shù)中初始化列表的東西來控制成員如何銷毀,析構(gòu)部分是隱式的。成員銷毀時(shí)發(fā)生什么完全依賴于成員的類型。銷毀類類型成員需要執(zhí)行成員自己的析構(gòu)函數(shù)。內(nèi)置類型沒有析構(gòu)函數(shù),因此銷毀內(nèi)置類型成員什么也不需要做。

隱式銷毀一個(gè)內(nèi)置指針類型的成員不會delete它所指向的對象。

什么時(shí)候調(diào)用析構(gòu)函數(shù)

無論何時(shí)一個(gè)對象被銷毀,就會自動調(diào)用其析構(gòu)函數(shù):

如下面的例子:?

?p是一個(gè)內(nèi)置指針,塊末尾調(diào)用delete,p指向的內(nèi)存釋放。

局部對象p2的引用計(jì)數(shù)減1,如果引用計(jì)數(shù)變?yōu)?,那么釋放對象。

item,vec,塊結(jié)束后自動銷毀,釋放資源。

習(xí)題13.1.3節(jié)練習(xí)

練習(xí)13.9

答:

13.9

析構(gòu)函數(shù)的名字是波浪號加類名,參數(shù)列表為空的一個(gè)函數(shù)。和析構(gòu)函數(shù)的功能相反,構(gòu)造函數(shù)初始化對象,析構(gòu)函數(shù)負(fù)責(zé)銷毀對象,釋放對象占用的資源。

13.10

當(dāng)一個(gè)StrBlob銷毀時(shí)候,shared_ptr類會自動地釋放對象所占用的資源,銷毀對象。

StrBlobPtr對象銷毀的時(shí)候,weak_ptr對象會銷毀自己的對象。因?yàn)槭且环N弱引用,所以對原來的智能指針沒有任何影響。其它內(nèi)置類型的成員,則自動銷毀。

13.11

class HasPtr{public:HasPtr(const string & s = string()):ps(new string(s)),i(0){}HasPtr(const HasPtr&);~HasPtr(){delete ps;}private:string *ps;int i; }; HasPtr::HasPtr(const HasPtr & p):ps(new string(*p.ps)),i(p.i){}

13.12

在下面的代碼片段中會發(fā)生幾次析構(gòu)函數(shù)的調(diào)用?

是3次,函數(shù)執(zhí)行完畢后銷毀局部對象accum,item1,item2共3個(gè)局部變量。

13.13

#include <memory> using namespace std; struct X{ X(int c):a(c){cout << "X(INT C)" << endl;} X(){cout << "X()" << endl;} X(const X&){cout << "X(const &X)" << endl;} X& operator=(const X&){cout << "X&operator=(const &X)" << endl;return *this;} ~X(){cout << "~X()" << endl;}private: int a = 0; int b; };int main() { //直接初始化 X a; X b(1234); X c(a); c = a;return 0; } X() X(INT C) X(const &X) X&operator=(const &X) ~X() ~X() ~X()

13.1.4三/五法則

一、需要析構(gòu)函數(shù)的類也需要拷貝和賦值操作。

需要拷貝操作的類也需要賦值操作,反之亦然。

?

?

?

注:我們定義的析構(gòu)函數(shù)會釋放動態(tài)內(nèi)存,如不定義自己的析構(gòu)函數(shù),合成的析構(gòu)函數(shù)不會自動釋放動態(tài)內(nèi)存。

?

?

?

????????這記錄了需要拷貝構(gòu)造函數(shù)和拷貝賦值運(yùn)算符但是不需要析構(gòu)函數(shù)的例子。具體程序如下。

練習(xí)題答案

練習(xí)13.1.4

答:這是一個(gè)典型的應(yīng)該定義拷貝控制成場合。如果不定義拷貝構(gòu)造函數(shù)和拷貝賦值運(yùn)算符,依賴程序合成的版本,則在拷貝構(gòu)造和賦值時(shí),會簡單賦值數(shù)據(jù)成員。對本問題來說,序簡單賦值給新對象。

? ? ? ? 因此,代碼中對a、b、c三個(gè)對象調(diào)用函數(shù),會輸出三個(gè)相同的序號-————合成拷貝構(gòu)造函數(shù)被調(diào)用時(shí)簡單復(fù)制序號,使得三個(gè)對象具有相同的序號。

結(jié)論:合成的默認(rèn)拷貝構(gòu)造函數(shù)和拷貝賦值運(yùn)算符會直接拷貝所有的成員,不會成員的值。

? 練習(xí)13.15

答:在定義的時(shí)候,a、b、c獲得的mysn的值分別是0、1、2,但調(diào)用f的時(shí)候,還會進(jìn)行三次拷貝,第一次拷貝給函數(shù)f的變量,調(diào)用拷貝構(gòu)造函數(shù)mysn的值繼續(xù)增加1,變?yōu)?,以此類推,調(diào)用第二次mysn變?yōu)?、5,所以輸出結(jié)果是3、4、5。

當(dāng)然這假定第一個(gè)元素給值0的情況。

練習(xí)13.16.

答:如果f的參數(shù)是const Numbered &,那么輸出結(jié)果變?yōu)?,1,原因是如果參數(shù)類型傳遞的是const Numbered &,而不是類型本身,因?yàn)閭鬟f的是引用類型,所以并不會拷貝對象,所以輸出結(jié)果是0、1、2。

13.17。

13.14問題中的程序?qū)崿F(xiàn)如下:

#include <iostream> #include <memory> using namespace std; class Numbered; void f(Numbered); class Numbered{ friend void f(Numbered ); private:static int*ptr;int *p;int mysn; public:Numbered():p(ptr),mysn(*p){++ *p;}void delete_(){delete ptr;}};int * Numbered::ptr = new int(0); void f(Numbered s){cout << s.mysn << endl;}int main() { Numbered a,b = a ,c = b; f(a); f(b); f(c);//調(diào)用一個(gè)對象來銷毀static int *ptr 所指向的空間,當(dāng)然也把ptr定義為共有的成員,那樣就不用定義專門的成員函數(shù)了。但破壞了封裝的特性。 a.delete_(); return 0; }

運(yùn)行結(jié)果

0 0 0

14.15題的程序?qū)崿F(xiàn):
?

#include <iostream> #include <memory> using namespace std; class Numbered; void f(Numbered); class Numbered{ friend void f(Numbered ); private:static int*ptr;int *p;int mysn; public:Numbered():p(ptr),mysn(*p){++ *p;}Numbered(const Numbered &s){p = s.p;//從指針中取走內(nèi)存中的值mysn = *p;++ *p;}void delete_(){delete ptr;}};int * Numbered::ptr = new int(0); void f(Numbered s){cout << s.mysn << endl;}int main() { Numbered a,b = a ,c = b; f(a); f(b); f(c);//調(diào)用一個(gè)對象來銷毀static int *ptr 所指向的空間 a.delete_(); return 0; }

結(jié)果是

3

4

5

13.16實(shí)現(xiàn):

#include <iostream> #include <memory> using namespace std;class Numbered; void f(const Numbered &);class Numbered{friend void f(const Numbered &);private:static int*ptr;int *p;int mysn; public:Numbered():p(ptr),mysn(*p){++ *p;}Numbered(const Numbered &s){p = s.p;//從指針中取走內(nèi)存中的值mysn = *p;++ *p;}void delete_(){delete ptr;}};int * Numbered::ptr = new int(0); void f(const Numbered &s){cout << s.mysn << endl;}int main() { Numbered a,b = a ,c = b; f(a); f(b); f(c);//調(diào)用一個(gè)對象來銷毀static int *ptr 所指向的空間 a.delete_(); return 0; }

結(jié)果是0? ?1? ?2

習(xí)題13.1.4解答完畢。

下面是以上用到程序的自己定義的版本,多定義了拷貝賦值運(yùn)算符。

#include <string> #include <iostream> #include <memory> using namespace std; class Numbered{public://本程序需要定義一塊與類對象獨(dú)立的自由空間,用static可以達(dá)到這個(gè)目的static int * ptr;Numbered():p(ptr),mysn(*p){++ *p;}Numbered(const Numbered & s):p(s.p),mysn(*p){++ *p;}Numbered & operator=(const Numbered &s){p = s.p;mysn = *p;++ *p; //這里不能寫成*p ++,否則會先把p增加,然后解引用return *this;}int num()const{return mysn;}private:int *p;int mysn = 0;}; int * Numbered::ptr = new int(0); int main() { Numbered n1; cout << n1.num() << endl; Numbered n2 = n1; cout << n2.num() << endl; Numbered n3; cout << n3.num() << endl; n3 = n1; cout << n3.num() << endl; n1 = n3; cout << n1.num() << endl; Numbered n4; cout << n4.num() << endl; n4 = n2; cout << n4.num() << endl; Numbered n5; cout << n5.num() << endl; delete Numbered::ptr;return 0; } 0 1 2 3 4 5 6 7

????????通過以上習(xí)題可以總結(jié)出一個(gè)結(jié)論,類需要使用的,但是不依賴類對象存在的事情,被定義為static是最合適的。這個(gè)對象是類使用的,但是和對象無關(guān),有關(guān)系。? ?

13.1.5?使用=default

13.1.6 阻止拷貝

?

//file.ccstruct NoCopy{ NoCopy() = default; NoCopy(const NoCopy &) = delete; NoCopy & operator=(const NoCopy &) = delete; ~NoCopy() = delete;private:int a = 0; };int main() { NoCopy n1,n2; //error: NoCopy n3 = n1; //error: n2 = n1; //error: return 0; }

編譯后結(jié)果:?

file.cc: In function ‘int main()’: file.cc:14:8: error: use of deleted function ‘NoCopy::~NoCopy()’14 | NoCopy n1,n2;| ^~ file.cc:6:1: note: declared here6 | ~NoCopy() = delete;| ^ file.cc:14:8: error: use of deleted function ‘NoCopy::~NoCopy()’14 | NoCopy n1,n2;| ^~ file.cc:6:1: note: declared here6 | ~NoCopy() = delete;| ^ file.cc:14:11: error: use of deleted function ‘NoCopy::~NoCopy()’14 | NoCopy n1,n2;| ^~ file.cc:6:1: note: declared here6 | ~NoCopy() = delete;| ^ file.cc:14:11: error: use of deleted function ‘NoCopy::~NoCopy()’14 | NoCopy n1,n2;| ^~ file.cc:6:1: note: declared here6 | ~NoCopy() = delete;| ^ file.cc:16:6: error: use of deleted function ‘NoCopy& NoCopy::operator=(const NoCopy&)’16 | n2 = n1;| ^~ file.cc:5:10: note: declared here5 | NoCopy & operator=(const NoCopy &) = delete;| ^~~~~~~~

=delete通知讀者和編譯器,我們不希望定義這些成員。

使用=delete要注意=delete和=default的差別:

1.=delete必須出現(xiàn)在函數(shù)第一次聲明的時(shí)候。一個(gè)默認(rèn)的成員只影響為這個(gè)成員而生成的代碼,因此,=default直到編譯器生成代碼的時(shí)候才需要。而另一方面,編譯器需要直到一個(gè)函數(shù)是刪除的,以便禁止試圖使用它的操作。所以必須第一次出現(xiàn)的時(shí)候就需要=delete,不然編譯器和用戶還以為是一般的函數(shù),到處使用。

2.與=default不同之處在于,=delete可以用在任何函數(shù)之中(我們只能對編譯器合成的默認(rèn)構(gòu)造函數(shù)或者拷貝構(gòu)造成員才能使用=default)。雖然刪除的函數(shù)主要用途是阻止禁止拷貝控制成員,但當(dāng)我們希望引導(dǎo)函數(shù)匹配的過程時(shí),刪除函數(shù)有時(shí)也是有用的。

一、析構(gòu)函數(shù)不能是刪除的成員。

一個(gè)類或者它的成員的析構(gòu)函數(shù)都不能是刪除的,否則這個(gè)都無法創(chuàng)建這個(gè)類類型的對象。因?yàn)檫@個(gè)類型的對象無法刪除。下面是c++primer 5th的原文:

二、析構(gòu)函數(shù)是刪除的不能定義這個(gè)類型的成員,但是可以定義指向這個(gè)類型的指針,但是不能釋放這些對象。??????

自己編寫一遍試試:

#include <string> #include <iostream> #include <memory> using namespace std; class NoDtor{ public:NoDtor() = default; //顯示使用合成默認(rèn)析構(gòu)函數(shù)~NoDtor() = delete;//我們不能銷毀NoDtor類型的成員 }; int main() {NoDtor nd; //錯(cuò)誤,析構(gòu)函數(shù)是刪除的,不能定義此類型對象NoDtor *p = new NoDtor(); // 正確,析構(gòu)函數(shù)是刪除的也可以動態(tài)分配這個(gè)類型的對象delete p; //錯(cuò)誤,delete類類型對象會調(diào)用析構(gòu)函數(shù),但是析構(gòu)函數(shù)是刪除的 return 0; }

?三、合成的拷貝控制成員可能是刪除的

1.如果類的某個(gè)類類型的成員的析構(gòu)函數(shù)是刪除的或者不可訪問的(例如private的),那么

這個(gè)類的合成的析構(gòu)函數(shù)就是刪除的。c++primer 5th原文:

2.如果類的某個(gè)成員的拷貝構(gòu)造函數(shù)是刪除的或者不可訪問的,那么這個(gè)類的合成的拷貝構(gòu)造函數(shù)就被定義為刪除的;如果一個(gè)成員析構(gòu)函數(shù)被定義為刪除的或者不可訪問的,那么這個(gè)類的合成的拷貝構(gòu)造函數(shù)也被定義為刪除的。c++primer 5th原文:

?3.如果類的某個(gè)成員的拷貝賦值運(yùn)算符被定義為刪除的或者不可訪問的,或者是類的某個(gè)成員是const 或者 引用,那么這個(gè)類合成的拷貝賦值運(yùn)算符就定義為刪除的。

class NoDtor{ public:NoDtor() = default; //顯示使用合成默認(rèn)析構(gòu)函數(shù)//我們不能銷毀NoDtor類型的成員 private:int &a; const int b; }; int main() {NoDtor n1;//因?yàn)轭怤oDtor有引用或者const成員,那么默認(rèn)構(gòu)造函數(shù)被定義為刪除的,所以不能創(chuàng)建這個(gè)類型對象 NoDtor n2; //error:原因同上 n2 = n1; //error:有const 或者引用類型(沒有合適的初始),拷貝賦值運(yùn)算符定義為刪除的return 0; } ~

4.如果類的某個(gè)成員的析構(gòu)函數(shù)是刪除的或者不可訪問的,那么類有一個(gè)引用成員,它沒有初始化器,或者類有一個(gè)const成員,它沒有類內(nèi)初始化器且類型未顯式定義默認(rèn)構(gòu)造函數(shù),則該類的默認(rèn)構(gòu)造函數(shù)被定義為刪除的。c++primer5th原文:

自己總結(jié)如下:

1.析構(gòu)函數(shù):如果一個(gè)類的析構(gòu)函數(shù)是delete的或者不可見的,那么會影響如下:

(1)該類的合成的默認(rèn)構(gòu)造函數(shù)是刪除的。不能定義此類型的對象。即使是調(diào)用此類型的合適的構(gòu)造函數(shù)也不行。

?(2)如果一個(gè)類的某個(gè)成員的析構(gòu)函數(shù)是刪除的或不可見的,那么這個(gè)類的合成的拷貝構(gòu)造函數(shù)就是定義刪除的。

??????? 其實(shí)也好理解,成員的析構(gòu)函數(shù)是刪除的或者不可見的,當(dāng)然也就不能定義那個(gè)成員類型了,當(dāng)然談不上那個(gè)成員的初始化了,那就這個(gè)類類型的合成拷貝構(gòu)造函數(shù)就自然被定義為刪除的了。按照常理推倒,即時(shí)自己定義拷貝構(gòu)造函數(shù)也不能調(diào)用去初始化這個(gè)類的對象,因?yàn)橛袀€(gè)成員是不能定義的。

(3)如果一個(gè)類的某個(gè)成員的析構(gòu)函數(shù)是刪除的不可見的,那么這個(gè)類的合成的析構(gòu)函數(shù)也是刪除的。

提醒:析構(gòu)函數(shù)除了不影響拷貝賦值運(yùn)算符,其余2個(gè)拷貝控制成員都影響,也影響默認(rèn)構(gòu)造函數(shù)。一共考慮四個(gè)函數(shù),default constructor、copy constructor、copy assignment operator、deconstructor,影響了3個(gè)。析構(gòu)函數(shù)真是夠刁的?;厥照竞苤匾?#xff0c;和現(xiàn)實(shí)世界一樣。

2.拷貝構(gòu)造函數(shù):如果一個(gè)類的拷貝構(gòu)造函數(shù)是delete的或者不可見的,那么會影響如下:

????????如果一個(gè)類的某個(gè)類類型成員拷貝構(gòu)造函數(shù)是刪除的或者不可見的,那么這個(gè)類類型的合成的拷貝構(gòu)造函數(shù)是刪除的或不可見的。

3.拷貝賦值運(yùn)算符:如果一個(gè)類的拷貝賦值運(yùn)算符是delete的或者不可見的,那么會影響如下:

??????? 如果一個(gè)類的某個(gè)成員的拷貝賦值運(yùn)算符是刪除的或者不可見的,那么這個(gè)類類型的合成的拷貝賦值運(yùn)算符就是刪除的或不可見的。

4.如果一個(gè)類有const或者引用成員(引用沒有初始化,const沒有初始值,或者const的類成員不能默認(rèn)構(gòu)造),那么會影響如下事情:

(1)首先這個(gè)類的合成的默認(rèn)構(gòu)造函數(shù)是刪除的。

(2)這個(gè)類的合成的拷貝賦值運(yùn)算符被定義為刪除的。

?????? 也就是說,如果一個(gè)類有數(shù)據(jù)成員不能默認(rèn)構(gòu)造、拷貝、賦值或銷毀,那么對應(yīng)的成員函數(shù)將被定義為刪除的。補(bǔ)充就是析構(gòu)還會影響拷貝構(gòu)造和默認(rèn)構(gòu)造,以及const或者引用的也影響默認(rèn)構(gòu)造和拷貝賦值運(yùn)算符的合成。

?注:給引用賦值以后,左邊引用會改變指向?qū)ο笾?#xff0c;而不會改變本身的值,賦值后,左側(cè)和右側(cè)引用指向不同的對象,這種行為看起來不是我們所期望嗯,所以,對于有引用類型的成員,我們定義為刪除的。

?四、private拷貝控制。

#include <string> #include <iostream> #include <memory> using namespace std; class Test{public:Test()= default;~Test()=default;private:Test(const Test &);Test& operator=(const Test &); }; int main() {Test t; //rightTest t1(t); //wrong:拷貝構(gòu)造函數(shù)是private的t1 = t; //wrong:拷貝賦值運(yùn)算符是private的 return 0; }

#include <string> #include <iostream> #include <memory> using namespace std; class Test{public:Test()= default;~Test()=default;Test copy(const Test &t){Test t1 = t; return t;}private:Test(const Test &);Test& operator=(const Test &); }; int main() {Test t;t.copy(t);return 0; }

編譯時(shí)候錯(cuò)誤(鏈接時(shí)候錯(cuò)誤)是這樣的,如下:

/usr/bin/ld: /tmp/cckYppzr.o: in function `Test::copy(Test const&)': 3.cc:(.text._ZN4Test4copyERKS_[_ZN4Test4copyERKS_]+0x36): undefined reference to `Test::Test(Test const&)' /usr/bin/ld: 3.cc:(.text._ZN4Test4copyERKS_[_ZN4Test4copyERKS_]+0x49): undefined reference to `Test::Test(Test const&)' collect2: error: ld returned 1 exit status

?13.1.6節(jié)練習(xí)答案:

練習(xí)13.18

答:

#include <string> #include <iostream> #include <memory> using namespace std; class Employee{private:static int cnt;string name;int number;public:Employee():number(cnt){++ cnt;}Employee(const string &s):name(s),number(cnt){++ cnt;}Employee(const Employee &s):name(s.name),number(cnt){++ cnt;}Employee & operator=(const Employee &s){name = s.name;number = cnt;++ cnt; return *this; }int num()const{cout << number << endl; return number;}}; int Employee::cnt = 0; int main() { Employee e1; e1.num(); Employee e2; e2.num(); Employee e3(e2); e3.num(); e1 = e3; e1.num();return 0; }

?執(zhí)行結(jié)果:

0 1 2 3

練習(xí)13.19

答案:

需要定義自己的拷貝控制成員。如果不定義的,那么直接拷貝號碼了,號碼一樣了。

練習(xí)13.20

答案:

TestQuery有2個(gè)數(shù)據(jù)成員,一個(gè)shared_ptr<vector<string>>類型的file,另一個(gè)map<string,shared_ptr<vector>>類型的wm成員。

TextQuery,拷貝時(shí)候,shared_ptr成員可以直接拷貝指針,左右兩邊指針都遞增引用計(jì)數(shù);map類型也是一樣,first成員直接拷貝,second成員直接拷貝指針,second成員指向的內(nèi)存的引用計(jì)數(shù)增加1。當(dāng)析構(gòu)的時(shí)候,分別調(diào)用2個(gè)類型的析構(gòu)函數(shù),自動釋放對象占有的內(nèi)存。

QueryResult類有3個(gè)數(shù)據(jù)成員,string sought、shared_ptr<set<line_so>> 類型的lines、shared_ptr<vector<string>>的file成員。

拷貝的時(shí)候,三個(gè)類型分別調(diào)用自己的拷貝構(gòu)造函數(shù),sought會直接拷貝,shared_ptr會直接拷貝指針,右側(cè)lines指向的內(nèi)存的引用計(jì)數(shù)增加1,file成員也一樣。析構(gòu)的時(shí)候分別調(diào)用的析構(gòu)函數(shù)。

練習(xí)13.21

答:不需要。理由如下:

兩個(gè)類雖然都沒有定義的拷貝控制成員,但是它們用智能指針管理共享的動態(tài)對象(輸入文件的內(nèi)容,查詢結(jié)果的行號集合),用標(biāo)準(zhǔn)庫容器保存大量的容器。而這些標(biāo)準(zhǔn)庫機(jī)制都有良好的拷貝控制成員,用合成的拷貝控制成員簡單地拷貝、賦值、銷毀它們,即可保證正確的資源管理。因此,這兩個(gè)類并不需要定義自己的拷貝控制成員。實(shí)際上,這兩個(gè)類的對象之間就存在資源共享,目前的設(shè)計(jì)能很好地實(shí)現(xiàn)這種共享,同類的對象之間的資源共享也能很自然地解決。

練習(xí)13.22

答:

//文件HasPtr.cpp #include "HasPtr.h" #include <string> #include <iostream> #include <new> using namespace std; HasPtr& HasPtr::operator= (const HasPtr& s) {string * new_ps = new string(*s.ps);//釋放掉ps原來指向的內(nèi)存delete ps;i = s.i;ps = new_ps;return *this; }HasPtr::HasPtr(const HasPtr &s):ps(new string(*s.ps)),i(s.i){}HasPtr::~HasPtr(){ delete ps; } //文件HasPtr.h #pragma once #include <string> #include <new> #include <iostream> using namespace std; class HasPtr { public:HasPtr(const string& s = string()):ps(new string(s)),i(0) {}HasPtr& operator= (const HasPtr& h);HasPtr(const HasPtr&);~HasPtr(); private:string* ps;int i; }; //文件main.cpp #include <string> #include <iostream> #include <new> #include "HasPtr.h" using namespace std; int main() {return 0; }

練習(xí)13.23

答:差異就是自己沒考慮到自復(fù)制時(shí)候的情況。自己的代碼沒考慮到自復(fù)制,無法處理自賦值。

練習(xí)13.24

答:如果沒定義析構(gòu)函數(shù),那么程序結(jié)束的時(shí)候會發(fā)生內(nèi)存泄漏,如果沒有定義拷貝構(gòu)造函數(shù),那么系統(tǒng)會使用默認(rèn)拷貝構(gòu)造函數(shù),會直接拷貝指針。

練習(xí)13.25

答:

//main.cpp #include <string> #include <iostream> #include <vector> #include <new> #include <memory> #include "StrBlob.h" using namespace std; int main() { StrBlob s1({"st1","st2","st3","st4"}); cout << s1.back() << endl; StrBlob s2 = s1; s1.pop_back(); cout << s1.size() << endl; cout << s2.size() << endl; cout << s1.back() << endl; cout << s2.back() <<endl;return 0; }

//StrBlob.cpp #include "StrBlob.h" #include <string> //string #include <vector> //vector #include <new> //new #include <memory> //memory #include <stdexcept> //out_of_range using namespace std; StrBlob::StrBlob():data(make_shared<vector<string>>()) {} StrBlob::StrBlob(initializer_list<string> il) : data(make_shared<vector<string>>(il)) {}StrBlob::StrBlob(const StrBlob& s) : data(make_shared<vector<string>>(*s.data)){ } StrBlob::~StrBlob() {} StrBlob& StrBlob::operator=(const StrBlob& s) {data = make_shared<vector<string>>(*s.data);return *this; }void StrBlob::push_back(const string & s) {data->push_back(s); } void StrBlob::pop_back() {check(0,"pop_back on empty StrBlob.");data->pop_back(); }string& StrBlob::front() {return data->front(); } string& StrBlob::back() {check(0,"back on empty StrBlob."); return data->back(); }void StrBlob::check(size_type i,const string & msg) const {if (i >= data->size())throw out_of_range(msg); } #pragma once #include <vector> #include <string> #include <new> #include <memory> using namespace std; class StrBlob { public:typedef vector<string>::size_type size_type;StrBlob();StrBlob(initializer_list<string> il);StrBlob(const StrBlob&);StrBlob& operator=(const StrBlob&);~StrBlob();size_type size()const { return data->size(); }bool empty()const { return data->empty(); }//add and delete elementvoid push_back(const string& t);void pop_back();//element accessstring& front();string& back(); private:shared_ptr<vector<string>> data;void check(size_type i,const string & msg)const;};

13.25.

答:拷貝構(gòu)造函數(shù)和拷貝賦值運(yùn)算符應(yīng)該為自身分配動態(tài)內(nèi)存,而不是與右側(cè)操作數(shù)共享對象。

StrBlob使用的是智能指針,可以用合成析構(gòu)函數(shù)來管理,如果一個(gè)StrBlob對象離開其作用域,則會自動調(diào)用std::shared_ptr的析構(gòu)函數(shù),當(dāng)use_count為0時(shí)釋放動態(tài)分配的內(nèi)存。

13.26

答:

//StrBlob.h #include <string> #include <iostream> #include <memory> #include <vector> #include <new> using namespace std; class StrBlob{public:using size_type = vector<string>::size_type;StrBlob();StrBlob(initializer_list<string> il);string &front()const;string &back()const;void pop_back();void push_back(const string &s);//copy constructorStrBlob(const StrBlob &s);//copy assignment operatorStrBlob& operator=(const StrBlob &s);//deconstructor ~StrBlob();private:shared_ptr<vector<string>> data;void check(size_type i,const string & msg)const; }; //functions.cc #include <string> #include <iostream> #include <memory> #include <vector> #include <new> #include "StrBlob.h" using namespace std; StrBlob::StrBlob():data(make_shared<vector<string>>()){} StrBlob::StrBlob(initializer_list<string> il):data(make_shared<vector<string>>(il)){}//copy constructor StrBlob::StrBlob(const StrBlob &s):data(make_shared<vector<string>>(*s.data)){} //copy assignment operator StrBlob& StrBlob::operator=(const StrBlob &s) {data = make_shared<vector<string>>(*s.data);return *this; } //deconstructor StrBlob::~StrBlob(){}string& StrBlob::front()const {return data->front(); } string& StrBlob::back()const {return data->back(); }void StrBlob::pop_back() {data->pop_back(); } void StrBlob::push_back(const string &s) {data->push_back(s); }void StrBlob::check(size_type i,const string &msg)const { if(i >= data -> size())throw out_of_range(msg); } #include <string> #include <iostream> #include <memory> #include <vector> using namespace std; int main() {return 0; }

13.2.2

本節(jié)內(nèi)容主要涉及實(shí)現(xiàn)類指針的類

//functions.cc #include "hasptr.h" #include <string> #include <iostream> #include <memory> #include <new> using namespace std; HasPtr::HasPtr(const string &s):ps(new string(s)),use(new size_t(0)),i(0) { //++ is priority than *(*use) ++; //or can be written ++*use } HasPtr::HasPtr(const HasPtr &rhs):ps(rhs.ps),i(rhs.i),use(rhs.use) { (*use) ++; }HasPtr& HasPtr::operator=(const HasPtr &rhs) { //自己賦值給自己什么都不發(fā)生 ++ *rhs.use;//遞增右側(cè)運(yùn)算對象的引用計(jì)數(shù) if(--*use == 0){delete ps;delete use;} i = rhs.i; use = rhs.use; ps = rhs.ps;return *this; }HasPtr::~HasPtr() { //析構(gòu)函數(shù)會遞減引用計(jì)數(shù),如果引用計(jì)數(shù)變?yōu)?,那么銷毀該對象以及該對象所占用的資源 if(--*use == 0){delete ps;delete use;} }//hasptr.h #include <string> #include <iostream> #include <memory> #include <new> #include <vector> using namespace std; class HasPtr{public:HasPtr(const string &s = string());HasPtr(const HasPtr &rhs);HasPtr& operator=(const HasPtr &rhs);~HasPtr();size_t use_count()const{return *use;}private:string *ps;int i;size_t *use; };//main.cc #include <string> #include <iostream> #include <memory> #include <vector> #include <new> #include "hasptr.h" using namespace std; int main() {HasPtr h1;cout << (h1.use_count()) << endl;HasPtr h2 = h1;cout << h1.use_count() << endl;cout << h2.use_count() << endl;HasPtr h3;cout << h3.use_count() << endl;h3 = h2;h2 = h2;cout << h3.use_count() << endl;cout << h2.use_count() << endl;cout << h1.use_count() << endl;return 0; }

這里主要有這么幾個(gè)問題,

首先:引用計(jì)數(shù)應(yīng)該用動態(tài)對象去記錄。

其次:類指針的類在幾個(gè)構(gòu)造函數(shù)或者析構(gòu)函數(shù)的設(shè)計(jì)注意如下問題:

1.拷貝構(gòu)造函數(shù):

對于引用計(jì)數(shù),先從右側(cè)對象拷貝過來,然后要遞增引用計(jì)數(shù)。其余成員逐個(gè)拷貝。就相當(dāng)于:

先把所有成員都拷貝過來(包括動態(tài)內(nèi)存成員),然后再把動態(tài)成員中的引用計(jì)數(shù)自增即可。

2.拷貝賦值運(yùn)算符:

????????要正確處理自賦值情況,先遞增右側(cè)引用計(jì)數(shù),再遞減左側(cè)引用計(jì)數(shù)。拷貝賦值運(yùn)算符進(jìn)行構(gòu)造函數(shù)和析構(gòu)函數(shù)的雙重任務(wù),在左側(cè)引用計(jì)數(shù)變?yōu)?的情況下,執(zhí)行析構(gòu)函數(shù)部分的工作(銷毀左側(cè)對象)。

3.析構(gòu)函數(shù):

? ? ? ? 析構(gòu)函數(shù)遞減引用計(jì)數(shù),如果左側(cè)的引用計(jì)數(shù)變?yōu)?,那么銷毀對象。

13.2.2節(jié)練習(xí)

練習(xí)13.27:定義你自己的引用計(jì)數(shù)版本的HasPtr。

答:略

練習(xí)13.28

答:略

13.3節(jié)

13.29

答:解釋swap(HasPtr &,HasPtr &)中對swap的調(diào)用會不會導(dǎo)致遞歸循環(huán)。

在此swap函數(shù)中又調(diào)用了swap來交換HasPtr成員ps和i。但這兩個(gè)成員的類型分別為指針和整型,都是內(nèi)置類型,因此函數(shù)中的swap調(diào)用被解析為std::swap,而不HasPtr的特定版本的swap(也就是自身),所以不會導(dǎo)致遞歸循環(huán)。

練習(xí)13.30

為你的類值版本的HasPtr編寫swap函數(shù),并測試它。為你的swap函數(shù)添加一個(gè)打印語句,指出函數(shù)什么時(shí)候執(zhí)行。

答:里面有三個(gè)文件,即是答案

//main.cc #include <string> #include <new> #include <iostream> #include "hasptr.h" using namespace std; int main() { HasPtr h1("hello,china"),h2("hello,world!"); swap(h1,h2);return 0; }//functions.cc #include "hasptr.h" #include <string> #include <iostream> #include <memory> using namespace std; HasPtr::HasPtr(const string &s):ps(new string(s)){} HasPtr::HasPtr(const HasPtr &s):ps(new string(*s.ps)),i(s.i){} HasPtr& HasPtr::operator=(const HasPtr &s){ string *new_ps = new string(*s.ps); delete ps; ps = new_ps; i = s.i;return *this; } HasPtr::~HasPtr() {delete ps; }void swap(HasPtr & lhs,HasPtr & rhs) { using std::swap; swap(lhs.ps,rhs.ps); swap(lhs.i ,rhs.i); cout << "swap(HasPtr &,HasPtr & )" << endl; }//hasptr.h #include <string> #include <iostream> #include <new> using namespace std; //類值版本 class HasPtr{public:friend void swap(HasPtr &,HasPtr &);HasPtr(const string &s = string());HasPtr(const HasPtr &);HasPtr& operator=(const HasPtr &);~HasPtr();size_t use_count()const;private:string *ps;int i; };void swap(HasPtr &,HasPtr &);

運(yùn)行結(jié)果:

swap(HasPtr &,HasPtr & )

13.31

解:

//main.cc #include <string> #include <new> #include <iostream> #include "hasptr.h" #include <vector> #include <algorithm> using namespace std; int main() { HasPtr h1("hello,china"),h2("hello,world!"),h3("hello,award.."); HasPtr h4("bc"),h5("cde"),h6("fgh"); HasPtr h7("efasdf"),h8("hiasdff"),h9("ijiasdf"); HasPtr h10("jsdf"),h11("kjk"),h12("lsdff"); HasPtr h13("msdf"),h14("nsdf"),h15("osdf"); HasPtr h16("psdf"),h17("qasf"),h18("rsfd");swap(h1,h2); vector<HasPtr> vs; vs.push_back(h1); vs.push_back(h2); vs.push_back(h3); vs.push_back(h4); vs.push_back(h5); vs.push_back(h6); vs.push_back(h7); vs.push_back(h8); vs.push_back(h9); vs.push_back(h10); vs.push_back(h11); vs.push_back(h12); vs.push_back(h13); vs.push_back(h14); vs.push_back(h15); vs.push_back(h16); vs.push_back(h17); vs.push_back(h18); sort(vs.begin(),vs.end()); for(auto i : vs)cout << i.str() << endl;return 0; }//functions.cc #include "hasptr.h" #include <string> #include <iostream> #include <memory> using namespace std; HasPtr::HasPtr(const string &s):ps(new string(s)),i(0){} HasPtr::HasPtr(const HasPtr &s):ps(new string(*s.ps)),i(s.i){} HasPtr& HasPtr::operator=(const HasPtr &s){ string *new_ps = new string(*s.ps); delete ps; ps = new_ps; i = s.i;return *this; } HasPtr::~HasPtr() {delete ps; }void swap(HasPtr & lhs,HasPtr & rhs) { using std::swap; swap(lhs.ps,rhs.ps); swap(lhs.i ,rhs.i); cout << "swap(HasPtr &,HasPtr & )" << endl; }bool HasPtr::operator<(const HasPtr &rhs)const { cout << "opeartor<" << endl; return (*ps < *rhs.ps); }//hasptr.h #include <string> #include <iostream> #include <new> using namespace std; //類值版本 class HasPtr{public:friend void swap(HasPtr &,HasPtr &);HasPtr(const string &s = string());HasPtr(const HasPtr &);HasPtr& operator=(const HasPtr &);bool operator<(const HasPtr &)const;string str()const{return *ps;}~HasPtr();size_t use_count()const;private:string *ps;int i; };void swap(HasPtr &,HasPtr &);

運(yùn)行結(jié)果:

swap(HasPtr &,HasPtr & ) opeartor< opeartor< swap(HasPtr &,HasPtr & ) opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< opeartor< bc cde efasdf fgh hello,award.. hello,china hello,world! hiasdff ijiasdf jsdf kjk lsdff msdf nsdf osdf psdf qasf rsfd

說明,一開始弄個(gè)3個(gè)元素(vector的元素?cái)?shù)),發(fā)現(xiàn)結(jié)果沒有調(diào)用swap,后來是18個(gè)元素,發(fā)現(xiàn)調(diào)用了一此swap。說明當(dāng)元素?cái)?shù)量比較少的時(shí)候,sort用的是其它方式,而元素?cái)?shù)量比較多的時(shí)候,用的swap交換元素。

13.32

答:默認(rèn)的swap版本交換兩個(gè)對象的非靜態(tài)成員,對HasPtr而言,就是交換string 的指針ps,引用計(jì)數(shù)指針use和整型數(shù)值i。可以看出,這種語意是符合期望的——兩個(gè)HasPtr指向了原來對方的string,兩者互換string后,各自的引用計(jì)數(shù)本應(yīng)該是不變的(都是減1后加1)。因此,默認(rèn)swap版本已經(jīng)能正確處理類指針HasPtr的交換,專用swap版本不會帶來更多的收益。

//strvec.cc#include <string> #include <iostream> #include <memory> #include "strvec.h" using namespace std;//一個(gè)類對象是const的,它的元素不是const的 StrVec::StrVec(const StrVec &rhs) { //allocate_n_copy:可以開辟新的存儲空間,然后把給定參數(shù)范圍的元素逐個(gè)拷貝進(jìn)去,然后 //返回構(gòu)造好的空間的首元素地址和尾后元素的地址 pair<string*, string*> data = allocate_n_copy(rhs.elements,rhs.first_free); elements = data.first; first_free = cap = data.second; } StrVec& StrVec::operator=(const StrVec & rhs) {auto data = allocate_n_copy(rhs.begin(),rhs.end());free(); //拷貝賦值運(yùn)算符比拷貝構(gòu)造函數(shù)多一個(gè)釋放左側(cè)對象的空間的函數(shù)elements = data.first;first_free = cap = data.second;return *this; } size_t StrVec::capacity()const {return cap - elements; } size_t StrVec::size()const {return first_free - elements; }void StrVec::free() { if(elements){for(string * p = first_free; p != elements;){alloc.destroy(-- p); }alloc.deallocate(elements,cap - elements); //cap -elements 可以計(jì)算一共有多少空間} }void StrVec::chk_n_alloc() { //如果容器已滿,那么重新分配 if(size() == capacity())reallocate(); }pair<string*,string*> StrVec::allocate_n_copy(const string* b,const string * e) {string * const data = alloc.allocate(e - b);return {data,uninitialized_copy(b,e,data)};} allocator<string> StrVec::alloc;void StrVec::reallocate_1() { size_t new_capacity = size() ? 1 : 2 * size(); auto new_elements = alloc.allocate(new_capacity); auto new_first_free = uninitialized_copy(elements,first_free,new_elements); free(); auto new_cap = new_elements + new_capacity; elements = new_elements; first_free = new_first_free; cap = new_cap; }void StrVec::reallocate() {auto new_capacity = size() ? 2*size() : 1; auto new_elements = alloc.allocate(new_capacity); auto dest = new_elements; auto elem = elements; for(int i = 0; i != size() ; ++ i) {//std::move(),在需要構(gòu)造的地方也是可以使用的 alloc.construct(dest ++,std::move(*elem++)); } free(); elements = new_elements; first_free = dest; cap = new_elements + new_capacity;}void StrVec::push_back(const string & rhs) {chk_n_alloc(); //確保有空間容納新元素alloc.construct(first_free ++,10,'c');}void StrVec::pop_back() {alloc.destroy(first_free --); }StrVec::~StrVec() {free(); }string * StrVec::begin()const {return elements; } string * StrVec::end()const {return first_free; }void StrVec::reserve(size_t s) { if(s <= size()) return ;//先移動元素,然后再釋放左側(cè)對象空間 auto const new_elements = alloc.allocate(s); auto dest = new_elements; auto elem = elements; for(size_t i = 0; i != size();++ i ) {alloc.construct(dest ++, std::move(*elem ++));} free(); elements = new_elements; cap = new_elements + s; first_free = dest;}void StrVec::resize(size_t t) { if(t > capacity())return ; else if(t < size()){auto new_first_free = first_free;for(size_t i = 0; i != size() - t; ++ i){alloc.destroy(-- new_first_free);}first_free = new_first_free;return;} else if(t == size()){return ;} else if(t > size()){auto new_first_free = first_free;for(size_t i = 0;i != t - size() ; ++ i){alloc.construct(new_first_free ++,""); } first_free = new_first_free; return;} }void StrVec::print_info()const { cout << "capacity:" << capacity() << endl; cout << "size:" <<size()<< endl; }

文件main.cc

#include <string> #include <iostream> #include <memory> #include "strvec.h" using namespace std; int main() { StrVec v; //cout << v.capacity() << endl; //cout << v.size() << endl;for(int i = 0;i != 10; ++i){v.push_back(string("abcde"));cout << v.size() << endl;} cout <<"capacity:" << v.capacity() << endl; cout <<"size:" << v.size() << endl;for(string * p = v.begin(); p != v.end(); ++p)cout << *p << endl; v.print_info();v.resize(5); v.resize(5); v.resize(100); cout << "big" << endl; v.print_info(); v.reserve(100); v.print_info(); cout << "after resize: " << endl; for(string* p = v.begin(); p != v.end(); ++ p)cout << *p << endl; v.print_info();return 0; } //strvec.h #include <string> #include <iostream> #include <memory> using namespace std; class StrVec {public:StrVec(): //allocator成員進(jìn)行默認(rèn)初始化elements(nullptr),first_free(nullptr),cap(nullptr){}StrVec(const StrVec &); //拷貝構(gòu)造函數(shù)StrVec& operator=(const StrVec &);~StrVec();void push_back(const string &);void pop_back();size_t size()const;size_t capacity()const;string *begin()const;string *end()const;//習(xí)題13.39void resize(size_t );void reserve(size_t );void shrink_to_fit();void print_info()const;private:static allocator<string> alloc;string *elements; //指向指向數(shù)組首元素的指針string *first_free; //指向第一個(gè)空閑元素的指針string *cap; //指向數(shù)組尾后位置的指針void free(); //銷毀元素并且釋放內(nèi)存void reallocate_1(); 獲得更多內(nèi)存并拷貝已有元素void reallocate();pair<string*,string*> allocate_n_copy(const string*,const string*);//開辟空間,拷貝給定范圍的元素void chk_n_alloc();};

運(yùn)行結(jié)果:?

1 2 3 4 5 6 7 8 9 10 capacity:16 size:10 cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc capacity:16 size:10 big capacity:16 size:5 capacity:100 size:5 after resize: cccccccccc cccccccccc cccccccccc cccccccccc cccccccccc capacity:100 size:5

strvec.h中雖然申明了shrink_to_fit,但是這個(gè)函數(shù)目前實(shí)現(xiàn)不了,因?yàn)閍llocator對象分配的空間,目前不能釋放一部分;因?yàn)獒尫臿llocator對象分配的空間需要函數(shù)deallocate函數(shù),這個(gè)函數(shù)形如:

deallocate(p,n);其中p必須是先前調(diào)用allocate得到的指針,而且n必須是先前調(diào)用allocate時(shí)候的空間的大小。

所以只能全部釋放呀。不能部分釋放。所以shrink_to_fit是不能實(shí)現(xiàn)的。

習(xí)題13.40自編答案

StrVec::StrVec(initializer_list<string> li) { //迭代器可以直接給指針賦值,棒呀,都什么情況可以這樣操作呢? auto data = allocate_n_copy(li.begin(),li.end()); elements = data.first; first_free = cap = data.second; }

練習(xí)13.42

//functions.cc #include <string> #include <iostream> #include "strblob.h" #include "strblobptr.h" #include <vector> #include "strvec.h" using namespace std;//strblob.h成員函數(shù)實(shí)現(xiàn) StrBlob::StrBlob():data(make_shared<StrVec>()){} StrBlob::StrBlob(const StrVec &s):data(make_shared<StrVec>(s)){}StrBlob::size_type StrBlob::size()const{return data->size(); } void StrBlob::push_back(const string &ele) {data->push_back(ele); }void StrBlob::pop_back(){data->pop_back(); } string & StrBlob::front() { return data->front();} string & StrBlob::back() { return data->back(); } const string& StrBlob::front()const{return data->front(); } const string& StrBlob::back()const{return data->back(); }bool StrBlob::empty()const { return data->empty(); }/* following is the member functions of class StrBlobPtr */ StrBlobPtr::StrBlobPtr():curr(0){} StrBlobPtr::StrBlobPtr(StrBlob b,size_t i):wptr(b.data),curr(i){}shared_ptr<StrVec> StrBlobPtr::check(size_t i,const string &msg)const{shared_ptr<StrVec> ret = wptr.lock(); //question1:什么情況下,ret會不指向任何對象呢? if(!ret)throw runtime_error("unbounded StrBlobPtr"); else if(i >= ret->size())throw out_of_range(msg); elsereturn ret; }StrBlobPtr StrBlob::begin()const { return StrBlobPtr(*this); } StrBlobPtr StrBlob::end()const { return StrBlobPtr(*this,this->size()); }string & StrBlobPtr::deref()const { auto ret = check(curr,"out of range"); return *(ret->begin() + curr); }StrBlobPtr &StrBlobPtr::incr() {check(curr,"out of range");++ curr;return *this; }//strvec.h StrVec::StrVec():elements(nullptr),first_free(nullptr),cap(nullptr){} allocator<string> StrVec::alloc; string &StrVec::front()const { return *elements;} string & StrVec::back()const { return *(first_free - 1);} string *StrVec::begin()const { return elements; } string * StrVec::end()const { return first_free; }StrVec::StrVec(const StrVec &s) { auto data = allocate_n_copy(s.begin(),s.end()); elements = data.first; cap = first_free = data.second; } StrVec& StrVec::operator=(const StrVec& s) { auto data = allocate_n_copy(s.begin(),s.end()); free(); elements = data.first; cap = first_free = data.second;return *this; }pair<string*,string*> StrVec::allocate_n_copy(const string* b,const string *e) { string * new_elements = alloc.allocate(e - b); string * dest = new_elements; const string * elem = b; while(elem != e){alloc.construct(dest ++,std::move(*elem ++)); }return {new_elements,dest}; }void StrVec::free() { auto elem = elements; while(elem != first_free){alloc.destroy(elem ++);} alloc.deallocate(elements,cap - elements); } size_t StrVec::size()const { return (first_free - elements);} void StrVec::push_back(const string &s) { if(first_free == cap)reallocate(); alloc.construct(first_free ++ ,s); } void StrVec::pop_back() { check(0,"pop on empty StrVec."); alloc.destroy(--first_free); } bool StrVec::empty()const { if(!elements)return true; elsereturn false;} StrVec::~StrVec() { free(); }void StrVec::reallocate() { size_t new_size = size()? 2*size():1; auto new_elements = alloc.allocate(new_size); auto new_first_free = uninitialized_copy(elements,first_free,new_elements); free(); elements = new_elements; first_free = new_first_free; cap = new_elements + new_size; }void StrVec::check(size_t i,const string &msg)const { if(i >= size()) throw out_of_range(msg); }bool StrBlobPtr::operator!=(const StrBlobPtr &s) {auto p = this->wptr.lock();auto q = s.wptr.lock();if(!(p->elements == q->elements && p->first_free == q->first_free && p->cap == q->cap))return true;elsereturn false; } //main.cc#include <string> #include <iostream> #include "strblob.h" #include "strblobptr.h" #include <vector> #include "strvec.h" using namespace std; int main() { StrVec s1; s1.push_back("abc"); s1.push_back("def"); s1.push_back("fgh"); s1.push_back("qhk"); //for(string* ptr = s1.begin(); ptr != s1.end() ;++ ptr) // cout << *ptr << endl; for(auto p = s1.begin(); p != s1.end() ; ++p)cout << *p << endl;StrBlob str(s1); StrBlobPtr pk(str); pk.incr();cout << "deref :" << endl; cout << pk.deref() << endl; cout << "end ." << endl; cout << str.size() << endl; cout << str.empty() << endl; cout << str.front() << endl; cout << str.back() << endl;return 0; }

//strblob.h #ifndef STRBLOB_H #define STRBLOB_H#include <iostream> #include <string> #include <memory> #include <vector> #include "strvec.h" using namespace std; class StrBlobPtr; class StrBlob{public:friend class StrBlobPtr;StrBlob();StrBlob(const StrVec &l);using size_type = vector<string>::size_type;size_type size()const;void push_back(const string &ele);void pop_back();string &front();string &back();const string &front()const;const string &back()const;bool empty()const;StrBlobPtr begin()const;StrBlobPtr end()const;private:shared_ptr<StrVec> data;void check(size_type i,const string &msg);};#endif

//strblobptr.h#ifndef INCLUDE_H_STRBLOBPTR #define INCLUDE_H_STRBLOBPTR#include <string> #include <iostream> #include <memory> #include <vector> #include "strvec.h" using namespace std; class StrBlobPtr{public: StrBlobPtr(); StrBlobPtr(StrBlob b,size_t curr = 0); string & deref()const; StrBlobPtr & incr(); bool operator!=(const StrBlobPtr &s);private:weak_ptr<StrVec> wptr;size_t curr;shared_ptr<StrVec> check(size_t i,const string &msg)const; };#endif //strvec.h#ifndef STRVEC_H #define STRVEC_H#include <string> #include <iostream> #include <memory> using namespace std; class StrVec {friend class StrBlobPtr;public:StrVec();StrVec(const StrVec &);StrVec& operator=(const StrVec &);~StrVec();string & front()const;string & back()const;string * begin()const;string * end()const;void pop_back();void push_back(const string &s);size_t size()const;bool empty()const;void reallocate();private:static allocator<string> alloc;string * elements;string * first_free;string * cap;pair<string*,string*> allocate_n_copy(const string *,const string *);void free();//主要檢查是否i已經(jīng)達(dá)到容器的尾后位置void check(size_t i,const string &msg)const; };#endif

在g++編譯器中執(zhí)行:

g++ main.cc functions.cc -o exe ./exe

得到如下結(jié)果:

abc def fgh qhk deref : def end . 4 0 abc qhk

結(jié)果,目前來來,全部通過編譯。

13.43更傾向于for_each算法,因?yàn)椴挥胒or去迭代。

13.44

//string.h #ifndef STRING_H_INCLUDE #define STRING_H_INCLUDE#include <utility> #include <string> #include <memory> using namespace std; class String {public:String();char * begin()const;char * end()const;char front()const;char back()const;size_t size()const;bool empty()const;void push_back(char ch);void pop_back();size_t capacity()const; private:void check(size_t ,const string &msg)const;pair<char*,char*>allocate_n_copy(char* ,char *);void free();char * elements;char * first_free;char * cap;void reallocate();static allocator<char> alloc;};#endif //functions.cc#include <iostream> #include <string> #include "string.h" #include <memory> #include <algorithm> using namespace std;String::String():elements(nullptr),first_free(nullptr),cap(nullptr){} allocator<char> String::alloc; char* String::begin()const {return elements; }char* String::end()const {return first_free; }void String::check(size_t i,const string &msg)const { if(i >= size())throw out_of_range(msg); }size_t String::size()const { return first_free - elements; } bool String::empty()const { if(size() == 0)return true; elsereturn false; } void String::push_back(char ch) { if(size() == capacity())reallocate();alloc.construct(first_free ++,ch); } void String::pop_back() {check(0,"pop on empty String.");alloc.destroy(--first_free); }char String::front()const { check(0,"front on empty String.");return *elements; }char String::back()const { check(0,"back on empty String."); char * ptr = first_free - 1;return *ptr; }size_t String::capacity()const {return cap - elements; } void String::free() /*這里一定要注意,考慮元素個(gè)數(shù)是0個(gè)的情況*/ { if(!elements)return ; for_each(elements,first_free,[](char ch){alloc.destroy(&ch);});//for_each算法省去了循環(huán) alloc.deallocate(elements,capacity()); }pair<char*,char*> String::allocate_n_copy(char * b,char * e) { auto new_elements = alloc.allocate(e - b);return {new_elements,uninitialized_copy(b,e,new_elements)}; }void String::reallocate() { auto new_size = size()?2 *size() : 1; auto new_elements = alloc.allocate(new_size); auto new_first_free = uninitialized_copy(elements,first_free,new_elements); free(); elements = new_elements; first_free = new_first_free; cap = new_size + new_elements; } //main.cc #include <string> #include <iostream> #include <memory> #include "string.h" using namespace std;int main() { String s; s.push_back('a'); cout << s.front() << endl; cout << s.back() << endl;return 0; }

13.6.2節(jié)學(xué)習(xí)

類似string類(以及其他標(biāo)準(zhǔn)庫類),如果我們自己的類能夠同時(shí)支持移動和拷貝,那么也能從中受益。為了讓我們自己的類型同時(shí)支持移動和拷貝,那么我們需要定義自己的移動構(gòu)造函數(shù)和移動賦值運(yùn)算符。這兩個(gè)成員類似拷貝構(gòu)造函數(shù)和拷貝賦值運(yùn)算符,但是它們是從給定對象“竊取”資源而不是拷貝資源。

? ? ? ? 類似拷貝構(gòu)造函數(shù),移動構(gòu)造函數(shù)的第一個(gè)參數(shù)也是該類類型的一個(gè)引用。不同于拷貝構(gòu)造函數(shù)的是,這個(gè)引用參數(shù)在移動構(gòu)造函數(shù)中是一個(gè)右值引用。與拷貝構(gòu)造函數(shù)一樣,任何其他的參數(shù)必須有默認(rèn)實(shí)參。

習(xí)題13.6.2

習(xí)題13.49

類StrVec的移動構(gòu)造函數(shù)和移動賦值運(yùn)算符:

//(1)接管資源后,要把被接管資源置于安全狀態(tài)(可析構(gòu)的狀態(tài):容器clear,內(nèi)置指針nullptr,allocator //先destroy元素,然后deallocator) //(2)注意代碼書寫也不一樣 StrVec::StrVec(StrVec && v)noexcept:elements(v.elements),first_free(v.first_free),cap(s.cap) { v.elements = v.first_free = v.cap = nullptr; } //1.(代碼從參數(shù)列表開始就注意了) //2.判斷是否是左右操作數(shù)是同一個(gè)對象,防止自賦值情況 //3.要把被接管或者被移動的對象置于安全狀態(tài)(可以析構(gòu)的狀態(tài)) StrVec& StrVec::operator=(StrVec && v)noexcept {if(this != &v){free();elements = v.elements;first_free = v.first_free;cap = v.cap;v.elements = v.first_free = v.cap = nullptr; }return *this; }

String.h的?

String::String(String && s)noexcept:elements(s.elements),first_free(s.first_free),cap(s.cap) {s.elements = s.first_free = s.cap = nullptr; }String &String::operator=(String&& s):noexcept { if(this != &s){free();elements = s.elements;first_free = s.first_free;cap = s.cap;s.elements = s.first_free = s.cap = nullptr;}return *this; }

Message::Message(Message && rhs):contents(std::move(rhs.contents)) {move_Folders(&rhs); } Message& Message::operator=(Message && m) { if(this != &m){remove_from_Folders();contents = std::move(m.contents);move_Floders(&m); } return *this; }


注意:接管資源后,要把被接管資源置于安全狀態(tài)(可析構(gòu)的狀態(tài):容器clear,內(nèi)置指針nullptr,allocator對象先destroy元素,然后deallocator。

hasptr類指針版本

文件hasptr.h

#ifndef HASPTR_H #define HASPTR_H#include <string> #include <iostream> #include <new> using namespace std; class HasPtr{public:HasPtr(const string &s = string());HasPtr(const HasPtr &);HasPtr& operator=(const HasPtr &);HasPtr(HasPtr &&)noexcept;HasPtr& operator=(HasPtr &&)noexcept;~HasPtr();size_t use_count()const{cout <<"use_count:" << *use << endl;return *use;}private:size_t * use;string * ps;int i; };inline HasPtr::HasPtr(const std::string &s): ps(new std::string(s)), i(0), use(new std::size_t(1)) {}// copy constructor copies all three data members // and increments the counter inline HasPtr::HasPtr(const HasPtr &p): ps(p.ps), i(p.i), use(p.use) { ++*use; }inline HasPtr::HasPtr(HasPtr &&p)noexcept: ps(p.ps), i(p.i), use(p.use) { p.ps = 0; p.use = 0; }inline HasPtr::~HasPtr() { //you must charge whether the object is released before //f.g.it can operate delete use and delete ps,so it is released before,it can't released again at all if(!use) //important{return ;}if (--*use == 0) { // if the reference count goes to 0delete ps; delete use; use = nullptr; //it'll be a minute can determine that whether it is released ps = nullptr; } }inline HasPtr & HasPtr::operator=(HasPtr &&rhs)noexcept {//這里先判斷是不是自賦值情況,然后才能做賦值運(yùn)算,如果先做賦值運(yùn)算,那么釋放的對象就不對了if (this != &rhs) {if (--*use == 0) { // do the work of the destructordelete ps;ps = nullptr;delete use;use = nullptr;//無論什么時(shí)候,內(nèi)置指針釋放后,置為nullptr是個(gè)不錯(cuò)的主意}ps = rhs.ps; // do the work of the move constructori = rhs.i;use = rhs.use; }return *this; } inline HasPtr& HasPtr::operator=(const HasPtr &rhs) {++*rhs.use; // increment the use count of the right-hand operandif (--*use == 0) { // then decrement this object's counterdelete ps;ps = nullptr; // if no other users delete use; use = nullptr; // free this object's allocated members}ps = rhs.ps; // copy data from rhs into this objecti = rhs.i;use = rhs.use; return *this; // return this object }#endif

0和nullptr是一回事。作用一樣。

hasptr類值版本:

#ifndef HASPTR_H #define HASPTR_H#include <string> #include <iostream> #include <new> using namespace std; /** 注意:當(dāng)你釋放了某個(gè)對象的空間后,余下的部分可以交給編譯器去做。但是不能讓編譯器再使用* 這個(gè)對象的被釋放的值,否則會引發(fā)未定義的行為。比如:Segment fault,core dumped.* 比如: 執(zhí)行h1 = h2; 之后,h1的資源就被拷貝賦值運(yùn)算符釋放掉了,此時(shí)h1(占用內(nèi)存空間的值)不能再使用 * 了,最后由編譯器釋放就可以了.*/ class HasPtr{public:HasPtr(const string &s = string());HasPtr(const HasPtr &);HasPtr& operator=(const HasPtr &);HasPtr(HasPtr &&)noexcept;HasPtr& operator=(HasPtr &&)noexcept;~HasPtr();private:string * ps;int i; }; inline HasPtr::HasPtr(const std::string &s): ps(new std::string(s)), i(0) {}// copy constructor copies all three data members // and increments the counter inline HasPtr::HasPtr(const HasPtr &p): ps(new string(*p.ps)), i(p.i) {}inline HasPtr::HasPtr(HasPtr &&p)noexcept: ps(p.ps), i(p.i){ p.ps = 0;} inline HasPtr::~HasPtr() { //you must charge whether the object is released before //f.g.it can operate delete use and delete ps,so it is released before,it can't released again at allif(!ps)delete ps; }inline HasPtr & HasPtr::operator=(HasPtr &&rhs)noexcept {//這里先判斷是不是自賦值情況,然后才能做賦值運(yùn)算,如果先做賦值運(yùn)算,那么釋放的對象就不對了if(this != &rhs){delete ps;ps = rhs.ps;i = rhs.i;rhs.ps = nullptr; } return *this; } inline HasPtr& HasPtr::operator=(const HasPtr &rhs) {string *new_ps = new string(*rhs.ps);delete ps;ps = nullptr; //這個(gè)很重要,提現(xiàn)代碼的嚴(yán)謹(jǐn)性,一會析構(gòu)函數(shù)需要檢測,檢測到nullptr,不用再釋放了,否則引發(fā)未定義行為ps = new_ps;i = rhs.i;return *this; // return this object }#endif

13.6.3節(jié)練習(xí)

13.55

void StrBlob::push_back(string &&rhs) { data->push_back(std::move(rhs));//這里必須加std::move(),因?yàn)閞hs是一個(gè)右值,但是是一個(gè)變量, //rhs本質(zhì)上是一個(gè)左值。所以要加std::move() }

13.56

會導(dǎo)致無限循環(huán)(每次調(diào)用自己)。

13.57

Foo Foo::sorted()const &{return Foo(*this).sorted();} //Foo(*this)是構(gòu)造的一個(gè)臨時(shí)Foo對象,所以成功調(diào)用右值版本的成員函數(shù) #include <string> #include <iostream> #include <new> using namespace std; class Foo {private:int a = 0;public:Foo() = default;Foo(int b):a(b){}void sort()&&{cout << "call sort() &&" <<endl;}void sort()const &{Foo ret(*this);cout << "call sort()const&" << endl;return Foo(*this).sort(); }};int main() { Foo o1,o2,o3; o1.sort(); (Foo(o1)).sort();return 0; } call sort()const& //o1.sort()調(diào)用左值版本,但是左值版本會調(diào)用右值版本所以有2個(gè)輸出 call sort() && //也是左值版本的輸出 call sort() && //(Foo(o1)).sort()因?yàn)檫@個(gè)是調(diào)用右值版本,所以有一個(gè)輸出

以下是自己先前寫的一些代碼,絕大部分都有問題。但是畢竟是自己親力所為,所以就不刪除了。

13.2節(jié)練習(xí)

練習(xí)13.22

#include <string> #include <iostream> #include <memory> using namespace std; class HasPtr{public: //這個(gè)構(gòu)造函數(shù)指定默認(rèn)初始值,s為 string()HasPtr(const string &s = string()):ps(new string(s)),i(0){cout << "call 默認(rèn)構(gòu)造函數(shù)" << endl;}HasPtr(const HasPtr&s):ps(new string(*s.ps)),i(s.i){cout << "call 拷貝構(gòu)造函數(shù)" << endl;}HasPtr&operator=(const HasPtr& s){i = s.i;//先拷貝底層string//這里有錯(cuò)誤,ps原來指向的內(nèi)存沒有釋放掉,內(nèi)存泄漏.ps = new string(*s.ps);cout << "call 拷貝賦值運(yùn)算符"<< endl; return *this;}~HasPtr(){delete ps;cout << "call 析構(gòu)函數(shù)" <<endl;}string&operator*(){return *ps;}HasPtr&operator=(const string &s){*ps = s;return *this;}private:int i;string *ps; };int main() { HasPtr h1("good2"); HasPtr h2("good1"); h2 = h1; h1 = h1; h2 = "hello,world"; cout << *h1 << endl; cout << *h2 << endl;return 0; }

乍一看,好像好像上面的程序沒有問題,其實(shí)已經(jīng)發(fā)生嚴(yán)重錯(cuò)誤——內(nèi)存泄露。

正確的拷貝賦值運(yùn)算符定義如下:

HasPtr&operator=(const HasPtr& s){i = s.i;//先拷貝底層stringstring *new_ps = new string(*s.ps);delete ps;ps = new_ps;cout << "call 拷貝賦值運(yùn)算符"<< endl; return *this;}

練習(xí)13.27答案一:

#include <string> #include <iostream> #include <memory> using namespace std; class HasPtr{public:HasPtr(const string &s=string()):ps(new string(s)),i(0),use(new size_t(1)){use ++;}//下面++*use,指的是遞增use指向的引用計(jì)數(shù)HasPtr(const HasPtr &p):ps(p.ps),i(p.i),use(p.use){++ *use;}~HasPtr(){if(--*use == 0){delete ps;delete use;}}HasPtr& operator=(const HasPtr &p){++ *p.use;-- *use;if(*use == 0){delete ps;delete use;}ps = p.ps;i = p.i;use = p.use;return *this;}private:string *ps;int i;size_t * use; };int main() {return 0; }

注意:切記不要出現(xiàn)++ use;出現(xiàn)這句,那么會把指針use指向別的單元。也就是執(zhí)行這句代碼后,use已經(jīng)不是指向原來use的內(nèi)存空間了。

13.27答案2:

#include <string> #include <iostream> #include <memory> using namespace std; class HasPtr{public:HasPtr(const string &s=string()):ps(new string(s)),i(0),use(new size_t(1)){cout <<"initialize:"<< "*use" << *use << endl;}HasPtr(const HasPtr &p):ps(p.ps),i(p.i),use(p.use){cout << "before add:"<<*use << endl;++ *use;cout << "copy constructor:" << *use << endl;}~HasPtr(){if(--*use == 0){delete ps;delete use;}}HasPtr& operator=(const HasPtr &p){++ *p.use;-- *use;if(*use == 0){delete ps;delete use;}ps = p.ps;i = p.i;use = p.use;return *this;}HasPtr& operator=(const string &p){//只是把一個(gè)新的string對象賦值給本HasPtr,其余什么都不變,注意這個(gè)思路的,這個(gè)string不能也算一個(gè)對象,所以不占用引用計(jì)數(shù)。*ps = p;return *this; }string& operator*(){return *ps; }size_t use_count()const{return *use;}private:string *ps;int i;size_t * use; };int main() { HasPtr p1("good"); HasPtr p2 = p1; cout << p1.use_count() << endl; cout << p2.use_count() << endl; cout << *p1 << endl; cout << *p2 << endl;return 0; }

1.這段代碼增加了醫(yī)院運(yùn)算符*,用來獲取HasPtr的string對象,結(jié)果是string&,所以結(jié)果是一個(gè)左值。

可以利用這句代碼改變HasPtr對象內(nèi)的string的值。

2.增加了string賦值,string賦值,僅僅改變一個(gè)HasPtr內(nèi)的string的值,其余內(nèi)容一切不改變。

總結(jié)

以上是生活随笔為你收集整理的c++ primer 5th第13章拷贝控制知识点和自编习题答案的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

亚洲成av人影片在线观看 | 三级黄色大片在线观看 | 国产亚洲日本 | 精品av在线播放 | 热re99久久精品国产99热 | 不卡电影免费在线播放一区 | 亚洲激情久久 | 在线国产精品视频 | 久久国产福利 | 伊人久久精品久久亚洲一区 | 在线看片一区 | 天天干.com | 亚洲人成影院在线 | 国产精品综合久久久久久 | 看毛片网站 | 在线观看韩国av | 亚洲精品国产精品国自产观看浪潮 | 久草在线最新 | 91日本在线播放 | 欧美做受高潮 | 欧美成人免费在线 | 狠狠色丁香婷综合久久 | 99久久精品免费一区 | 98超碰在线观看 | 久久精品中文字幕一区二区三区 | 91亚洲精品久久久蜜桃网站 | 中文字幕在线网址 | 国产精品久久久免费看 | 国产高清在线一区 | 久久视频免费看 | 成人精品亚洲 | 91九色视频观看 | 色噜噜日韩精品欧美一区二区 | 深爱婷婷 | 国产精品自产拍在线观看桃花 | 免费av成人在线 | 国产一级在线播放 | 亚洲婷婷免费 | 国内外激情视频 | 午夜精品一区二区三区在线 | 国产精品久久av | 国产成人在线免费观看 | 亚洲精品免费在线视频 | 亚洲成人av电影在线 | 欧美一区在线观看视频 | 国产亚洲精品久久久久久久久久久久 | 日韩在线一区二区免费 | 欧美激情第一区 | 国产精品一区二区麻豆 | 伊香蕉大综综综合久久啪 | 麻豆精品国产传媒 | 国产精品自产拍在线观看蜜 | 日韩中文字幕免费电影 | 国产视频一区二区在线 | 最新免费中文字幕 | 久久精品婷婷 | 天天天天天天天天操 | 精品一区欧美 | 国产精品久久久久久久久久久久午夜片 | 亚洲午夜精品在线观看 | 亚洲精品高清一区二区三区四区 | 91久久久久久久一区二区 | 中文字幕在线一区二区三区 | 色综合五月 | 草久视频在线观看 | 91传媒视频在线观看 | 精品久久久久久国产偷窥 | 久久久婷 | 最新国产精品久久精品 | 韩日精品中文字幕 | 国产精品久久av | 日本高清免费中文字幕 | 色婷婷精品大在线视频 | 99免费在线播放99久久免费 | 黄色一级性片 | 中文日韩在线 | 国产中文视| 99精品热视频| 欧美福利久久 | 午夜精品婷婷 | 亚洲精品456在线播放第一页 | 九九九热 | 欧美日韩视频一区二区三区 | 毛片播放网站 | 国产精品久久一卡二卡 | 在线观看一 | 国产99久久久国产精品成人免费 | 精品国产a | 亚洲一级二级三级 | 久久人人爽人人爽人人 | 亚洲精品白浆高清久久久久久 | 亚洲最快最全在线视频 | 精品久久久久久久 | 黄色小网站在线观看 | 亚洲视频1区2区 | 丁香久久综合 | 国产日韩欧美在线一区 | av高清一区| 在线观看91久久久久久 | www..com毛片| av不卡中文字幕 | 成人久久视频 | 免费在线观看一区 | 亚洲少妇自拍 | 免费激情网| 涩涩资源网 | 996久久国产精品线观看 | 中文字幕在线观看网 | 国产精品久久av | 国产精品第一页在线 | 91久久在线观看 | 中文字幕视频播放 | 97在线观看免费 | 精品主播网红福利资源观看 | 成人亚洲精品久久久久 | 91麻豆传媒 | www黄在线 | 97免费视频在线 | 欧美少妇18p| 成人h视频 | 黄色片毛片 | 免费网站在线观看成人 | 国产一区福利 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 国产视频精品在线 | 手机av在线免费观看 | 久久精品理论 | 在线观看精品国产 | 黄色日批网站 | 成人免费视频免费观看 | 高清国产午夜精品久久久久久 | 亚洲国产中文在线观看 | 丁香视频全集免费观看 | 91在线91 | 久久精品永久免费 | 91爱爱电影| 久久国产精品久久久久 | 亚洲精品在线视频网站 | 人人插人人艹 | 99综合电影在线视频 | 午夜视频一区二区 | 欧美久草视频 | 日韩精品在线视频免费观看 | 免费看片在线观看 | 久久精品国产v日韩v亚洲 | 一区二区三区精品在线 | 免费黄色在线 | 最新成人在线 | 日韩av中文字幕在线免费观看 | wwwwww色 | 中文视频一区二区 | 在线免费视频 你懂得 | 中文字幕乱在线伦视频中文字幕乱码在线 | 日韩高清一区在线 | 美女视频黄,久久 | 亚洲国产伊人 | 91视频麻豆 | 日批视频在线播放 | 日韩精品久久久久久久电影竹菊 | 欧美精品国产精品 | 欧美激情视频在线免费观看 | 四虎成人免费影院 | 国产96精品 | 精品你懂的 | 日韩激情中文字幕 | 国产精品99久久久精品 | 99久高清在线观看视频99精品热在线观看视频 | 久久美女免费视频 | 国产精品免费观看在线 | 免费一区在线 | 久久a级片 | 日本黄色免费在线观看 | 日韩三级av | av大全在线看 | 97国产大学生情侣白嫩酒店 | 国产 色 | 国产成人av网 | 精品一区欧美 | 久久不射电影院 | 99视频在线观看免费 | 在线日韩亚洲 | 探花视频在线观看+在线播放 | 高清视频一区二区三区 | 久久久国产99久久国产一 | 国产精品久久久久久久久久久杏吧 | 91大神免费视频 | 国产裸体永久免费视频网站 | 91精品免费在线观看 | 又污又黄的网站 | 色婷婷综合视频在线观看 | 精品国产一区二区三区久久影院 | 二区三区精品 | 福利视频一区二区 | 91在线免费播放 | 蜜桃传媒一区二区 | 国产一线在线 | 日韩一区二区免费视频 | 色中色资源站 | 国产精品黑丝在线观看 | 亚洲综合激情网 | 欧美男女爱爱视频 | 亚洲三级毛片 | 视频一区在线免费观看 | 可以免费看av | 亚洲视频网站在线观看 | 久免费视频 | 在线观看中文字幕亚洲 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 国产精品欧美久久久久无广告 | 欧美久久影院 | 日韩av二区| 国产成人精品久久二区二区 | 久久精品国产v日韩v亚洲 | 欧美另类xxxxx| 久草在线久草在线2 | 在线观看国产麻豆 | 天天干天天操天天干 | 欧美黑人性猛交 | 91精品视频一区 | 国产人免费人成免费视频 | 国产精品麻豆91 | 在线观看中文字幕dvd播放 | 婷婷丁香狠狠爱 | 性色av免费观看 | 中文字幕在线专区 | 国产精品免费av | 在线视频 精品 | 亚洲国产精品一区二区尤物区 | 欧美日韩大片在线观看 | 欧美与欧洲交xxxx免费观看 | 久久人人爽人人人人片 | 高清色免费 | 日韩在线看片 | 草久在线播放 | 国产一区二区在线播放视频 | 黄色大片入口 | 天操夜夜操 | 精品国产一区二区三区久久久蜜月 | 欧美视频不卡 | 国产精品va | 亚洲精品男人天堂 | 色综合久久88色综合天天人守婷 | 欧美另类亚洲 | 1024手机看片国产 | 91久久久久久久一区二区 | 日韩乱码中文字幕 | 9999在线观看 | 国产一区欧美日韩 | 亚洲成人高清在线 | 国产视频精品网 | 97人人澡人人爽人人模亚洲 | 日韩理论片在线 | 免费av网站在线看 | 91色国产在线 | 婷婷久操| 亚洲成人免费在线观看 | 2024国产在线| 精品在线亚洲视频 | 天天操天天操天天操天天操天天操 | 97人人澡人人爽人人模亚洲 | 久久久久二区 | 久久久精品日本 | 国产韩国精品一区二区三区 | 天天插天天干天天操 | 日韩电影在线观看中文字幕 | 美女黄濒 | 爱av在线网| 亚洲一级电影视频 | 日韩视频一区二区三区在线播放免费观看 | 91精品国产自产在线观看永久 | 在线国产中文字幕 | 狠狠干激情 | 亚洲一区二区三区精品在线观看 | 99精品国产一区二区三区麻豆 | 成人在线一区二区 | 久久黄色成人 | av大片网址 | 国产黄色片在线免费观看 | 亚洲免费黄色 | 精品在线免费视频 | 四虎影视成人精品国库在线观看 | 日韩欧美在线观看一区二区三区 | 天天色成人网 | 日日夜夜天天久久 | 日韩v在线91成人自拍 | 成人影片在线免费观看 | 国产精品精品国产婷婷这里av | 亚洲精品中文在线资源 | 国产色就色 | 国产日产欧美在线观看 | 久久久视频在线 | 日日爽天天 | www色 | 国产又粗又猛又黄又爽视频 | 看黄色91| 福利区在线观看 | 精品在线视频播放 | 一区二区视频电影在线观看 | 天天干天天草天天爽 | 亚洲精品在线观看免费 | 色亚洲激情 | 久久久久色 | 综合网成人 | 亚洲国产美女精品久久久久∴ | 中文字幕色在线视频 | 国产精品成人久久久久 | 日韩性久久 | 亚洲激情在线视频 | 在线看免费 | 日本三级香港三级人妇99 | 黄色片网站av | 波多野结衣在线观看视频 | 日韩免费中文字幕 | 精品国模一区二区三区 | 国产粉嫩在线观看 | 久久久国产视频 | 999男人的天堂 | 玖操 | 日韩精品一区二区在线 | 93久久精品日日躁夜夜躁欧美 | 3d黄动漫免费看 | 美女视频免费精品 | 国产亚洲婷婷免费 | 国产伦精品一区二区三区高清 | 狠狠干激情 | 97色免费视频 | 欧美精品久久久久久久久老牛影院 | 国产免费影院 | 亚洲最大激情中文字幕 | 在线国产欧美 | 在线观看视频福利 | 免费视频久久久久久久 | 久久久午夜视频 | 国产麻豆视频在线观看 | v片在线播放 | 国产午夜精品久久 | 国产精品麻 | 免费日韩 精品中文字幕视频在线 | 久久精品免费播放 | 久艹在线观看视频 | 成人影片免费 | 在线电影播放 | 人人澡人人澡人人 | 黄色一级大片免费看 | 91av综合| 日韩欧美在线免费观看 | 狠狠干婷婷色 | 日韩大片在线观看 | 欧美在线日韩在线 | 精品亚洲va在线va天堂资源站 | 国产一区在线不卡 | 91精品国产麻豆国产自产影视 | 久久久久久久久久久电影 | 亚洲综合在线发布 | 91爱爱中文字幕 | 超碰97免费| 免费三级影片 | 日韩一区二区三免费高清在线观看 | 国产小视频在线免费观看 | 天天干天天爽 | 免费精品视频在线 | av高清不卡 | 天天摸天天干天天操天天射 | 国产一区在线不卡 | 久久久免费电影 | 中文字幕在线影视资源 | 亚洲成人免费在线观看 | 在线观看一二三区 | 国产精品一区二区精品视频免费看 | 手机av电影在线 | 欧美色综合 | 在线观看国产麻豆 | 欧美一级片在线播放 | 久久久这里有精品 | 亚洲日本一区二区在线 | 日韩欧美视频在线播放 | 色婷婷激情电影 | 毛片久久久 | 久久久黄色av | 五月激情婷婷丁香 | 久久久久久精 | 999一区二区三区 | 国产精品999久久久 久产久精国产品 | 蜜臀av网址| 午夜精品久久久久久久久久久久 | 丁香婷婷网 | 999视频精品 | 成片免费观看视频大全 | 91porny九色在线播放 | 国产午夜精品久久 | 中文字幕在线看视频国产 | 亚洲视频综合在线 | 久久精品在线 | 精品国产伦一区二区三区观看体验 | 人人干97 | 久久久香蕉视频 | 国产精品自产拍在线观看网站 | 在线视频a | 国产a级精品 | 国产视频 亚洲精品 | 国产精品成人自产拍在线观看 | 美女黄频在线观看 | 国产精品视频永久免费播放 | 人人干人人干人人干 | 久久蜜臀一区二区三区av | 中文字幕a在线 | 精品视频9999 | 久久久99久久 | 日本精品视频网站 | 色综合久久久久久久 | 国产精品一区二区av麻豆 | 久草在线网址 | 日韩av一区二区在线影视 | 成人黄色在线观看视频 | 国产黄色免费观看 | 精品免费在线视频 | 色五月成人 | 国产一级久久久 | 激情久久一区二区三区 | 婷婷五综合 | 91精品视频一区 | 国产视频一级 | 九九在线视频免费观看 | 黄色a在线| 国产精彩视频一区 | 天天色综合久久 | 中文字幕在线观看第三页 | 欧美在线观看视频一区二区三区 | 亚洲综合视频在线观看 | a久久久久| av不卡免费在线观看 | 婷婷网五月天 | av在线播放不卡 | 在线观看免费日韩 | 久久久久福利视频 | 国产精品国产三级在线专区 | 99re在线视频观看 | 狠狠狠色狠狠色综合 | 国产录像在线观看 | 亚洲日本在线视频观看 | 久久字幕精品一区 | 二区三区中文字幕 | 在线观看黄色小视频 | 欧美精品成人在线 | 色婷婷五 | 人人搞人人爽 | 在线精品视频免费播放 | 六月丁香社区 | 欧美性生活小视频 | 欧美日韩一级久久久久久免费看 | 国产精品免费在线播放 | 日韩xxxxxxxxx | 99色婷婷| 99在线热播 | 国产成人精品在线 | 色婷婷亚洲综合 | 麻豆免费精品视频 | h网站免费在线观看 | 91在线视频 | 丁香花在线视频观看免费 | 中文在线www| 色播99| 成年人视频在线免费 | 99中文字幕在线观看 | 久久久亚洲麻豆日韩精品一区三区 | 国产日韩欧美视频在线观看 | 日韩视频在线观看免费 | 91精选| 黄色一级大片在线观看 | 久久伦理 | 免费成人在线网站 | 免费在线国产 | 欧美日韩视频在线观看免费 | 国产日韩在线播放 | 欧美久久综合 | 国内免费的中文字幕 | 日本激情动作片免费看 | 欧美影院久久 | 亚洲成人黄色网址 | 日韩中文字幕视频在线观看 | 在线观看日韩 | 丁香av | 中文字幕在线中文 | 国产日韩精品久久 | 亚洲精品国产精品国自产 | 日韩在线短视频 | 人人超碰在线 | 久久精品网站视频 | 国产精品一区二区在线看 | 国产中文在线观看 | 日韩综合第一页 | 中文字幕av在线电影 | 狠狠躁18三区二区一区ai明星 | www.五月天婷婷 | 欧美伊人网 | 国产日韩一区在线 | av三级在线播放 | 草久中文字幕 | 久久无码精品一区二区三区 | 97在线影视| 日韩在线观看网站 | 中文字幕在线看视频国产中文版 | 国产亚洲欧美日韩高清 | 五月激情丁香图片 | 日韩成人不卡 | 久久久久9999亚洲精品 | 国产中文字幕一区 | 免费观看9x视频网站在线观看 | 久久久久综合网 | 欧美日韩视频在线观看免费 | 91一区啪爱嗯打偷拍欧美 | 国产黄色片免费 | 中文字幕人成人 | 玖草在线观看 | 一区二区久久久久 | 天天干天天干天天干天天干天天干天天干 | 亚洲精品视频在线免费播放 | 国产麻豆精品传媒av国产下载 | 国产91精品高清一区二区三区 | 中文字幕资源网 国产 | 在线亚洲成人 | 中文字幕首页 | 青青久草在线视频 | 久久热首页 | 69av免费视频 | 成人在线一区二区 | 91av欧美| www.五月天色 | 日韩欧美在线免费观看 | 天天操天天操天天操天天操天天操天天操 | 国产精品va在线播放 | 久久视频在线免费观看 | 亚洲精品午夜久久久久久久 | 在线观看一二三区 | 亚洲欧美婷婷六月色综合 | 日韩精品免费在线播放 | 国产精品久久久久久久久久ktv | 香蕉在线播放 | 中文字幕高清在线播放 | 亚洲精品资源在线 | 人人爽人人 | 久久天天操 | 亚洲区视频在线观看 | 久草网站在线 | 亚洲清纯国产 | 亚洲日本欧美 | 色综合久久88色综合天天人守婷 | 久久国产精彩视频 | 国产又粗又猛又黄又爽 | av官网 | 婷婷综合在线 | 国产精品高清在线观看 | 成年人看片 | 国产精品入口麻豆 | 黄视频色网站 | www.夜夜爱 | 久久精品电影网 | 亚洲国内精品 | 99在线观看精品 | 免费在线播放av电影 | 天堂网一区二区三区 | 中文字幕电影网 | 97狠狠操| 婷婷伊人五月天 | 91精品久久久久久综合乱菊 | 人人干天天射 | 久久黄色免费观看 | 女人魂免费观看 | 欧美在线视频免费 | 色婷久久 | 国产一区二区不卡视频 | 狠狠干我| 国产一级做a爱片久久毛片a | 欧美日韩性 | 免费在线观看亚洲视频 | 99久久精品国产网站 | 久久视频二区 | 91免费观看国产 | 国产精品69av | 色是在线视频 | 日韩精品久久久免费观看夜色 | 亚洲va在线va天堂 | 99爱这里只有精品 | 久久不色 | 亚洲欧洲xxxx | 91人人澡人人爽人人精品 | 国产精品免费看久久久8精臀av | 天天性天天草 | 久久久久久久久影院 | 中文字幕成人在线观看 | www久草| 欧美精品日韩 | 中国一级片视频 | 亚洲成人资源 | 欧美美女激情18p | 综合天堂av久久久久久久 | 日韩黄色在线电影 | 午夜狠狠干 | 911久久| 五月婷婷六月丁香激情 | 久久久国产在线视频 | 人人爱人人舔 | 91av久久| 人人爽人人爽人人片av免 | 亚洲 欧洲av| 欧美日韩国产二区 | 久久天天躁夜夜躁狠狠85麻豆 | 8x成人在线 | 国产成人一区三区 | 亚洲日本中文字幕在线观看 | 欧美一区二区三区在线看 | 欧美性色综合网 | 18网站在线观看 | 国产资源在线视频 | 午夜影院在线观看18 | 在线观看亚洲精品 | 91九色国产蝌蚪 | 久久综合久久综合这里只有精品 | 午夜国产福利在线观看 | 欧美va天堂va视频va在线 | 日韩精品一区电影 | 成人不用播放器 | 色综合久久久 | 色悠悠久久综合 | 国产精品视频 | 国产高清在线视频 | 亚洲综合黄色 | 久久久国产99久久国产一 | 国产九色视频在线观看 | 91精彩在线视频 | 国语自产偷拍精品视频偷 | 免费视频网 | 一级a性色生活片久久毛片波多野 | 爱射综合| 亚洲国产精品传媒在线观看 | www狠狠操 | 在线观看国产www | 国产精品五月天 | 久久er99热精品一区二区三区 | 日本久久免费视频 | 日韩专区一区二区 | 久久久国产成人 | 天天躁日日躁狠狠 | 色永久免费视频 | 99精品在线直播 | 中文字幕最新精品 | 国产精品福利午夜在线观看 | 久久高清免费视频 | 国产精品免费视频观看 | 免费网站黄 | 亚洲精品乱码久久久久 | 午夜精品久久久久99热app | 成人久久18免费网站图片 | 国产护士在线 | 蜜臀久久99精品久久久无需会员 | 中文字幕精品久久 | 亚洲美女在线一区 | 中文字幕精品www乱入免费视频 | 夜夜高潮夜夜爽国产伦精品 | 999热线在线观看 | 黄色成人av网址 | 91在线porny国产在线看 | 精品视频www | 久久国产精品色婷婷 | 亚洲国产日韩在线 | 韩国精品视频在线观看 | 久久人人添人人爽添人人88v | 国产中文字幕视频在线观看 | av在线网站免费观看 | 免费人人干 | 久草青青在线观看 | 91私密视频 | 又长又大又黑又粗欧美 | 中文字幕在线免费播放 | 久久草网| 欧美精品小视频 | 丁香六月伊人 | av电影一区二区 | www日日| 中文字幕一区二区三区乱码不卡 | 久久在线一区 | 国产一区私人高清影院 | 欧美九九九 | 精品国模一区二区 | 天天色天天射综合网 | 婷婷丁香花 | 96精品高清视频在线观看软件特色 | 久久一区二区三区超碰国产精品 | 成人性生交大片免费观看网站 | 久久国产精品一二三区 | 国产视频99 | 亚洲欧美成人综合 | 久久久久激情视频 | 午夜精品福利一区二区三区蜜桃 | 亚洲国产美女久久久久 | 日本三级在线观看中文字 | 欧美少妇18p | 在线观看电影av | 久久久久久久久久久成人 | 国产一卡在线 | 国产97视频在线 | 国产黄色片免费 | 中文字幕在线久一本久 | 日韩国产在线观看 | 精品国产一区二区三区男人吃奶 | 精品国内自产拍在线观看视频 | 黄色电影网站在线观看 | 手机成人在线电影 | 国产黄网在线 | 精品久久久久久久 | 深爱激情婷婷网 | 综合色在线 | 最新av网址在线 | av免费网页 | 五月婷久久 | 福利精品在线 | 亚洲国产精品电影 | 亚洲成人av在线电影 | 天天爽人人爽 | 91精品国产入口 | 一区二区三区日韩在线观看 | 91中文字幕网 | a电影免费看 | 欧美一级片 | 丁香婷婷综合色啪 | 欧美少妇18p| 欧美一区二区日韩一区二区 | 91av精品| 久草干| 日韩免费视频网站 | 国产精品成人品 | 一级黄色a视频 | 最新中文字幕视频 | 1000部18岁以下禁看视频 | 国产 日韩 中文字幕 | 精品美女久久 | 国产一级精品绿帽视频 | 日韩欧美一区二区三区视频 | 91视频88av | 免费av观看网站 | 99国产精品免费网站 | 日韩高清一 | 激情伊人五月天 | 色偷偷网站视频 | 精品国产伦一区二区三区观看体验 | 午夜影院三级 | 精品国产一区二 | 国产精品美女久久久 | 国产精选在线 | 日韩精品中文字幕有码 | 久久精品精品电影网 | 国产高清福利在线 | 麻豆国产电影 | 久久69精品 | 97综合视频 | 人人天天夜夜 | 亚洲永久免费av | 久久久高清免费视频 | 午夜精品视频一区 | 国产福利91精品一区 | 久草视频视频在线播放 | 国产精品欧美 | 欧美日韩视频一区二区三区 | 日本韩国精品一区二区在线观看 | 欧美大片大全 | 精品国产诱惑 | 黄网站色成年免费观看 | 国产精品久久久久久久久久不蜜月 | 久久久久亚洲精品男人的天堂 | 亚洲丝袜一区二区 | 五月开心婷婷网 | 亚洲精品久久久久久久不卡四虎 | 亚洲永久国产精品 | 在线 高清 中文字幕 | 日本公妇色中文字幕 | 91av久久| 最近2019好看的中文字幕免费 | 国产尤物在线视频 | 日本女人逼 | 国产精品福利av | 久久日韩精品 | 免费看片网页 | 九草视频在线 | 久久精品国产久精国产 | 久久精品九色 | 亚洲中字幕 | 国产精品18久久久久vr手机版特色 | 久久综合久久综合这里只有精品 | 国产精品精品国产 | 91精品久久香蕉国产线看观看 | 91色在线观看视频 | 亚洲视频免费视频 | 亚洲精品av中文字幕在线在线 | 91香蕉视频污在线 | 国产精品一二 | 日韩在线精品一区 | 国产精品6 | 色综合久久久久久中文网 | 亚洲一区欧美激情 | 亚洲第一区精品 | 91香蕉视频 | 超碰999| 草莓视频在线观看免费观看 | 成人欧美日韩国产 | 亚洲美女视频在线 | 久久视频精品 | 久久视频在线看 | 日韩av免费大片 | 亚洲电影影音先锋 | 五月在线视频 | 伊人亚洲综合网 | 精品久久久久_ | 91香蕉视频黄色 | 日韩精品91偷拍在线观看 | 亚洲精品裸体 | 中文字幕丰满人伦在线 | 久久图| 黄色毛片大全 | 少妇自拍av| 91高清免费在线观看 | 在线精品观看 | 久久久私人影院 | 激情一区二区三区欧美 | 超碰97公开 | 亚洲黄色免费网站 | 又粗又长又大又爽又黄少妇毛片 | 在线免费观看欧美日韩 | 五月婷婷久久综合 | 日韩高清av在线 | 久草在线资源免费 | 欧美性成人| 久久久久国产a免费观看rela | 日韩久久精品一区二区 | 国产涩图 | 欧美日韩高清一区二区 国产亚洲免费看 | 久久精品精品电影网 | 欧洲高潮三级做爰 | 中文字幕在线观看播放 | 精品视频在线播放 | 久草在线视频国产 | 伊人天堂网 | 五月天激情视频在线观看 | www在线观看视频 | 免费色视频网站 | 国产精品久久久久久超碰 | 97精品久久人人爽人人爽 | 色播五月激情综合网 | 2023国产精品自产拍在线观看 | 97爱 | 欧美日韩免费一区 | 中文在线免费视频 | 久久综合偷偷噜噜噜色 | 中文字幕在线观看第一区 | 成人免费大片黄在线播放 | 粉嫩高清一区二区三区 | 九九爱免费视频在线观看 | 久久免费av | 国产福利一区在线观看 | 中文字幕在线观看不卡 | 激情欧美一区二区免费视频 | 97免费视频在线播放 | av一级久久 | 久久精品99视频 | 国产日韩在线播放 | 日本黄色免费在线 | 欧美日韩国产页 | 97在线观看免费高清 | 国产亚洲字幕 | 九色免费视频 | 亚洲va综合va国产va中文 | 日本在线视频网址 | 国产女教师精品久久av | 日韩欧美黄色网址 | 午夜视频在线观看一区二区三区 | 久草在线免费播放 | 成片免费 | 日韩一二三区不卡 | 亚洲精品女人久久久 | 久福利| 国产高清成人av | 日日爱夜夜爱 | 热热热热热色 | 91av视频在线观看免费 | 丁香六月婷婷开心婷婷网 | 日韩视频免费在线 | 欧美精品一区二区三区四区在线 | 久草97| 久久久久久久久免费视频 | 五月综合在线观看 | 日韩天天干 | 在线观看不卡视频 | 久久久综合精品 | 欧美亚洲精品在线观看 | 日韩欧美一二三 | 亚洲一级黄色片 | 久久视频一区二区 | 国产婷婷色| 免费一级毛毛片 | 欧美一级激情 | 天天爽天天射 | 五月婷影院| 久久综合精品国产一区二区三区 | 成人h视频在线 | 免费看成年人 | 国产亚洲综合性久久久影院 | 在线观看91精品视频 | 国产精品精品国产色婷婷 | 天天干干 | 婷婷伊人网 | 五月天精品视频 | 色婷婷狠狠五月综合天色拍 | 精品福利在线视频 | 久久久黄色av| 国模一区二区三区四区 | 久久无码av一区二区三区电影网 | 久久久电影 | 亚洲色综合 | 欧美日韩另类在线观看 | 五月天综合色激情 | 久久亚洲综合色 | 免费成人在线电影 | av电影免费观看 | 国产一级片在线播放 | 国产精品一区免费在线观看 | 成年人网站免费观看 | 麻豆精品视频在线 | 在线观看91精品视频 | 国产福利精品在线观看 | 中文字幕在线影院 | 中文亚洲欧美日韩 | 国产视频每日更新 | 国产精品美女在线 | 日韩成人免费电影 | 在线观看免费成人 | 三级在线视频观看 | 亚洲精品美女在线 | 免费观看不卡av | 99久久久国产精品 | 国产精品久久久久久麻豆一区 | 青草视频在线看 | 久久久久黄 | 午夜久久影视 | 午夜视频一区二区 | 亚洲精品影视在线观看 | 国产精品欧美一区二区三区不卡 | 九九久久婷婷 | 97视频在线观看视频免费视频 | 天天操天天操天天爽 | 日本公妇色中文字幕 | 久久精品直播 | 国产精品ⅴa有声小说 | .国产精品成人自产拍在线观看6 | 久草在线资源免费 | 毛片视频网址 | 国产精品日韩在线播放 | 国产成人精品免高潮在线观看 | 91麻豆看国产在线紧急地址 | 国产性天天综合网 | www.久草视频 | 久久精品影片 | 久草久热 | 中文字幕最新精品 | 国产伦精品一区二区三区四区视频 | 中文字幕在线播出 | 亚洲色视频 | 九色一区二区 | 成人一区二区三区在线 | 久久观看最新视频 | 精品国产视频在线观看 | 青青五月天| 国产精品6999成人免费视频 | 亚洲无吗天堂 | 久久精品99国产国产精 | 国产精品久久久久影视 | 国产做aⅴ在线视频播放 | 日韩欧美一二三 | 国产成人精品av | 激情综合色综合久久 | 丁香伊人网 | 在线观看日本高清mv视频 | 久久久久国产精品www | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 91色蜜桃| 日韩三级免费观看 | 欧美最猛性xxxxx(亚洲精品) | 日韩av网页 | 在线观看的黄色 | 国产 视频 久久 | 天天草天天干天天 | 干干操操| 午夜久久网 | 一区二区三区日韩视频在线观看 | 美女免费黄视频网站 | 蜜臀av性久久久久av蜜臀妖精 | 天天天干夜夜夜操 | 久久国产免费视频 | 狠狠干网址 | 99久久99久久综合 |