关于this指针

关于this指针一个类的对象中实际只包含了该对象的数据成员信息,当我们创建了多个类的对象时,使对象1调用该类的成员函数,为什么可以改变对象1中的信息,而不去设置其他对象的信息?成员函数在类中只有一份,所有该类的对象共同使用,编译器是如何识别并处理的呢?编译器识别一个类分为三步:1.识别类的类名2.识别类的成员变量3.识别类的成员函数并对成员函数进行修改修改方式:成员函数有一个隐藏…

大家好,又见面了,我是你们的朋友全栈君。

一个类的对象中实际只包含了该对象的数据成员信息,当我们创建了多个类的对象时,使对象1调用该类的成员函数,为什么可以改变对象1中的信息,而不去设置其他对象的信息?成员函数在类中只有一份,所有该类的对象共同使用,编译器是如何识别并处理的呢?

编译器识别一个类分为三步:
1.识别类的类名
2.识别类的成员变量
3.识别类的成员函数并对成员函数进行修改

修改方式:
成员函数有一个隐藏的this指针,它指向调用该函数的对象
编译器为每个成员函数多加了一个参数,即this指针,它指向当前对象,并在函数内部的每个成员变量前都加上this指针

编译器是这样处理的:

class Person
{
public:
    void SetInfo(Person *this, const char * name, const char *gender, int age)
    {
        strcpy(this->_name, name);
        strcpy(this->_gender, gender);
        this->_age = age;
    }
private:
    char _name[20];
    char _gender[3];
    int _age;
};
this指针的性质:
  • this指针其本身的内容是不能被改变的,其类型为:类类型 * const
  • this指针不是对象本身的一部分,不影响该对象的大小
  • this指针的作用域在类的非静态成员函数的内部,只能在其内部进行使用,其他任何函数都不能,静态成员函数内部无this指针,后面会详述。
  • this指针是类中非静态成员函数的第一个默认隐含参数,编译器自动传递和维护,用户不可显示传递

函数调用约定
是指当一个函数被调用时,函数的参数会被传递给被调用函数,返回值会被返回给调用函数,总之,就是函数调用者与被调函数之间关于参数传递,返回值传递,堆栈清理,寄存器使用的一种约定。

它需要二进制级别兼容的强约定,函数调用者和函数体若使用不同的调用约定,可能会造成程序执行错误。

几种常用的调用约定:
这里写图片描述

其中,_cdecl是C/C++的默认调用约定,VC的调用约定中并没有_thiscall这个关键字,它是类成员函数默认调用约定;
C/C++中的main函数的调用约定必须是_cdecl,不允许更改。

对于_cdecl调用约定,为什么是调用者而不是函数体自己来平衡堆栈呢?

在这里我们应该要考虑类似于像scanf和printf这样的函数,这里我们应该明白这两个函数的参数都是可变的,如果参数不固定的话,在被调用函数内就无法知道参数究竟使用了多少个字节,所以为了实现可变参数,我们必须要在被调函数执行之后我们才知道参数究竟用了多少字节,所以我们在调用者来进行堆栈平衡操作。

这里我们重点说一下_thiscall调用约定:

  • 它只能用在类的成员函数上
  • 参数从右向左进行压栈
  • 若参数个数确定,this指针通过ecx寄存器传递给被调用者;若参数不确定,this指针在所有参数被压栈后压入堆栈。
  • 对于参数不确定的函数,调用者清理堆栈,否则函数自己清理堆栈。

问题:this指针是否可以为空?

上代码测一下先:

class Test
{
public:
    Test(int data = 0)
        :_data(data)
    {}
    void A()
    {
        cout << "A():" << this << endl;
    }
    void B()
    {
        cout << _data << endl;
    }
private:
    int _data;
};
int main()
{
    Test t;
    Test *p = NULL;
    p->A();
    p->B();
}

运行结果是:当走到 p->A();时,程序可以正常运行,运行结果是这:
这里写图片描述
而再接着往下走,走到p->B();时,程序崩溃。

为什么会这样呢?
分析一下:
函数A()中未调用任何成员变量
函数B()调用了成员变量_data
这里的p相当于一个this指针,当函数走到p->A();这一步时,编译器会将p交给ecx,再直接去调用A()函数,由于A()函数中未涉及取空指针中的内容,只是简单的打印,因此不会出现问题。
走到p->B();时,编译器会将_data修改成this->_data,而访问空指针中的内容程序一定崩溃。

因此,this指针是可以为空的,只要在成员函数内部不访问其内容,程序可以正常执行的,但是安全起见,我们还是不要让this指针为空指针最好。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/145721.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux 修改密码后SSH登录不了「建议收藏」

    linux 修改密码后SSH登录不了「建议收藏」linux修改账号密码后,密码正确一直登录不了

    2022年9月5日
    2
  • python编写淘宝秒杀脚本

    python编写淘宝秒杀脚本最近想抢冰墩墩的手办和钥匙圈,但是同志们抢的速度太快了,无奈,还是自己写脚本吧。添加火狐浏览器插件Omnibug是一个插件,可以简化web度量实现的开发。检查每个传出请求(由浏览器发送)的模式;如果出现匹配,URL将显示在开发人员工具面板中,并进行解码以显示请求的详细信息。在火狐浏览器的插件中直接搜索、下载即可安装geckodriver在python中使用selenium爬取动态渲染网页,这个过程之中需要安装浏览器驱动geckodrivergeckodrivergithub链接将下载后到

    2022年6月9日
    26
  • 索尼a5100微单参数_索尼微单a5100拍摄教程

    索尼a5100微单参数_索尼微单a5100拍摄教程入门的第一款微单—SONYa51002018-11-1109:33:0012点赞20收藏33评论开篇:这一刻我想将你永恒定格上帝给我我们一双黑色的眼睛,让我们用它来寻找光明。在寻找光明的途中我们发现光和影这一双神奇的上帝之手,让我们生活的环境变得那么的美好。有无数个美好的瞬间在我们眼前,多么希望将这个美好定格,留下这美丽的瞬间。正文:咔嚓之间将你定格——入门的第一款相机说到相机可谓是百家争…

    2025年6月7日
    0
  • 总结Redis一些使用

    总结Redis一些使用

    2021年7月12日
    69
  • pytest fixtures_premier fixture

    pytest fixtures_premier fixturefixture的优势Pytest的fixture相对于传统的xUnit的setup/teardown函数做了显著的改进:命名方式灵活,不局限于setup和teardown这几个命名conf

    2022年7月30日
    4
  • File i/o3

    File i/o3

    2022年3月3日
    36

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号