C++ 构造函数+析构函数+函数参数的传递

简介: C++ 构造函数+析构函数+函数参数的传递

郑重声明:小编理解甚浅,如有错误还请及时指正。

一、构造函数

c++ 构造函数详解_C/C++_若尘的博客-CSDN博客

#include <iostream>
using namespace std;
class Clock{
    private:int h,m,s;///数据成员
    public:///对外接口
        ///默认构造函数
        Clock():h(0),m(0),s(0){
           // h=0,m=0,s=0;
        }
        ///有参构造函数
        Clock(int hh,int mm,int ss):h(hh),m(mm),s(ss){
        }
        ///函数重载的理解
        Clock(int mm,int ss):h(5),m(mm),s(ss){
        }
        ///拷贝构造函数
        Clock(const Clock& x){
            h=x.h;m=x.m;s=x.s;
        }
        void showtime(){
            cout<<"("<<h<<":"<<m<<":"<<s<<")"<<endl;
        }
        void settime(int hh,int mm,int ss){
            h=hh,m=mm,s=ss;
        }
};
int main(){
    ///默认构造函数
    Clock a;
    a.showtime();
    ///有参构造函数
    Clock b(1,2,3);
    b.showtime();
    ///函数重载
    Clock c(2,3);
    c.showtime();
    ///拷贝函数
    Clock d(c);
    d.showtime();
    return 0;
}

二、析构函数

我们都知道,C语言里的局部变量在使用完毕后要释放内存空间,而模块使用后也要把申请的资源释放掉,这一步调用该过程的析构函数释放。

#include <iostream>
using namespace std;
class Clock{
    private:int h,m,s;///数据成员
    public:///对外接口
        ///默认构造函数
        Clock():h(0),m(0),s(0){
           // h=0,m=0,s=0;
        }
        ///有参构造函数
        Clock(int hh,int mm,int ss):h(hh),m(mm),s(ss){
            cout<<"有参构造函数"<<endl;
        }
        ///函数重载的理解
        Clock(int mm,int ss):h(5),m(mm),s(ss){
        }
        ///拷贝构造函数
        Clock(const Clock& x){
            cout<<"拷贝构造函数开始"<<endl;
            h=x.h;m=x.m;s=x.s;
            cout<<"拷贝构造函数结束"<<endl;
        }
        ~Clock(){
            cout<<"析构函数结束"<<endl;
        }
        void showtime(){
            cout<<"("<<h<<":"<<m<<":"<<s<<")"<<endl;
        }
        void settime(int hh,int mm,int ss){
            h=hh,m=mm,s=ss;
        }
};
void outputClock(Clock c){
    cout<<"outputClock"<<endl;
    c.showtime();
}
int main(){
    Clock c(2,3,4);
    outputClock(c);
    return 0;
}

三、类模块在函数参数传递中的通用理解

先来回想一下函数调用的过程:1.记录返回地址 2.为被调函数的执行分配资源,创造运行条件3.将控制交给被调函数执行。


函数在进行参数传递时,有两种方式,传值调用和传引用调用,这两种方式的区别就在于第二步资源的分配。前者会为实参数据在被调函数内部生成数据副本,后者是直接操作主调函数被的原始数据。

也就是说,对于类模块的参数传递中的传值方式,将会把主调函数中的实参转化为被调函数空间里的形参。

对于普通的变量(相当于C语言中的局部变量),这一过程仅仅是申请空间后进行赋值,执行完毕后系统会释放其内存,这也就保证了他的作用域。

而对于有自己的元素组成和构造方式的模块来说,仅仅是单纯的申请空间是无法生成新的模块的,这就需要用拷贝构造函数用实参生成形参。在执行完毕后,也不能单纯的释放空间来达到限制模块的作用域的目的,这就需要析构函数把模块申请的资源释放掉。

对于参数传引用的方式,仅仅是对同一个数据模块,在主调函数里是一个名字,在被调函数里是一个名字,相当于一个人的大名和小名~在这种方式下,无论是简单的变量,还是复杂的类对象,处理和理解都是一样的。

以下代码为了更好的体现这两个代码的执行过程,使用了this指针,this指针是C++自带的可以输出当前操作对象地址的东西(好高大上啊 %%)


这是传值调用代码~

#include <iostream>
using namespace std;
class Clock{
    private:int h,m,s;///数据成员
    public:///对外接口
        ///默认构造函数
        Clock():h(0),m(0),s(0){
           // h=0,m=0,s=0;
        }
        ///有参构造函数
        Clock(int hh,int mm,int ss):h(hh),m(mm),s(ss){
            cout<<"有参构造函数"<<" "<<this<<endl;
        }
        ///函数重载的理解
        Clock(int mm,int ss):h(5),m(mm),s(ss){
        }
        ///拷贝构造函数
        Clock(const Clock& x){
            cout<<"拷贝构造函数开始"<<" "<<this<<endl;
            h=x.h;m=x.m;s=x.s;
            cout<<"拷贝构造函数结束"<<" "<<this<<endl;
        }
        ~Clock(){
            cout<<"析构函数结束"<<" "<<this<<endl;
        }
        void showtime(){
            cout<<"("<<h<<":"<<m<<":"<<s<<")"<<endl;
        }
        void settime(int hh,int mm,int ss){
            h=hh,m=mm,s=ss;
        }
};
void outputClock(Clock c){
    cout<<"outputClock"<<endl;
    c.showtime();
}
int main(){
    Clock c(2,3,4);
    outputClock(c);
    return 0;
}

根据运行结果可以知道,主函数里的c和outputClock中的c的地址不同。

60a6bcefe26f4b118e50f46e4d0afd1d.png

稍微一改就是传引用调用了

代码:

#include <iostream>
using namespace std;
class Clock{
    private:int h,m,s;///数据成员
    public:///对外接口
        ///默认构造函数
        Clock():h(0),m(0),s(0){
           // h=0,m=0,s=0;
        }
        ///有参构造函数
        Clock(int hh,int mm,int ss):h(hh),m(mm),s(ss){
            cout<<"有参构造函数"<<" "<<this<<endl;
        }
        ///函数重载的理解
        Clock(int mm,int ss):h(5),m(mm),s(ss){
        }
        ///拷贝构造函数
        Clock(const Clock& x){
            cout<<"拷贝构造函数开始"<<" "<<this<<endl;
            h=x.h;m=x.m;s=x.s;
            cout<<"拷贝构造函数结束"<<" "<<this<<endl;
        }
        ~Clock(){
            cout<<"析构函数结束"<<" "<<this<<endl;
        }
        void showtime(){
            cout<<"("<<h<<":"<<m<<":"<<s<<")"<<endl;
        }
        void settime(int hh,int mm,int ss){
            h=hh,m=mm,s=ss;
        }
};
void outputClock(Clock& c){
    cout<<"outputClock"<<endl;
    c.showtime();
}
int main(){
    Clock c(2,3,4);
    outputClock(c);
    return 0;
}

运行结果~

60a6bcefe26f4b118e50f46e4d0afd1d.png

四,过程分析(其实是作业)

#include <iostream>
using namespace std;
class Point{///定义一个类对象相当于一个模块
  private:
    int x,y;///类对象的数据成员,对外不可见
  public:///类对象的对外接口
    ///构造函数:与类同名,初始化对象的数据成员,使得小模块形成完整模块
    Point(int px,int py):x(px),y(py){
      cout<<"普通构造函数生成"<<this<<endl;;
      ShowPoint();
    }
    ///拷贝构造函数:完成完整模块之间的复制
    Point(Point& p):x(p.x),y(p.y){
      cout<<"拷贝构造函数生成"<<this<<endl;
      ShowPoint();
    }
    ///显示函数
    void ShowPoint(){ cout<<"("<<x<<","<<y<<")"<<this<<endl;}
    ///析构函数:在类前面加~,释放模块申请的资源
    ~Point(){cout<<"析构函数调用"<<this<<endl;}
};
///void ShowPointInfo(Point p) 传值调用:为实参数据在被调函数内部生成数据副本
///void ShowPointInfo(Point& p) 传引用调用:直接操作主调函数被的原始数据
void ShowPointInfo(Point p){
  cout<<"ShowPointInfo begin"<<endl;
  p.ShowPoint() ;
    cout<<"ShowPointInfo end"<<endl;
}
///传值调用需要用拷贝构造函数用实参生成形参,在执行完毕后,需要析构函数把模块申请的资源释放掉
int main(){
  Point pa(3,4);
  ShowPointInfo(pa);
    cout<<endl;
  return 0;
}
/**以下结合this指针分析程序执行过程
运行结果:
普通构造函数生成0x6dfed0
(3,4)0x6dfed0
拷贝构造函数生成0x6dfed8
(3,4)0x6dfed8
ShowPointInfo begin
(3,4)0x6dfed8
ShowPointInfo end
析构函数调用0x6dfed8
析构函数调用0x6dfed0
过程分析:
首先调用构造函数对类对象的数据成员进行初始化,构造出类对象A
然后在函数传值调用时调用拷贝构造函数将主调函数里的实参转为被调函数里的形参
这一过程是生成了一个新的类对象B
被调函数执行完毕后,类对象B被析构,返回主调函数
主调函数执行完毕,类对象A被析构,程序结束
**/

完结撒花~


目录
相关文章
|
1天前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
13 4
|
22天前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
1月前
|
存储 前端开发 C++
C++ 多线程之带返回值的线程处理函数
这篇文章介绍了在C++中使用`async`函数、`packaged_task`和`promise`三种方法来创建带返回值的线程处理函数。
44 6
|
1月前
|
C++
C++ 多线程之线程管理函数
这篇文章介绍了C++中多线程编程的几个关键函数,包括获取线程ID的`get_id()`,延时函数`sleep_for()`,线程让步函数`yield()`,以及阻塞线程直到指定时间的`sleep_until()`。
21 0
C++ 多线程之线程管理函数
|
1月前
|
编译器 C语言 C++
详解C/C++动态内存函数(malloc、free、calloc、realloc)
详解C/C++动态内存函数(malloc、free、calloc、realloc)
138 1
|
1月前
|
编译器 C语言 C++
C++入门6——模板(泛型编程、函数模板、类模板)
C++入门6——模板(泛型编程、函数模板、类模板)
38 0
C++入门6——模板(泛型编程、函数模板、类模板)
|
1月前
|
C++
C++构造函数初始化类对象
C++构造函数初始化类对象
16 0
|
1月前
|
C++
C++入门4——类与对象3-2(构造函数的类型转换和友元详解)
C++入门4——类与对象3-2(构造函数的类型转换和友元详解)
20 0
|
2天前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
13 4
|
25天前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
22 4