溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C++--智能指針、邏輯操作符的陷阱

發(fā)布時(shí)間:2020-07-04 20:38:26 來(lái)源:網(wǎng)絡(luò) 閱讀:433 作者:淡淡_小孩 欄目:編程語(yǔ)言

一.智能指針

內(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é)果如圖所示
C++--智能指針、邏輯操作符的陷阱
可以看出輸出結(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é)果如圖所示
C++--智能指針、邏輯操作符的陷阱
可以看到,指針在在對(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;
}

C++--智能指針、邏輯操作符的陷阱
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;
}

C++--智能指針、邏輯操作符的陷阱
可以從代碼及運(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ù)替換邏輯操作符重載

向AI問(wèn)一下細(xì)節(jié)

免責(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)容。

AI