C++类中的静态成员

目录

1.静态成员变量:

2.静态成员函数

在C++中类中的静态成员分为两类,一类是静态成员变量,一类是静态成员函数。什么是静态成员变量和静态成员函数呢?就是在前面加static关键字。

1.静态成员变量:

它具有以下几个特点:

        1.所有的对象共享同一份数据

        2.在编译阶段分配内存,在主函数之前进行构造

        3.类内声明,类外初始化

        4.可以通过类或者对象访问共有的静态成员变量

        5.可发生继承时,静态成员变量不会被继承,子类和父类共享同一个静态成员变量。

        6.静态成员变量不占用对象内存

        7.静态成员变量也是有访问权限的,私有的类外无法访问

下面,我们逐点进行分析:

首先,所以对象共享同一份数据,那么这个静态成员变量就不是独属于某一个对象,这也证明了第6点:静态成员变量不占用对象的内存。看这样一段代码:

#include<iostream>
using namespace std;
class A {
public:
	int a;
	static int b;
	A() {
		this->a = 5;
	}
};
int A::b = 9;//应该在类外进行初始化,在类内初始化会报错
int main() {
	cout <<"类A的大小是:"<< sizeof(A) << endl;
	A a1;
	A a2;
	a2.b = 6;
	cout << a1.b << endl;
	cout << a2.b << endl;
	return 0;
}

定义了一个类A,里面包括一个成员变量和一个静态成员变量。我们利用sizeof来计算这个类的大小,按道理说,计算的结果应该是8,

结果显示类A的大小是4,这就说明静态成员变量不占用对象的内存。同时,在主函数中定义了两个A的对象,他们都可以访问到静态成员变量b,通过对象a2去修改静态成员变量的值,最终输出来的结果是6,而不是初始化的值9,所以说是所有对象共享同一份数据。

对于静态成员变量的访问有两个形式,一种是通过对象访问,一种是通过类访问,如上图。

#include<iostream>
using namespace std;
class A {
public:
	int a;
	static int b;
	A() {
		this->a = 5;
	}
};
class B : public A {
public:
	int num;
};
int A::b = 9;//应该在类外进行初始化,在类内初始化会报错
int main() {
	A a1;
	B b1;
	cout<<a1.b<<endl;
	cout<<b1.b << endl;
	return 0;
}

我们定义了一个类B,让他继承自类A,然后在主函数中都定义了这两个类的对象,他们都可以访问到静态成员变量。

说明子类和父类也是共享的同一个静态成员变量。

同时,也可以设置静态成员变量的访问权限,如果设置成私有的话,那么在类外就无法进行访问,需要在静态成员函数中才能访问。

2.静态成员函数

        1.静态成员函数只能访问静态成员变量

        2.没有this指针,所以静态成员函数内部不可以使用成员变量和成员函数

我们知道在类中函数传参的时候会有一个隐含参数this指针,但是在上面我们用这个this指针被告诉报错了,这就说明在静态成员函数中是没有this指针的,那么它也就无法访问类中的非静态成员和非静态成员函数的(因为要通过this指针找到这个对象,才能访问它的成员变量)。

        所以静态成员函数只能访问静态成员变量。

#include<iostream>
using namespace std;
class A {
public:
	int num;
	static int t;
	static void fun() {
		//num = 9;//报错,没有this指针,
		t = 10;
		cout << t << endl;
	}
};
int A::t = 5;
int main() {
	A a;
	a.fun();
	A::fun();
	return 0;
}

静态成员函数的访问方式和静态成员变量的方式相同。

<think>好的,我现在需要回答用户关于C++静态成员变量的定义、初始化及用法的教程。首先,我需要回顾一下相关的知识,确保自己理解正确,然后再整理成用户容易理解的步骤。 首先,静态成员变量属于整个,而不是的某个实例。这意味着所有的对象共享同一个静态成员变量。这点很重要,用户可能需要明确静态成员和普通成员变量的区别。 接下来是声明和定义部分。根据引用[2],静态成员变量需要在内声明,使用static关键字,然后在外进行初始化。比如在内声明为static int number;,然后在外通过int ClassName::number = 0;来初始化。这里需要注意,初始化时不需要再加static关键字,否则会导致编译错误。这部分可能需要详细说明,因为用户可能会在这里犯错。 然后是关于初始化的原因。用户可能疑惑为什么静态成员变量不能像普通成员变量一样在声明时初始化。根据引用[1],这是因为静态成员变量不属于任何对象,它们的存储空间在全局数据区,因此需要在外显式分配和初始化。这样设计可能是为了避免重复定义的问题,确保静态变量只有一个实例存在。 关于初始化的方式,引用[2]提到有两种方法:声明并初始化,或者只进行初始化。但根据标准C++,正确的做法是在外进行定义和初始化,例如int Student::number = 1;。而像Student::number = 2;这样的赋值应该在函数内部进行,可能是在的成员函数中或者在main函数里。需要区分初始化和赋值的不同。 接下来是使用方法。用户需要知道如何访问静态成员变量。可以通过名加作用域解析运算符,如ClassName::staticVar,或者通过对象访问,如obj.staticVar。但通常推荐使用名访问,以明确其静态属性。 然后,可能需要提到静态成员变量的应用场景,比如统计实例的数量,作为共享的配置参数等。例如,在构造函数中递增静态计数变量,在析构函数中递减,这样可以跟踪当前存在的对象数目。 另外,要注意静态成员变量的访问权限。如果声明为private,则需要通过静态成员函数或友元来访问,这部分可能需要简单提及,但用户的问题可能更关注基本用法,所以暂时不需要深入。 还需要对比普通静态变量和静态成员变量的初始化区别。根据引用[3],普通静态变量(如函数内的static变量)在定义时分配内存,系统自动初始化为0,而静态成员变量必须显式初始化。这可能是因为的作用域和链接性问题,需要用户手动处理。 最后,确保示例代码的正确性。例如,在内声明static int count;,在外int MyClass::count = 0;,然后在构造函数中递增,在析构函数中递减。这样用户可以直观看到静态变量如何工作。 总结步骤:声明、初始化、访问、应用示例。需要分点说明,确保结构清晰。同时,引用相关文献来支持解释,比如引用[1]说明为什么需要外初始化,引用[2]说明初始化的两种方式,引用[3]对比普通静态变量的初始化。 在编写过程中,要避免使用复杂术语,用简单的中文解释,并检查LaTeX格式是否正确,如变量名用$...$包裹,独立公式用$$...$$。同时,生成相关问题,帮助用户进一步学习。</think>### C++静态成员变量教程 #### 一、静态成员变量的定义与声明 1. **内声明**:使用$static$关键字声明 ```cpp class MyClass { public: static int count; // 声明静态成员变量 }; ``` 2. **外初始化**:必须显式分配存储空间 ```cpp int MyClass::count = 0; // 外初始化(不可加static) ``` *原因*:静态成员变量存储在全局数据区,需独立于对象存在[^1]。 #### 二、初始化规则 - **必须外初始化**:内只能声明,不能赋值 ```cpp // 错误示例:内直接赋值 class MyClass { static int count = 0; }; // 编译错误 ``` - **支持常量表达式初始化**(C++11起) ```cpp class MyClass { static constexpr double PI = 3.1415926; // 允许内初始化 }; ``` #### 三、访问方式 1. **通过名访问**(推荐) ```cpp MyClass::count = 10; // 直接操作静态成员变量 ``` 2. **通过对象访问** ```cpp MyClass obj; obj.count = 20; // 合法但不推荐 ``` #### 四、典型应用场景 1. **对象计数器** ```cpp class Car { public: Car() { ++count; } // 构造函数中递增 ~Car() { --count; } // 析构函数中递减 static int count; // 统计现存对象数量 }; int Car::count = 0; // 初始化为0 ``` 2. **共享配置参数** ```cpp class Logger { public: static bool debug_mode; // 所有日志实例共享调试模式状态 }; ``` #### 五、与普通静态变量的区别 | 型 | 存储位置 | 初始化方式 | 生命周期 | |---------------------|------------|--------------------------|--------------| | 普通静态变量 | 全局数据区 | 定义时自动初始化为0[^3] | 程序运行期间 | | 静态成员变量 | 全局数据区 | 必须显式外初始化[^2] | 程序运行期间 | #### 六、代码示例 ```cpp #include <iostream> using namespace std; class Student { public: Student() { ++total; } static int total; // 声明静态成员变量 }; int Student::total = 0; // 初始化 int main() { Student s1, s2; cout << "学生总数:" << Student::total; // 输出2 return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值