C++新特性(类与对象的各种指针和引用)

本文介绍了C++中的对象与函数的关系,强调了使用对象指针和引用作为函数参数的优势。通过实例展示了对象指针在函数参数中的应用,以及对象引用的便捷性。此外,还探讨了对象数组的声明、初始化和使用。接着讲解了子对象的概念,以及如何在类中初始化和使用它们。堆对象的概念也被引入,包括如何动态创建、删除单个和数组形式的堆对象,并强调了创建失败和使用后的内存管理注意事项。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

对象与函数的关系(知道如何把对象指针和引用作为函数参数)

 

将对象指针作为函数的参数,传递给函数处理有两个好处

1、减少数据分配的时间和空间,提高了程序运行的效率;

2、在被调函数中,可以直接改变实参对象的值,实现函数之间的信息交换。

通过一个例子感受 对象指针作为函数的参数

#include <iostream>
using namespace std;
class CPoint{
    public :
        CPoint(int x,int y);
        void copy(CPoint *point); //在成员函数中,参数是对象指针
        void setXY(int x,int y);
        void disp();
    private:
        int m_x;            //数据成员
        int m_y;
};
CPoint::CPoint(int x,int y)     //带参数的构造函数
{
    m_x=x;
    m_y=y;
}
void CPoint::copy(CPoint *point)
{
    m_x=point->m_x;          //通过对象指针访问该对象的数据成员,赋值
    m_y=point->m_y;                   //给调用成员函数的对象的数据成员
}
void CPoint::disp(){
    cout<<"x="<<m_x<<";y="<<m_y<<endl;
    }
void CPoint::setXY(int x,int y){
    m_x=x;
    m_y=y;
}
void func(CPoint *p){           //函数的参数是对象指针,通过指针调用对象的成员函数
    p->setXY(55,33);
}
int main()
{
    CPoint p1(5,5),p2(25,25);
    p1.copy(&p2);              //把对象p2的地址赋值给对象指针
    p1.disp();
    func(&p2);              //把对象p2的地址赋值给对象指针
    p2.disp();
    return 0;
}

对象引用作为函数参数:比把对象指针作为函数参数更为普遍,因为使用对象引用作为函数参数不仅具有对象指针的优点,而且使用对象引用更为直观和简单。所以在C++中普遍采用对象引用作为函数参数,看例子,注意对比上面的程序:

#include <iostream>
using namespace std;
class CPoint{
    public :
        CPoint(int x,int y);
        void copy(CPoint &point); //在成员函数中,参数是对象指针
        void setXY(int x,int y);
        void disp();
    private:
        int m_x;            //数据成员
        int m_y;
};
CPoint::CPoint(int x,int y)     //带参数的构造函数
{
    m_x=x;
    m_y=y;
}
void CPoint::copy(CPoint &point)  //函数的参数是对象引用
{
    m_x=point.m_x;          //通过对象引用访问该对象数据成员,赋值给调用成员函数的对象的数据成员
    m_y=point.m_y;
}
void CPoint::disp(){
    cout<<"x="<<m_x<<";y="<<m_y<<endl;
    }
void CPoint::setXY(int x,int y){
    m_x=x;
    m_y=y;
}
void func(CPoint &p){           //在一般函数中,参数是对象引用,通过对象引用调用成员函数
    p.setXY(55,33);
}
int main()
{
    CPoint p1(5,5),p2(25,25);
    p1.copy(p2);              //把对象p2的地址赋值给对象引用
    p1.disp();
    func(p2);              //把对象p2的地址赋值给对象引用
    p2.disp();
    return 0;
}

 

对象数组(一串连续的对象)

举一个例子来感受对象数组的声明,初始化,和使用:

#include <iostream>
#include<string.h>
using namespace std;
class CStudent{
    public:
        CStudent(char *name,int age,int score);//构造函数
        void disp();
    private:
        char m_name[16];                //数据成员
        int m_age;
        int m_score;
};
CStudent::CStudent(char *name,int age,int score){         //构造函数的实现
    strcpy(m_name,name);
    m_age=age;
    m_score=score;
}
void CStudent::disp()                             //显示构造函数
{
    cout<<"name:"<<m_name<<";age:"<<m_age<<";score:"<<m_score<<endl;
}
int main()
{
   CStudent csArray[4]={CStudent("srf",24,96),CStudent("dp",23,95),
        CStudent("aa",24,98),CStudent("bb",24,99)};
    for(int i=0;i<4;i++)
    {
        csArray[i].disp();                           //对象数组中的对象调用其成员函数
    }
    return 0;
}


 

子对象和堆对象的声明和使用

一个类中的成员是另一个类的对象时,称这个对象是子对象。换句话说,子对象就是类的对象成员。例如下面的橘子类(COrange)和苹果类(CApple):

class COrange
{
    public:
        COrange();
    private:
        ...
};
class CApple
{
    public:
        CApple();
    private:
        COrange orange;              //橘子类的子对象
};


CApple的成员orange是类COrange的对象,所以orange为子对象。当一个类中包含子对象时,这个类的构造函数中应该包含对该子对象的初始化后。并且只能采用成员初始化表的方法来初始化该子对象。通过例子感受:

包含对象成员的类应用。

#include <iostream>
#include<string.h>
using namespace std;
class COrange
{
    public:
        COrange(int heft,int sweet);
        void disp();
    private:
        int m_heft;
        int m_sweet;
};
COrange::COrange(int heft,int sweet)
{
    m_heft=heft;
    m_sweet=sweet;
}
void COrange::disp()
{
    cout<<m_heft<<";"<<m_sweet<<endl;
}
class CApple
{
    public:
        CApple(int heft,int sweet,int fragrant);
        void disp();
    private:
        COrange orange;         //声明COrange类的子对象orange
        int m_fragrant;                 //数据成员--香味
};
CApple::CApple(int heft,int sweet,int fragrant):orange(heft,sweet)
{                              //用成员初始化列表的方式初始化子对象,格式为:子对象名(参数表)
    m_fragrant=fragrant;
}
void CApple::disp(){
    orange.disp();
    cout<<m_fragrant<<endl;
}
int main()
{
    CApple apple(8,18,28);          //定义CApple类的对象apple
    apple.disp();
    return 0;
}

 

堆对象:用类定义的对象有一种特殊的对象,可以在程序运行时,随时创建和删除,满足实时的程序要求,这种对象称为堆对象。

因为这种对象的内存空间是创建在堆内存中,所以可以像堆空间一样随时申请和释放,其操作符也是new和delete。

创建格式为:new 类名(参数名);

如:

CTest *p=NULL;   //定义一个对象指针
p=new CTest();    //创建一个堆对象,并赋值给对象指针p


也可以创建堆对象数组

CTest *pa=NULL;
pa = new CTest[5];


删除时:

delete p;           //p是一个指向new创建的堆对象的对象指针
delete [] pa;          //pa是一个指向new创建的堆对象数组的对象指针

通过一个例子来感受对对象的创建,删除和使用。

#include <iostream>
#include<string.h>
using namespace std;
class CHeap
{
    public:
        CHeap();
        CHeap(int a,int b);
        ~CHeap();
        void disp();
    private:
        int m_a,m_b;
};
CHeap::CHeap()
{
    m_a=0;
    m_b=0;
    cout<<"调用默认的构造函数"<<endl;
}
CHeap::CHeap(int a,int b)
{
    m_a=a;
    m_b=b;
    cout<<"调用带参数的构造函数"<<endl;
}
CHeap::~CHeap()
{
    cout<<"调用析构函数"<<endl;
}
void CHeap::disp()
{
    cout<<"m_a="<<m_a<<";m_b="<<m_b<<endl;
}
int main()
{
    CHeap *pheap;
    pheap=new CHeap(); //创建堆对象,并把地址赋值给pheap
    if(pheap)                //判断是否创建成功
    {
        pheap->disp();  //通过对象指针,调用堆对象的成员函数
        delete pheap;      //删除堆对象
    }
    pheap = new CHeap(10,30); //把对象指针指向新的堆对象,使用参数构造。
       if(pheap)
    {
        pheap->disp();
        delete pheap;
    }

    return 0;
}


堆对象如果创建失败,new返回是NULL值,即空指针,所以使用前必须进行判断,如果创建成功,则在最后不使用时,应当删除这个堆对象,并释放其占用的堆内存空间。

堆对象数组的创建,删除,和使用,只看主函数部分的修改:

int main()
{
    CHeap *pheap;
    pheap=new CHeap[3]; //把对象指针指向堆对象数组的首元素
    if(pheap)                //判断是否创建成功
    {
      for(int i=0;i<3;i++)
      {
          (pheap+i)->disp(); //调用对象数组中对象元素的成员函数   使用对象指针加增量来访问对象数组中的每个对象元素
      }
        delete []pheap;      //删除堆对象数组
    }
   

    return 0;
}




 

指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4 指向的成员的指针C++中,可以说明指向的数据成员成员函数的指针。 指向数据成员的指针格式如下: ::* 指向成员函数的指针格式如下: (::*)() 例如,设有如下一个A: class A { public: int fun (int b) { return a*c+b; } A(int i) { a=i; } int c; private: int a; }; 定义一个指向A的数据成员c的指针pc,其格式如下: int A:: *pc = &A::c; 再定义一个指向A的成员函数fun的指针pfun,其格式如下: int (A:: *pfun)(int) = A::fun; 由于不是运行时存在的对象。因此,在使用这指针时,需要首先指定A的一个对象,然后,通过对象引用指针所指向的成员。例如,给pc指针所指向的数据成员c赋值8,可以表示如下: A a; a.*pc = 8; 其中,运算符.*是用来对指向成员的指针来操作该对象的。 如果使用指向对象指针来对指向成员的指针进行操作时,使用运算符->*。例如: A *p = &a; //a是A的一个对象,p是指向对象a的指针。 p ->* pc = 8; 让我们再看看指向一般函数的指针的定义格式: *() 关于给指向函数的指针赋值的格式如下: = 关于在程序中,使用指向函数的指针调用函数的格式如下: (*)() 如果是指向的成员函数的指针还应加上相应的对象对象成员运算符。 下面给出一个使用指向成员指针的例子: #include class A { public: A(int i) { a=i; } int fun(int b) { return a*c+b; } int c; private: int a; }; void main() { A x(8); //定义A的一个对象x int A::*pc; //定义一个指向数据成员的指针pc pc=&A::c; //给指针pc赋值 x.*pc=3; //用指针方式给成员c赋值为3 int (A::*pfun)(int); //定义一个指向成员函数的指针pfun pfun=A::fun; //给指针pfun赋值 A *p=&x; //定义一个对象指针p,并赋初值为x cout<*pfun)(5)<<endl; //用对象指针调用指向成员函数指针pfun指向的函数 } 以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向对象;pc是指向的数据成员;pfun是指向的成员函数。因此它们的值也是不相同的。 对象指针对象引用作函数的参数 1. 对象指针作函数的参数 使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处: (1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。 (2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。 当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M *m); void setxy(int i, int j) { x=i; y=j; } void print() { cout<<x<<","<<y<x; y=m->y; } void fun(M m1, M *m2); void main() { M p(5, 7), q; q.copy(&p); fun(p, &q); p.print(); q.print(); } void fun(M m1, M *m2) { m1.setxy(12, 15); m2->setxy(22,25); } 输出结果为: 5,7 22,25 从输出结果可以看出,当在被调用函数fun中,改变了对象的数据成员值[m1.setxy(12, 15)]指向对象指针的数据成员值[m2->setxy(22, 25)]以后,可以看到只有指向对象指针作参数所指向的对象被改变了,而另一个对象作参数,形参对象值改变了,可实参对象值并没有改变。因此输出上述结果。 2. 对象引用作函数参数 在实际中,使用对象引用作函数参数要比使用对象指针作函数更普遍,这是因为使用对象引用作函数参数具有用对象指针作函数参数的优点,而用对象引用作函数参数将更简单,更直接。所以,在C++编程中,人们喜欢用对象引用作函数参数。现举一例子说明对象引用作函数参数的格式。 #include class M { public: M() { x=y=0; } M(int i, int j) { x=i; y=j; } void copy(M &m); void setxy(int i, int j) { x=i; y=j; } void print() {cout<<x<<","<<y<<endl; } private: int x, y; }; void M::copy(M &m) { x=m.x; x=m.y; } void fun(M m1, M &m2;); void main() { M p(5, 7), q; q.copy(p); fun(p, q); p.print(); q.print(); } void fun(M m1, M &m2;) { m1.setxy(12, 15); m2.setxy(22, 25); } 该例子上面的例子输出相同的结果,只是调用时的参数不一样。 this指针 this指针是一个隐含于每一个成员函数中的特殊指针。它是一个指向正在被该成员函数操作的对象,也就是要操作该成员函数的对象。 当对一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,然后调用成员函数,每次成员函数存取数据成员时,由隐含作用this指针。而通常不去显式地使用this指针引用数据成员。同样也可以使用*this来标识调用该成员函数的对象。下面举一例子说明this指针的应用。 #include class A { public: A() { a=b=0; } A(int i, int j) { a=i; b=j; } void copy(A &aa;); //对象引用作函数参数 void print() {cout<<a<<","<<b<<endl; } private: int a, b; }; void A::copy(A &aa;) { if (this == &aa;) return; //这个this是操作该成员函数的对象的地址,在这里是对象a1的地址 *this = aa; //*this是操作该成员函数的对象,在这里是对象a1。 //此语句是对象aa赋给a1,也就是aa具有的数据成员的值赋给a1的数据成员 } void main() { A a1, a2(3, 4); a1.copy(a2); a1.print(); } 运行结果: 3, 4
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值