static用法
a.静态局部变量,成为静态局部变量(拥有记忆功能和全局存储权限)
b.静态全局变量(限制对应全局变量被被其他文件调用)
c.静态函数
d.静态类成员(标识此成员属于类而非属于某个特定对象)
1.静态局部变量
1.1静态局部变量在函数内定义,拥有静态存储期限而不再是自动存储期限,因为静态存储期限的变量拥有永久的
存储单元,所以在整个程序存储期间都会保留变量的值,尽管退出函数后变量依然存在,但不能使用。
1.2静态局部变量始终拥有块作用域。(与自动变量作用域相同)
1.3对基本类型的静态局部变量在定义时若为赋初值,则系统自动赋值为0.
1.4举例如下:
void static_local_variable()
{
static int count = 0;
count++;
}
第一次进入此函数,静态变量count被初始化为0(若不初始化,系统会自动初始化为0),接下来执行count++。
而之后调用此函数则只执行count++.
此函数与以下代码实现同样的功能:
int count = 0;
void fuc()
{
count++;
}
2.静态全局变量
2.1静态存储期限。如同声明为static的局部变量一样,外部变量拥有静态存储期限,存储在外部变量中的值奖
被永久的保留下来。
2.2文件作用域。外部变量拥有文件作用域;从变量被声明的位置开始,一直到所在文件的末尾。因此,跟随在
外部变量之后声明的所有函数都可以访问。
2.3举例如下:
//xxxx.cpp
void fucOne()
{
sCount++;
count++;
}
static int sCount = 0;
int count = 0;
void fucTwo()
{
sCount++;
count++;
}
a.fucOne()定义在sCount和count变量之前,编译时会提示未声明标示符。而fucTwo()定义在两变量之后,在
fucTwo()中则可正常使用。且在其他.cpp文件中不可访问。
b.若两个.cpp文件声明了同名的全局变量,其本质为生命了两个独立的静态全局变量。
c.count变量则可共享,在xxx.h文件中对其声明 extern int count;在任何一个包含此头文件的.cpp文件中初始化一次即可共享此全局变量。而同样的方法却不适用于静态变量。
(强烈建议:不要将静态/静态变量定义在.h文件中)
3.static函数
3.1内部函数(static函数,不可再其他文件中调用)
如果在一个原文件中定义的函数只能被此源文件使用,则在函数类型前加 static 关键字即可。(利用了
static的文件作用域)
举例如下:
static void fuc()
{
}
3.2外部函数(可在其他文件中调用)
举例如下:
//file.cpp
[extern] void fuc()
{
}
//main.cpp
int main()
{
extern void fuc();
}
4.静态类成员
4.1 类的静态成员可以是private的或public的。静态数据成员可以是常量、指针、引用、类类型等。
4.2 其静态成员属于类而非某个特定对象。其静态成员函数中也不能使用this指针。
4.2 类静态成员的使用
a.定义并初始化
一般来说,不能在类的内部初始化静态数据成员,相反的,必须在类的外部定义和初始化每个静态数据成
员。
举例如下:
//xxxx.h
class A
{
private:
static int count;//声明
public:
static void fuc();
}
int A::count = 0;//定义、初始化
(注意:最好将静态数据成员的定义初始化放在对应的.cpp文件中,若像如上定义,在此头文件被多次包含时,静态成员会被重复定义)
b.类内初始化
如果某个数据成员的应用场景仅限于编译器可以替换它的情况,则一个初始化的const或constexpr static不需要分别定义.
举例如下:
class Account
{
public:
static doule fuc(){};
private:
static constexpr int NUM = 30;
int array[NUM];
}
相反若它将用于值不能替换的场景中,则该成员必须有一条定义语句:
举例如下:
constexpr int Account::NUM;//只定义,不初始化,初始化已在类内提供
(建议:即使一个常量静态数据成员在类内被初始化了,通常情况下也应该在类外定义一下该成员)
c.静态成员的两个应用
(1)静态成员可以是不完全类型(不完全类型:只声明,未定义)
(2)可以用静态成员作为默认实参