C++ 之 this 指针或常量形参的传递问题或常量成员函数

11 篇文章 0 订阅

参考

项目
精通C++ (第九版)托尼·加迪斯、朱迪·沃尔特斯、戈德弗雷·穆甘达 (著) / 黄刚 等 (译)
搜索引擎Bing

描述

项目描述
编译器gcc version 8.1.0 (x86_64-win32-seh-rev0, Built by MinGW-W64 project)
操作系统Windows 10 专业版(64 位)

隐式形参 this 指针

默认情况下,编译器为类的每个成员函数提供了一个隐式形参,该形参指向被调用的成员函数所在的对象。该隐式形参称为 this


上述内容引用自 精通 C++ (第九版)

隐式形参 this 为一个指针,指向函数所在的对象。你可以在成员函数中直接使用隐式形参 this 。实际上,当你在成员函数中直接访问类中的成员时,就已经隐式的使用了 this 指针。

举个栗子

#include <iostream>
#include <string>
using namespace std;


class Notebook 
{
    public: 
    string content = "Hello World";
    void echo(){
        // 输出所属类的地址(以十进制展示)
        cout << (long long)this << endl;
        // 输出所属类的 content 成员属性
        cout << this -> content << endl;
    };
};

int main(){
    // 实例化
    Notebook notebook;

    notebook.echo();

    system("pause");
}

执行效果

6422000
Hello World
请按任意键继续. . .

注:

this 指针仅能指向由所属类实例化的对象,将实参赋予形参的操作将在创建类的实例对象的过程中自动完成。

冲突

当成员函数的形参与成员属性同名时,在成员函数的内部将发生冲突,成员属性将被隐藏。对此,请参考如下示例:

#include <iostream>
#include <string>
using namespace std;


class Notebook
{
    public:
    string content = "TwoMoons";
    void read(string content){
        // 输出 content 
        cout << content << endl;
    };
};

int main(){
    Notebook notebook;
    notebook.read("RedHEart");

    system("pause");
}

执行结果

RedHEart
请按任意键继续. . .

在成员函数中向控制台输出 content 中保存的数据,结果是将 形参 content 中的内容进行输出,而 成员属性 content 被忽略。

this 指针

通过 this 指针,我们可以在发生冲突的成员函数内部冲破同名形参的封锁,访问同名的成员属性。对此,请参考如下示例:

#include <iostream>
#include <string>
using namespace std;


class Notebook
{
    public:
    string content = "TwoMoons";
    void read(string content){
        // 输出 content 
        cout << content << endl;
        // 输出成员属性 content 
        cout << this -> content << endl;
    };
};

int main(){
    Notebook notebook;
    notebook.read("RedHEart");

    system("pause");
}

执行结果

RedHEart
TwoMoons
请按任意键继续. . .

常量形参

按值传递与按引用传递

向函数传递参数时,我们有两种选择,即按值传递或按引用传递。

按值传递与按引用传递

  1. 按值传递将实参复制后的副本传递给相关的形参。
  2. 按引用传递将实参所处的地址传递给形参。

比较

按值传递在实参较复杂(数据量大,执行复制操作将消耗一定的性能)时将对程序的性能造成较大的影响。按引用传递传递的是地址,通过形参中存储的地址我们可以直接访问实参所指向的内存空间。由于形参与实参指向同一内存空间,因此加大了无意修改内存空间中的数据的可能。

解决

为了避免形参修改实参指向的内存空间中的数据,我们可以将形参设定为常量。尝试修改常量形参,C++ 将抛出错误。

常量形参

举个栗子

#include <iostream>
#include <string>
using namespace std;


void fraud(const int &num1, int &num2){
    // num1 由于被设定为常量,因此不能被
    // 函数所修改。故执行
    // num1 += 100;
    // 将导致 C++ 抛出错误。

    num2 += 1;
    cout << (num1 + num2) << endl;
};


int main(){
    int num1 = 1;
    int num2 = 1;
    fraud(num1, num2);
    
    system("pause");
}

执行效果

3
请按任意键继续. . .

承诺(常量形参的传递问题)

承诺不修改 X 的函数不能将 X 传递给另一个函数,除非第二个函数也承诺不修改 X


上述内容引用自 精通 C++ (第九版)

对此,请参考如下示例:

#include <string>
#include <iostream>
using namespace std;


void underling(int &num){
    // underling 函数虽然没有承诺修改形参,
    // 但该函数并没有因此而修改形参。但即便如此,
    // C++ 也将抛出错误。
}

// fraud 函数通过关键字 const 承诺
// 不修改形参 num
void fraud(const int &num){
    // fraud 将承诺不修改的形参传递给
    // 没有承诺不修改形参 num 的函数。
    // 这将引发错误。即使 underling 没有
    // 修改形参 num 。
    underling(num);
}

int main(){
    int num = 666;
    fraud(num);

    cout << num << endl;
    
    system("pause");
}

为了使得函数 underling 能够正常接收实参,我们必须承诺不修改形参。对此,请参考如下示例:

#include <string>
#include <iostream>
using namespace std;


// underling 承诺不修改形参,于是
// underling 成功接收了 fraud 函数承诺
// 不修改的形参。程序由此正常运行。
void underling(const int &num){
    
}

// fraud 函数通过关键字 const 承诺
// 不修改形参 num
void fraud(const int &num){
    underling(num);
}

int main(){
    int num = 666;
    fraud(num);

    cout << num << endl;
    
    system("pause");
}

执行效果

666
请按任意键继续. . .

常量成员函数

常来成员函数即不能对所属对象进行修改的成员函数。要在类中声明一个常量成员函数仅需在该函数的函数头后添加关键字 const 即可。

举个栗子

#include <iostream>
#include <string>
using namespace std;


class Notebook
{
    public:
    string content = "Hello World";
    void visitor() const 
    {
        // 由于该函数已经被声明为常量成员函数,
        // 故使用如下两条语句都将导致 C++ 抛出错误。
        // this.content = "TwoMoons";
        // content = "TwoMoons";
    };
};

int main(){
    Notebook notebook;
    cout << notebook.content << endl;

    notebook.visitor();

    cout << notebook.content << endl;

    system("pause");
}

执行效果

在常量成员函数中,无论你是否通过隐式形参修改所属对象都将引发异常。

Hello World
Hello World
请按任意键继续. . .
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

BinaryMoon

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值