• C/C++ extern和static的使用


    前言

    在讲到extern和static的时候先了解一下定义和声明的基本概念

    定义(define):

    A variable is defined when the compiler allocates the storage for the variable,就是我们的变量个其存储的具体值相关联

    声明(declared)

    编译器声明这个变量的存在,宣告其类型但是并不关联某个存储的具体值

    你可以声明一个变量多次,但是你只能定义其一次并且给一个范围,我们定义一个变量也是声明,但不是所有的声明都是定义

    extern

    我们在全局声明/定义一个变量最好的一个方法是在头文件中用关键字extern声明一个变量
    在我们工程中,一般用一个头文件声明全部所需的全局变量(当然用extern),然后在所有其他的.c文件中include这个头文件,假设我们有三个文件分别是file3.h,file1.c,file2.c
    内容分别如下
    var.h

    extern int global_var;
    
    
    • 1
    • 2

    var.c

    #include "var.h"
    #include "prog1.h" //function declarations,我们示例中没有将函数原型头文件写出来
    
    int global_var = 33;
    
    int increment(void) { return global_variable++ }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    main.c

    #include "var.h"
    #include "prog1.h"
    #include <stdio.h>
    //注意我们没有include file1.c
    
    void use_it(void){
    	printf("global var : %d\n",global_var++);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    然后我们编译 (记住不编译头文件)

    gcc main.c var.c -o out.c
    
    • 1

    为什么我们的main不include var.c就知道global_var的具体值呢?因为我们说过一个全局变量只能定义一次,但是可以声明多次,global_var分别在main,c和var.c中声明了,但是只在var.c中定义,换个角度,global_var的生命周期是全局也就是整个软件的生命周期,整个软件的生命周期包含三个文件,且global_var不定义在堆栈中,而是声明在bss中,定义在initialed data区域中

    static

    static也是全局但是其作用域不是全局而是本文件中,所以其他的文件include一个含有static的头文件,且试图定义他会报错,因为static变量的作用域只在声明他的头文件中

    还是上述的程序但是我们把extern改为static了
    var.h

    static int global_var;
    
    
    • 1
    • 2

    var.c

    #include "var.h"
    #include "prog1.h" //function declarations,我们示例中没有将函数原型头文件写出来
    
    int global_var = 33;
    
    int increment(void) { return global_variable++ }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    main.c

    #include "var.h"
    #include "prog1.h"
    #include <stdio.h>
    //注意我们没有include file1.c
    
    void use_it(void){
    	printf("global var : %d\n",global_var++);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    开始编译发现错误

    c++ static members in class

    简而言之就是我们的class里面搞一个static的成员,我们知道static的作用域虽然是全局只存在于本文件,那么将一个static放在一个class中间是什么意思呢?
    在Cpp的类中使用static就不再和C一样局限于定义的文件中了,在Cpp的class中用static修饰成员有以下的特点

    • 当这个class建立的时候,此class内的static成员在只有一份,无论创建多少个class对象,且每个对象都是共享这个static成员的,换句话说无论多少个对象创建,class的static成员都是第一无二的,且内存中只有一份
    • static成员的初始化发生在此class所有对象创建前
    • 他的声明周期是全程序

    我们写一个程序,写一个class,在其public中搞一个static member,且在class的构造函数中对这个static member + 1,意味着此static成员作用是统计有多少个class对象成员

    static_mamber.h

    using namespace std;
    
    class Box{
    public:
            static int objcount;
    
            Box(double l,double b, double h);
    
            double volume();
    
    private:
            double length;
            double breadth;
            double height;
    
    };
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    static_member.cpp

    #include "static_member.h"
    #include <iostream>
    
    using namespace std;
    
    
    int Box::objcount = 0; //static成员的初始化在创建所有的class对象之前
    //构造函数
    Box::Box(double l, double b, double h)
            :length(l),breadth(b),height(h){
                    cout << "construct is called," << endl;
                    objcount++;
            }
    
    double
    Box::volume(){
            return length * breadth * height;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    main.cpp

    #include "static_member.h"
    #include <iostream>
    
    using namespace std;
    
    
    int main(void){
            Box Box1(3.3,1.2,1.5);
            Box Box2(8.5,6.0,2.0);
    
            cout << "total Box object is "<< Box::objcount << endl;
    
            return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    编译

    g++ static_member.cpp main.cpp  -o static_member.o
    
    • 1

    得到结果

    construct is called,
    construct is called,
    total Box object is 2
    
    • 1
    • 2
    • 3

    注意class是全局的也就是extern的因为在所有block({})外部的变量或者class或者函数如果不加static都默认是extern

    总结

    在C语言中extern修饰后的变量或者函数,可以在其他的文件中进行使用(需要include定义extern变量或者函数的头文件),但是static则不行,static和extern的作用域都是全局但是,static只允许本文件内对其修饰的变量更改,而extern允许在任何文件中更改

    在C++中static修饰的是某个class的一个成员,和C中的static完全不一样,首先C++中如果在头文件中声明某个class的某个成员是static,那么我们在其他文件中可以定义他(前提include对于的头文件),这是在C中是不行的,且C++ static member in clss意思是为此class创建一个独一无二的成员,不论你的class实例化多少次,static成员就一个,其他的class对象都是其copy,并且我们可以随时随地修改这个static成员

  • 相关阅读:
    MongoDB 基础了解(三)
    RK3399平台开发系列讲解(PCI/PCI-E)5.55、PCIE RC枚举EP过程
    【算法】选择排序
    力扣hot100 两数之和 哈希表
    SUID提权教程
    Soa: 一个轻量级的微服务库
    【Log4j2】开发环境配置日志策略打印完整MyBatis语句到文件
    MindSpore-AttributeError: module ‘mindspore‘ has no attribute ‘value_and_grad‘
    深入理解线段树
    将master分支的代码合并到你自己的分支
  • 原文地址:https://blog.csdn.net/qq_37026934/article/details/125448627