您好,登錄后才能下訂單哦!
內(nèi)存泄漏(C++主要的Bug來(lái)源)
1.動(dòng)態(tài)申請(qǐng)堆空間,用完后不歸還
2.C++語(yǔ)言中沒(méi)有垃圾回收機(jī)制
3.指針無(wú)法控制所指堆空間的生命周期
代碼示例
#include <iostream>
#include <string>
using namespace std;
class Test
{
int i;
public:
Test(int i)
{
this->i = i;
}
int value()
{
return i;
}
~Test()
{
}
};
int main()
{
for(int i=0; i<5; i++)
{
Test* p = new Test(i);
cout << p->value() << endl;
}
return 0;
}
運(yùn)行的結(jié)果如圖所示
可以看出輸出結(jié)果如預(yù)期一樣,但是指針在申請(qǐng)了內(nèi)存之后沒(méi)有進(jìn)行釋放,但當(dāng)出現(xiàn)大量的數(shù)據(jù)時(shí),程序會(huì)崩潰
智能指針的需求與目的
1.需要一個(gè)特殊的指針
2.指針生命周期結(jié)束時(shí)主動(dòng)釋放堆空間
3.一片堆空間最多只能有一個(gè)指針標(biāo)識(shí)
4.杜絕指針運(yùn)算何指針比較
解決方案
1.重載指針特征操作符(->)
2.只能通過(guò)類的成員函數(shù)重載
3.重載函數(shù)不能使用參數(shù)
4.只能定義一個(gè)重載函數(shù)
代碼示例
#include <iostream>
#include <string>
//智能指針的實(shí)現(xiàn)
using namespace std;
class Test
{
int i;
public:
Test(int i)
{
cout << "Test(int i)" << endl;
this->i = i;
}
int value()
{
return i;
}
~Test()
{
cout << "~Test()" << endl;
}
};
class Pointer
{
Test* mp;
public:
Pointer(Test* p = NULL)
{
mp = p;
}
Pointer(const Pointer& obj)
{
mp = obj.mp;
const_cast<Pointer&>(obj).mp = NULL;
}
Pointer& operator = (const Pointer& obj)
{
if( this != &obj )
{
delete mp;
mp = obj.mp;
const_cast<Pointer&>(obj).mp = NULL;
}
return *this;
}
Test* operator -> ()
{
return mp;
}
Test& operator * ()
{
return *mp;
}
bool isNull()
{
return (mp == NULL);
}
~Pointer()
{
delete mp;
}
};
int main()
{
Pointer p1 = new Test(0);
cout << p1->value() << endl;
Pointer p2 = p1;
cout << p1.isNull() << endl;
cout << p2->value() << endl;
return 0;
}
運(yùn)行的結(jié)果如圖所示
可以看到,指針在在對(duì)堆空間使用完之后,會(huì)自動(dòng)釋放內(nèi)存
小結(jié)
1.指針特征操作符可以被重載
2.重載指針特征符能夠使用對(duì)象代替指針
3.智能指針只能用于指向堆空間中的內(nèi)存
4.智能指針的意義在于最大程度的避免內(nèi)存問(wèn)題
A.邏輯運(yùn)算符的原生語(yǔ)義
1.操作數(shù)只有兩種值(true與false)
2.邏輯表達(dá)式不用完全計(jì)算就能確定最終值
3.最終結(jié)果只能是true或者false
代碼示例
#include <iostream>
#include <string>
using namespace std;
int func(int i)
{
cout << "int func(int i) : i = " << i << endl;
return i;
}
int main()
{
if( func(0) && func(1) )
{
cout << "Result is true!" << endl;
}
else
{
cout << "Result is false!" << endl;
}
cout << endl;
if( func(0) || func(1) )
{
cout << "Result is true!" << endl;
}
else
{
cout << "Result is false!" << endl;
}
return 0;
}
Q:邏輯操作符可以重載嗎?重載邏輯操作符有什么意義?
代碼示例及結(jié)果
#include <iostream>
#include <string>
using namespace std;
class Test
{
int mValue;
public:
Test(int v)
{
mValue = v;
}
int value() const
{
return mValue;
}
};
bool operator && (const Test& l, const Test& r)
{
return l.value() && r.value();
}
bool operator || (const Test& l, const Test& r)
{
return l.value() || r.value();
}
Test func(Test i)
{
cout << "Test func(Test i) : i.value() = " << i.value() << endl;
return i;
}
int main()
{
Test t0(0);
Test t1(1);
if( func(t0) && func(t1) )
{
cout << "Result is true!" << endl;
}
else
{
cout << "Result is false!" << endl;
}
cout << endl;
if( func(1) || func(0) )
{
cout << "Result is true!" << endl;
}
else
{
cout << "Result is false!" << endl;
}
return 0;
}
可以從代碼及運(yùn)行結(jié)果可以看出,無(wú)論邏輯操作符是否更換兩端順序,都會(huì)對(duì)兩端數(shù)據(jù)進(jìn)行運(yùn)算
B.問(wèn)題的本質(zhì)分析
1.C++通過(guò)函數(shù)調(diào)用擴(kuò)展操作符功能
2.進(jìn)入函數(shù)體前必須完成所有參數(shù)的計(jì)算
3.函數(shù)參數(shù)的計(jì)算次序是不定的
4.短路法則完全失效
短路法則
a.(表達(dá)式1)&&(表達(dá)式2) 如果表達(dá)式1為假,則表達(dá)式2不會(huì)進(jìn)行運(yùn)算,即表達(dá)式2“被短路”
b.(表達(dá)式1)||(表達(dá)式2) 如果表達(dá)式1為真,則表達(dá)式2不會(huì)進(jìn)行運(yùn)算,即表達(dá)式2“被短路
建議:實(shí)際工程開(kāi)發(fā)中避免重載邏輯操作符,通過(guò)重載比較操作符代邏輯操作符重載,直接使用成員函數(shù)代替邏輯操作符重載,使用全局函數(shù)對(duì)邏輯操作符重載
小結(jié)
1.C++從語(yǔ)法上支持邏輯操作符重載
2.重載后的邏輯操作符不滿足短路法則
3.工程開(kāi)發(fā)中不要重載邏輯操作符
4.通過(guò)重載比較操作符替換邏輯操作符重載
5.通過(guò)專用成員函數(shù)替換邏輯操作符重載
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。