指向类的成员的指针
在C++中,可以说明指向类的数据成员和成员函数的指针。两个基本操作符, 一个是*号(取地址中存放的值),一个是&(取变量地址).
指向数据成员的指针格式如下:
<类型说明符><类名>::*<指针名>
指向成员函数的指针格式如下:
<类型说明符>(<类名>::*<指针名>)(<参数表>)
int aint = 3;
int* pint;
pint = &aint; //改变指针的值
*pint = 8; //改变指针所指向的地址上的值
当定义了一个int型变量之后, C++会该变量分配一块内存空间,然后把3转换成4个字节初始化这片内存空间. 在这里, 我们要知道,其实变量名就相当于某个寄存器的别名, 也就是地址空间的名字. 我们使用这个变量名. 就像使用地址空间上面存放的值一样. 如果我们要得到aint这个变量的地址, 我们可以用&aint来获得. 程序下面又定义了一个指针变量pint,他是用来存放aint变量的内存地址的.那么我们如何使用指针呢?一般来说我们不用到变量的地址,我们更关心的变量的值,即地址中所存的值.我们用*号操作符来获取aint的值.然后输出.
符号 -> 用来调用指针做指结构或类的成员变量, 如 m->value = 8 (m为指针名)
例如,设有如下一个类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 <iostream.h>
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<<(p->*pfun)(5)<<endl; //用对象指针调用指向类成员函数指针pfun指向的函数
}
以上程序定义了好几个指针,虽然它们都是指针,但是所指向的对象是不同的。p是指向类的对象;pc是指向类的数据成员;pfun是指向类的成员函数。因此它们的值也是不相同的。
对象指针和对象引用作函数的参数
1. 对象指针作函数的参数
使用对象指针作为函数参数要经使用对象作函数参数更普遍一些。因为使用对象指针作函数参数有如下两点好处:
(1) 实现传址调用。可在被调用函数中改变调用函数的参数对象的值,实现函数之间的信息传递。
(2) 使用对象指针实参仅将对象的地址值传给形参,而不进行副本的拷贝,这样可以提高运行效率,减少时空开销。
当形参是指向对象指针时,调用函数的对应实参应该是某个对象的地址值,一般使用&后加对象名。下面举一例子说明对象指针作函数参数。
#include <iostream.h>
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;
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 <iostream.h>
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 <iostream.h>
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++编程语言中,指针和引用是两种非常重要的概念,它们都允许我们间接访问内存中的对象。本文将深入探讨这两个主题,并通过实例来对比它们的使用方法和特性。 首先,指针是C++中的一种特殊类型,它存储的是一个...
### 面向对象编程与C++中的继承和多态 #### 1. 面向对象编程思想的诞生 面向对象编程(Object-Oriented Programming, OOP)的引入,是为了克服传统面向过程编程的一些局限性。面向过程编程,如C语言,更侧重于按...
在C++编程中,基类指针和子类指针的相互赋值是一个重要的概念,涉及到面向对象程序设计中的多态性。多态性允许我们通过基类指针或引用来操作派生类的对象,从而实现更灵活的设计。本文将深入探讨这个主题。 首先,...
它常用于指针或引用,例如将常量指针转换为非常量指针,以便修改指针所指向的对象的值。在多线程设计中,const_cast也常用于去掉volatile属性。 static_cast是静态类型转换操作符,它可以将一个类型转换为另一个...
而复制构造函数的参数是一个同类型的对象引用,用于复制所有数据成员。两者被调用的时机也不同,普通构造函数在对象定义时调用,而复制构造函数在创建新对象时,当需要复制现有对象时调用。 总结来说,对象复制和...
C++中的指针是编程语言中的一个重要特性,它允许我们直接操作内存地址,从而实现高效的数据处理和复杂的程序设计。本文将深入浅出地探讨C++指针的基础知识及其应用。 首先,指针是一个变量,它存储的是另一个变量的...
引用是C++语法做的优化,引用的本质还是靠指针来实现的。引用相当于变量的别名。 引用可以改变指针的指向,还可以改变指针所指向的值。 引用的基本规则: 声明引用的时候必须初始化,且一旦绑定,不可把引用绑定到...
在C++编程中,浅拷贝(shallow copy)是一种对象复制的方式,它涉及到的是对象的数据成员的直接复制。然而,当对象包含动态分配的内存,例如指针成员时,浅拷贝可能会导致意外的结果,甚至引发错误。本文将深入探讨...
需要注意的是,`std::string` 实际存储的字符串数据不是直接包含在`std::string`对象中,而是通过指针引用的。字符串的数据是在堆上动态分配的,其大小由字符串的长度决定,而不是`sizeof()`运算符。因此,`sizeof()...
指针是C++中的精髓之一,它允许直接操作内存,实现高效的数据访问和对象间的动态链接。侯克林老师在此文档中可能涵盖了指针的基础知识,如声明、初始化和解引用,同时也可能深入讲解了指针在函数参数传递、动态内存...
在C++编程语言中,虚函数是一个非常重要的特性,它允许通过基类指针或引用调用派生类重写的成员函数,实现了多态性。本文将深入探讨C++虚函数的实现机制,以帮助读者理解其工作原理。 首先,我们要知道虚函数是如何...
C++中的虚函数是实现运行时多态性的重要机制,它允许通过基类指针或引用调用子类重写的同名方法。这种机制在面向对象编程中非常关键,因为它使得设计更加灵活,允许在不指定具体类型的情况下调用特定的行为。 首先...
在C++中,如果类没有定义自己的赋值操作符,编译器会提供一个默认的浅拷贝赋值操作符,这可能会导致问题(例如,当类包含指针或动态分配的资源时)。因此,通常建议为类提供自定义的赋值操作符(也称为"重载赋值操作...
C++除了基本数据类型,还支持自定义类型如结构体、枚举和联合体,以及C语言中的指针操作。 2. **方法定义方式**:Java中的方法都嵌入在类中定义;C++的方法可以类内或类外定义,类内定义的函数通常被认为是内联函数...
#### 2.3 浅谈C/C++内存泄漏及其检测工具 ##### 2.3.1 内存泄漏的定义 内存泄漏是指已分配的内存没有被释放,导致随着时间的推移,可用内存逐渐减少。 ##### 2.3.2 内存泄漏的发生方式 内存泄漏通常发生在以下...
在C++编程语言中,继承是面向对象编程的一个核心特性,它允许我们创建一个新类(派生类)从已有的类(基类)继承属性和行为。在继承的过程中,名字查找是确定如何访问成员函数或变量的关键步骤。本文将深入探讨C++...
标题《浅谈Javascript面向对象编程》涉及了JavaScript语言在实现面向对象编程(OOP)方面的核心概念及其灵活运用。接下来,我们详细地梳理一下从给定文件内容中提取的关于JavaScript面向对象编程的知识点。 1. 数据...
在C++编程中,内存管理是一项关键技能,尤其是在涉及动态分配和变长数组时。本文将探讨C++的内存分配机制以及如何动态分配变长数组。 首先,C++中的内存分配有三种主要方式: 1. **静态存储区域分配**:这部分内存...
- **成员变量访问**:非静态成员变量存储在对象实例中,可以通过对象指针或引用访问。 - **成员函数访问**:非虚函数的地址在编译时即可确定,而虚函数通过虚函数表(vtable)实现动态绑定,增加了运行时的开销。 ...