您的位置:首页 > 编程语言 > C语言/C++

c++类中静态函数不能调用类中的非静态函数原因

2015-10-09 13:53 225 查看
程序最终都将在内存中执行,变量只有在内存中占有一席之地时才能被访问。
 类的静态成员(变量和方法)属于类本身,在类加载的时候就会分配内存,可以通过类名直接去访问;
非静态成员(变量和方法)属于类的对象,所以只有在类的对象产生(创建类的实例)时才会分配内存,然后通过类的对象(实例)去访问。 
  在一个类的静态成员中去访问其非静态成员之所以会出错是因为在类的非静态成员不存在的时候类的静态成员就已经存在了,访问一个内存中不存在的东西当然会出错:
class CA{
private:
    int a;     //非静态成员,创建类的实例时分配内存,类的不同实例对应不同的内存区域  
    static int b;     //静态成员,类加载时分配内存,类的所有实例共享   

public:
    void fa(void)

{
        a = 1;
        b = 1;
 }       static void fb(void){
        //a = 1;        //Non-static member can't be accessed by static function
        b = 1;
    }
};

int CA::b = 1;   

int main(void)
{
    
    CA ca;
    ca.fa();

    CA::fb();
 return 0;
}

   类的静态函数和普通函数的最大区别就是 :
    (1)类的静态函数依赖类存在:
     (2)普通类的成员函数 依赖对象存在 普通类的成员函数有个this指针指向当前对象
而静态函数没有此指针。
因为它们在内存中放置的位置不同。 类的静态函数放置在静态区。而类的普通函数在函数内而存在。 类的静态函数可以通过函数名::静态函数 直接引用。而类的普通函数不可以 类的静态函数不能直接调用类的成员变量,但是能调用类的静态的成员变量。 类的普通函数可以直接调用类的成员函数,也可以调用类的静态成员变量,它是依赖对象存在的。




程序最终都将在内存中执行,变量只有在内存中占有一席之地时才能被访问。

类的静态成员(变量和方法)属于类本身,在类加载的时候就会分配内存,可以通过类名直接去访问;非静态成员(变量和方法)属于类的对象,所以只有在类的对象产生(创建类的实例)时才会分配内存,然后通过类的对象(实例)去访问。

在一个类的静态成员中去访问其非静态成员之所以会出错是因为在类的非静态成员不存在的时候类的静态成员就已经存在了,访问一个内存中不存在的东西当然会出错。

C++会区分两种类型的成员函数:静态成员函数和非静态成员函数。这两者之间的一个重大区别是,静态成员函数不接受隐含的this自变量。所以,它就无法访问自己类的非静态成员。

 

静态数据成员

    在类中,静态成员可以实现多个对象之间的数据共享,并且使用静态数据成员还不会破坏隐藏的原则,即保证了安全性。因此,静态成员是类的所有对象中共享的成员,而不是某个对象的成员。

    使用静态数据成员可以节省内存,因为它是所有对象所公有的,因此,对多个对象来说,静态数据成员只存储一处,供所有对象共用。静态数据成员的值对每个对象都是一样,但它的值是可以更新的。只要对静态数据成员的值更新一次,保证所有对象存取更新后的相同的值,这样可以提高时间效率。

静态数据成员的使用方法和注意事项如下:

1、静态数据成员在定义或说明时前面加关键字static。

2、静态成员初始化与一般数据成员初始化不同。静态数据成员初始化的格式如下:
     <数据类型><类名>::<静态数据成员名>=<值>

这表明:

(1) 初始化在类体外进行,而前面不加static,以免与一般静态变量或对象相混淆。

(2) 初始化时不加该成员的访问权限控制符private,public等。

(3) 初始化时使用作用域运算符来标明它所属类,因此,静态数据成员是类的成员,而不是对象的成员。

3、静态数据成员是静态存储的,它是静态生存期,必须对它进行初始化。

4、引用静态数据成员时,采用如下格式:

<类名>::<静态成员名>

 

静态成员函数

    静态成员函数和静态数据成员一样,它们都属于类的静态成员,它们都不是对象成员。因此,对静态成员的引用不需要用对象名。

在静态成员函数的实现中不能直接引用类中说明的非静态成员,可以引用类中说明的静态成员。如果静态成员函数中要引用非静态成员时,可通过对象来引用。下面通过例子来说明这一点。

#include <iostream>

using namespace std;

 

class M

{

public:

     M(int a) { A=a; B+=a;}

     static void f1(M m);

private:

     int A;

     static int B;

};

 

void M::f1(M m)

{

     cout<<"A="<<m.A<<endl; //静态成员函数中通过对象来引用非静态成员

     cout<<"B="<<B<<endl;

}

 

int M::B=0; //静态数据成员初始化的格式<数据类型><类名>::<静态数据成员名>=<值>

void main()

{

     M P(5),Q(10);

     M::f1(P); //静态成员函数调用时不用对象名

     M::f1(Q);

}

读者可以自行分析其结果。从中可看出,调用静态成员函数使用如下格式:

<类名>::<静态成员函数名>(<参数表>);

 

运行结果:

A=5

B=15

A=10

B=15
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  static 静态成员