到目前为止,我们设计的类中所有的成员变量和成员函数都是属于对象的,如我们在前面定义的book类,利用book类声明两个对象Alice和Harry,这两个对象均拥有各自的price和title成员变量,同时还拥有类中所有的成员函数。

除了这种情况以外,我们还有另外一种类型的成员,那就是与static结合的成员变量和成员函数。类中的成员变量或成员函数一旦与static关键字相结合,则该成员变量或成员函数就是属于类的,而不是再是属于任何一个对象的,当然任何一个对象都可以共享该成员变量及成员函数。

静态成员变量

静态成员变量声明非常简单,只需要将static关键字加在成员变量声明的前面即可,如例1所示,我们在例中声明了一个静态成员变量count,并将其设置为private属性。设计这个count变量主要是为了统计当前存活的student类对象的个数,当然这个类并不完善,通过这个例子,我们可以了解静态成员变量的声明语法。

class student
{
public:
student(){count ++;}
~student(){count --;}
private:
static int count;
//其它成员变量
};

静态成员变量在类内部声明之后还需要进行定义操作。

例2:

class student
{
public:
student(){count ++;}
~student(){count --;}
private:
static int count;
//其它成员变量
};
int student::count = 0;

class student
{
public:
student(){count ++;}
~student(){count --;}
private:
static int count;
//其它成员变量
};
int student::count = 0;

请仔细查看例2的代码,这段代码虽然与例1相比只是增添了一行代码,却是有几处需要我们特别留心。首先静态成员变量的定义必须在任何程序块之外;其次调用该变量的时候可以直接用类名加上域解析符“::”加上变量名的形式,这是静态成员变量特有的引用方式;在类外部进行定义的时候static关键字是不需要的。

C++语法中规定静态成员变量会被默认初始化为0,类外定义可有可无。而实际上在一些编译器中,如果不加上类外的定义,会出现一些不可知的情况,故在实际设计程序的时候最好还是将类外定义加上。

静态成员变量不会影响类及其对象的大小,也即sizeof结果不会受到影响。在上面的例2中,无论我们是否声明count这个静态成员变量,sizeof(student)或者sizeof(student的对象)其结果都是不会变的。

静态成员变量属于类而不属于任何一个对象,如此一来可以实现数据共享功能,如例3所示。

#include<iostream>
using namespace std;

class test
{
public:
static int num;
};
int test::num = 1;

int main()
{
test one;
test two;
test three;
cout<<test::num<<" "<<one.num<<" "<<two.num<<" "<<three.num<<endl;
test::num = 5;
cout<<test::num<<" "<<one.num<<" "<<two.num<<" "<<three.num<<endl;
one.num = 8;
cout<<test::num<<" "<<one.num<<" "<<two.num<<" "<<three.num<<endl;
two.num = 4;
cout<<test::num<<" "<<one.num<<" "<<two.num<<" "<<three.num<<endl;
three.num = 2;
cout<<test::num<<" "<<one.num<<" "<<two.num<<" "<<three.num<<endl;
return 0;
}

程序运行结果:
1 1 1 1
5 5 5 5
8 8 8 8
4 4 4 4
2 2 2 2

在本例中,为了方便样式static成员变量的共享特性,我们将静态成员变量设为public属性,如此一来在类外可以方便调用。在类外我们先将静态成员变量进行定义,并初始化为1。在主函数中我们为test类定义了三个对象,分别为one、two和three。之后分别用类名和对象名调用该静态成员变量修改其值,并各自调用的结果打印出来。从程序运行结果可以看出,四种调用静态成员变量的方法,其值都是相等的,如果其中有任何一个修改该静态成员变量,所有其他的调用静态成员变量都会跟着一起改变。这就是静态成员变量的共享特性,静态成员变量不属于任何对象,但是可以通过对象访问静态成员变量。静态成员变量属于类,因此可以通过类来调用静态成员变量。静态成员变量如果被设置为private或protected属性,则在类外同样无法访问,但定义该变量的时候却不受此限制,如例2所示,虽然静态成员变量count为private属性,但是它在类外定义的时候不受private限制。

静态成员函数

在类内除了能用static声明静态成员变量外,同样可以使用static声明静态成员函数,静态成员函数只能访问static成员变量。

例4:

#include<iostream>
using namespace std;

class test
{
public:
test(int a, int b){num = a; plus = b;}
static int getnum(){return num;}
static int add(){return num+plus;} //compile error
void setnum(int a){num = a;}
void setplus(int a){plus = a;}
private:
static int num;
int plus;
};
int test::num = 1;

int main()
{
test one;
one.setnum(5);
cout<<test::getnum()<<endl;
return 0;
}

在本例程中,类test内有一个静态成员变量num,一个普通成员变量plus,这两个变量都是private属性,成员函数有一个构造函数,两个普通成员函数和两个静态成员函数。在构造函数中,我们引用了静态成员变量num,这是允许的,同样在setnum函数中,同样引用了静态成员变量num,这个也是允许的。然而在add函数中我们不仅引用了静态成员变量num,同时还访问了非静态成员变量plus,而这是不允许的。静态成员函数只能访问静态成员变量,而不能访问非静态成员变量。普通成员函数(包括构造函数和析构函数)既可以访问普通成员变量,同时又可以访问静态成员变量。

访问静态成员变量和静态成员函数均有两种方式,其一是和普通的成员变量成员函数相同,通过对象来访问,其二则是可以通过类名加上域解析操作符访问。当然访问过程中仍然要遵循private、protected和public关键字的访问权限限定。访问静态成员变量和静态成员函数首选的方法是通过类来访问,毕竟静态成员变量和静态成员函数都是属于类的,与类相关联,而不是属于类的对象。普通成员变量或成员函数不可以通过类来访问。由于静态成员变量和静态成员函数都是属于类,而不是属于对象,因此静态成员函数内部也不存在this指针,因为静态成员函数不属于对象。

在静态成员函数内部可以声明静态变量,注意不是静态成员变量。如果在静态成员函数内部声明一个静态变量,则该类的所有对象将共享这个变量。

例5:

#include<iostream>
using namespace std;

class test
{
public:
static void add(int a);
};

void test::add(int a)
{
static int num = 0;
int count = 0;
num += a;
count += a;
cout<<num<<" "<<count<<endl;
}

int main()
{
test one,two,three;
one.add(5);
two.add(4);
three.add(11);
return 0;
}

程序运行结果:
5 5
9 4
20 11

为了直截了当地说明这个问题,我们只在类test中声明了一个静态成员函数add,并且在类外对这个函数进行了定义。注意,在类外定义静态成员函数是不需要static关键字的。在add函数内部我们定义了一个静态变量num,并且初始化为0,为了增强对比效果,我们又定义了一个普通的变量count。在主函数中定义了三个变量,分别调用静态成员函数,结果num值打印结果是累加的,而count则每次都是从0开始的,并不是累加的。通过对比我们很容易看出,static是被这三个对象同时共享的,三个对象一份数据。

C++类与static的更多相关文章

  1. 高级类特性----static关键字

    static 关键字 当我们编写一个类时,其实就是在描述其对象的属性和行为,而并没有产生实质上的对象,只有通过new关键字才会产生出对象,这时系统才会分配内存空间给对象,其方法才可以供外部调用. 我们 ...

  2. java 类中 static 的使用

    在类中 static 主要修饰变量,方法及代码块.大致的执行和使用,据个人理解如下: 1.修饰变量: 在修饰变量时,如 ,表示该变量是静态变量,也可称为类变量.当当前变量是静态变量时,该变量被该类的所 ...

  3. String类、static、Arrays类、Math类

    String类.static.Arrays类.Math类 String类.static.Arrays类.Math类 String类.static.Arrays类.Math类 String类.stati ...

  4. Java类中static的用法

    关于Java中static的使用有以下四种情况: 1.静态成员变量         被static修饰的成员变量,叫静态成员变量或类变量:没有被static修饰的变量,叫实例变量. 两者的区别是:  ...

  5. 千万别在Java类的static块里写会抛异常的代码!

    public class Demo{ static{ // 模拟会抛异常的代码 throw new RuntimeException(); } } 如果你在Java类的static块里写这样会抛异常的 ...

  6. C++11类内static成员变量声明与定义

    众所周知,将一个类内的某个成员变量声明为static型,可以使得该类实例化得到的对象实现对象间数据共享. 在C++中,通常将一个类的声明写在头文件中,将这个类的具体定义(实现)写在cpp源文件中. 因 ...

  7. 01 语言基础+高级:1-3 常用API第一部分_day08【String类、static、Arrays类、Math类】

    day08[String类.static.Arrays类.Math类] String类static关键字Arrays类Math类 教学目标能够使用String类的构造方法创建字符串对象能够明确Stri ...

  8. Java | 静态嵌套类(Static Nested Class)

    前言 本文内容主要来自 Java 官方教程中的<嵌套类>章节. 本文提供的是 JDK 14 的示例代码. 定义 静态嵌套类(Static Nested Class),是 Java 中对类的 ...

  9. Java(133-151)【String类、static、Arrays类、Math类】

    1.字符串概述和特点 string在lang包里面,因此可以直接使用 字符串的内容不可变 2.字符串的构造方法和直接创建 三种构造方法 package cn.itcast.day08.demo01; ...

  10. 类的static成员并用其实现一个单例模式

    对于特定类型的全体对象而言,有时候可能需要访问一个全局的变量.比如说统计某种类型对象已创建的数量.如果我们用全局变量会破坏数据的封装,一般的用户代码都可以修改这个全局变量,这时我们可以用类的静态成员来 ...

随机推荐

  1. Spring3.2 + Hibernate4.2

    Spring3.2 + Hibernate4.2 前三篇随笔中介绍了 用原生的JDBC访问数据库.一种高效的数据库连接池druid.用Spring的JDBC框架访问数据库. 本文继续介绍第三种数据库访 ...

  2. 关闭Windows 2008下面应用程序出错后的提示

    写了一个服务器端程序,没有能处理所有的错误,总有一些错误会抛出到系统中去.于是写了一个进程守护者,一旦发现服务器端退出,可以在第一时间重新启动服务器,也算是一种折中的方案吧.理论上讲应该是可行的,但是 ...

  3. 关于props和state以及redux中的state

    React的数据模型分为共有数据和私有数据,共有数据可以在组件间进行传递,私有数据为当前组件私有.共有数据在React中使用props对象来调用,它包含标签所有的属性名称和属性值,props对象有三个 ...

  4. Fragment与Activity交互(使用Bundle)

    将需要传输的数据封装在一个Bundle对象里,然后将该Bundle对象通过 fragment.setArguments()放到fragment内. Bundle arguments = new Bun ...

  5. Libevent核心原理

    Libevent 是一个事件驱动框架, 不能仅说他是一个网络库. notejs就是采用与libevent类似的libev来做核心驱动的.   Libevent支持三种事件:io事件.信号事件.时间事件 ...

  6. python3.5学习之路_day1_login

    登录程序1.输入用户名密码2.认证成功后显示欢迎信息3.输错三次后锁定 #!/usr/bin/env python #_*_coding:utf-8_*_ #by anthor zhangxiaoyu ...

  7. SQL SERVER BUG--Alwayson日志备份报错

    数据库版本 SQL SERVER 2012 企业版,版本号:11.0.5582.0 问题场景: 数据库配置Alwayson环境,同机房2节点同步自动切换+跨机房异步,在异步机房中选取同一节点做完整备份 ...

  8. 【成长之路】【python】python基础1

    1.python的优点 高级语言:不需考虑底层实现的细节 可移植性:python程序不需经过任何修改就可以在所有的平台系统上运行 可扩展性:可以把用c和c++实现的代码嵌到python中 可嵌入性:可 ...

  9. linux添加swap

    一. 起因 发现云服务器重装后没有swap分区 于是我们分出一些物理内存做swap分区.. 二. 经过 (1)创建块文件 sudo dd if=/dev/zero of=/tmp/big_swap b ...

  10. Java动态代理简单应用

    概念 代理模式是基本的设计模式之一,它是开发者为了提供额外的或不同的操作,而插入的用来代替“实际”对象的对象.这些操作通常涉及与“实际”对象的通信,因此代理通常充当着中间人的角色. Java动态代理比 ...