静态全局变量

#include <iostream>
using namespace std;

static size_t ctr = 0;
size_t count_calls()
{
return ++ctr;
}

int main()
{
for(size_t i=0; i < 10; ++i)
{
cout << "times:" << count_calls() << endl;;
}
}

特点

  • 该变量在全局数据区分配内存
  • 未经初始化的静态全局变量会被程序自动初始化为0
  • 静态全局变量在声明它的整个文件都是可见的,而在文件之外是不可见的

结果

c++ static_静态数据成员

全局变量和全局静态变量的区别

1)全局变量是不显式用static修饰的全局变量,但全局变量默认是动态的,作用域是整个工程,在一个文件内定义的全局变量,在另一个文件中,通过extern 全局变量名的声明,就可以使用全局变量。

2)全局静态变量是显式用static修饰的全局变量,作用域是声明此变量所在的文件,其他的文件即使用extern声明也不能使用。

 

静态局部变量

#include <iostream>
using namespace std;
size_t count_calls()
{
static size_t ctr = 0;
return ++ctr;
}

int main()
{
for(size_t i=0; i < 10; ++i)
{
cout << "times:" << count_calls() << endl;;
}
}

特点

  • 该变量在全局数据区分配内存
  • 静态局部变量在程序执行到该对象的声明处时被首次初始化,即以后的函数调用不再进行初始化
  • 静态局部变量一般在声明处初始化,如果没有显式初始化,会被程序自动初始化为0
  • 它始终驻留在全局数据区,直到程序运行结束。但其作用域为局部作用域,当定义它的函数或语句块结束时,其作用域随之结束

结果

c++ static_静态数据成员_02

 

 静态函数

#include <iostream>
using namespace std;
static void fn()
{
int n = 10;
++n;
cout << n << endl;
}

int main()
{
cout << "first use:" << endl;
fn();
cout << "second use:" << endl;
fn();
}

使用静态函数好处

  • 静态函数不能被其它文件所用
  • 其它文件中可以定义相同名字的函数,不会发生冲突

结果

c++ static_#include_03

 类中的静态数据成员

#include <iostream>
using namespace std;
class Myclass
{
public:
Myclass(int va, int vb, int vc);
void GetSum();
private:
int a;
int b;
int c;
static int sum;
};

int Myclass::sum = 10;

Myclass::Myclass(int va, int vb, int vc)
{
a = va;
b = vb;
c = vc;
sum += a + b + c;
}

void Myclass::GetSum()
{
cout << "Sum:" << sum << endl;
}

int main()
{
cout << Myclass::sum << endl;
Myclass classa(1, 1, 1);
classa.GetSum();

Myclass classb(1, 1, 1);
classb.GetSum();
}

结果

c++ static_#include_04

特点

  • 静态数据成员在程序中也只有一份拷 贝,由所有对象共享(对于非静态数据成员,每个类对象都有自己的拷贝)
  • 静态数据成员存储在全局数据区。静态数据成员定义时要分配空间,所以不能在类声明中定义
  • 静态数据成员和普通数据成员一样遵从public,protected,private访问规则
  • 静态数据成员初始化与一般数据成员初始化不同。静态数据成员初始化的格式为:数据类型 类名::静态数据成员名 =值

同全局变量相比,使用静态数据成员有两个优势

  • 静态数据成员没有进入程序的全局名字空间,因此不存在与程序中其它全局名字冲突的可能性
  • 可以实现信息隐藏。静态数据成员可以是private成员,而全局变量不能

 

 类中静态成员函数

#include <iostream>
using namespace std;
class Myclass
{
public:
Myclass(int va, int vb, int vc);
static void GetSum();
private:
int a;
int b;
int c;
static int sum;
};

int Myclass::sum = 10;

Myclass::Myclass(int va, int vb, int vc)
{
a = va;
b = vb;
c = vc;
sum += a + b + c;
}

void Myclass::GetSum()
{
cout << "Sum:" << sum << endl;
cout << "Sum:" << a << endl;
}

int main()
{
Myclass classa(1, 1, 1);
classa.GetSum();

Myclass classb(1, 1, 1);
classb.GetSum();
}

结果

c++ static_初始化_05

特点

  • 出现在类体外的函数定义不能指定关键字static
  • 静态成员之间可以相互访问,包括静态成员函数访问静态数据成员和访问静态成员函数(不限制在类中成员)
  • 非静态成员函数可以任意地访问静态成员函数和静态数据成员
  • 静态成员函数不能访问非静态成员函数和非静态数据成员
  • this 是缺省的(因为:静态成员函数由于不是与任何的对象相联系)
  • 由于没有this指针的额外开销,因此静态成员函数与类的全局函数相比速度上会有少许的增长
  • 不能将静态成员函数定义为虚函数

 

一个例子

#include <iostream>
#include <vector>
#include <set>
using namespace std;
class A
{
public:
A(int val) : a(val) {};
static void init_vec();
static void push_a(int//static函数可以改变static成员
static void print_vec()
{
cout << "hello" << vec.size() << endl;
for(vector<int>::iterator beg = vec.begin(); beg != vec.end(); ++beg)
cout << *beg << endl;
}
private:
static vector<int> vec;
int a;
};

vector<int>//在类的外边定义

int main()
{
A a(3);
a.push_a(34);
a.push_a(45);
a.push_a(5);
a.print_vec();
}

 继承

#include <iostream>
#include <string>
using namespace std;

class base
{
public:
static void print_1() { cout << "hello:" << ++val << endl; }
static int val;
};
int base::val = 1;
class derived :public base
{
};


int main()
{
base b;
derived d;
base::print_1();
derived::print_1();
b.print_1();
d.print_1();
}

结果

c++ static_#include_06