文档章节

C++手稿:静态和全局变量的作用域

Lennie002
 Lennie002
发布于 2015/10/17 01:54
字数 1301
阅读 216
收藏 8
点赞 0
评论 1

全局变量和静态变量的存储方式是一样的,只是作用域不同。如果它们未初始化或初始化为0则会存储在BSS段,如果初始化为非0值则会存储在DATA段,见进程的地址空间分配一文。 静态变量的作用域是当前源文件,全局变量的作用域是整个可执行程序。 值得注意的是:

  • 如果在头文件定义全局变量,在预编译期间 #include 的头文件会被拷贝进源文件中,编译器是不知道头文件的。

  • 虽然全局变量是全局作用域,但需要 extern 关键字来声明以通过编译。因为C++是强类型语言,编译时需要根据变量声明做类型检查。

全局变量的引用

C++源文件中引用外部定义的全局变量和引用外部函数是一样的语法,通过extern 来声明:

// file: a.cpp 
#include<iostream> 
extern int a;    //声明
int main() 
{
  std::cout<<a<<std::endl; 
  return 0;
 }
  // file: b.cpp 
  #include<iostream>
   int a = 2;   //定义

然后分别编译这两个文件,链接生成 a.out 并执行它:

$ g++ a.cpp b.cpp 
$ ./a.out
 2

extern 只是在当前文件中声明有这样一个外部变量而已,并不指定它来自哪个外部文件。所以即使 extern 变量名错误当前源文件也能通过编译,但链接会出错。

头文件中定义

因为头文件可能会被多次引用,在预编译时被引用的头文件会被直接拷贝到源文件中再进行编译。一个常见的错误便是把变量定义放在头文件中,例如下面的变量 int a :

// file: a.cpp
#include <iostream> 
#include "b.h"
int main() 
{  
     std::cout<<a<<std::endl;
     return 0;
} 
// file: b.cpp 
#include<iostream> 
#include"b.h"
 void f(){} 
// file: b.h
int a = 2;

头文件 b.h 中定义了 int a ,它被 a.cpp 和 b.cpp 同时引入。我们将a.cpp 和 b.cpp 分别编译是没有问题的,然后链接时会抛出错误:

duplicate symbol _a in:     
      /tmp/ccqpfU5e.o     
      /tmp/ccCRi9nO.o
ld: 1 duplicate symbol for architecture x86_64 collect2: error: ld returned 1 exit status

两个 .o 文件中的 _a 名称发生了冗余,这是变量重定义错误。

头文件中声明

因为声明操作是幂等的,而多次定义会引发重定义错误。所以 头文件中不应包含任何形式的定义,只应该包含声明 , 正确的办法是变量定义总是在源文件中进行,而声明放在头文件中:

#include <iostream> 
#include "b.h" 
int main()
{  
    std::cout<<a<<std::endl;  
    return 0;
} 
// file: b.cpp 
#include<iostream> 
#include"b.h" 
int a = 2;    //定义
// file: b.h
extern int a;   //声明 extern

然后编译链接执行都会通过,输出 2 :

$ g++ a.cpp b.cpp 
$ ./a.out 
 2

编译器看到 g++ a.cpp b.cpp 时会自动去查找 b.h 并进行预编译操作,因此不需要显式指定 b.h 。

静态全局变量

非静态全局变量是 外部可链接的 (external linkage),目标文件中会为它生产一个名称供链接器使用; 而静态全局变量是 内部可链接的 (internal linkage),目标文件中没有为链接器提供名称。因此无法链接到其他文件中,因此静态变量的作用域在当前源文件(目标文件)。 虽然静态和非静态全局变量可能存储在同一内存块,但它们的作用域是不同的。 来看例子:

// file: a.cpp 
#include <iostream>
 extern int a; 
 int main() 
 {  
      std::cout<<a<<std::endl;  
      return 0; 
 } 
// file: b.cpp 
static int a = 2;

然后 g++ a.cpp b.cpp 时发生链接错:

Undefined symbols for architecture x86_64:   
     "_a", referenced from:       
     _main in ccPLYjyx.o 
 ld: symbol(s) not found for architecture x86_64 collect2: error: ld returned 1 exit status

链接时未找到名称 _a ,因此静态变量在编译得到的目标文件中没有为链接器提供名称。所以其他目标文件无法访问该变量,静态全局变量的作用域是当前源文件(或目标文件)。

全局变量初始化

全局变量比较特殊,初始化有两种方式:

  • 静态初始化(static initialization):对于定义时给出初始化参数的全局变量,其初始化在程序加载时完成。根据是否被初始化、是否被初始化为0会把它存储在BSS或者DATA段中,参见进程的地址空间分配。

  • 动态初始化(dynamic initialization):定义变量时可以不给出初始化参数,而是在某个函数中进行全局变量初始化。

对于静态初始化,看这个例子:

class C
{ 
public:     
      C()
      {
           std::cout<<"init "; 
      } 
}; 
C c;
int main()
{
    std::cout<<"main"; 
    return 0; 
}

在 main() 进入之前,程序加载时动态初始化,程序输出为一行 init main 。

关于 全局变量的初始化顺序 ,同一源文件中的全局变量初始化顺序按照定义顺序,不同源文件(编译单元)的全局变量初始化顺序并未规定。 因此软件设计时不要依赖于其他编译单元的静态变量,可以通过单例模式来避免这一点。


本文转载自: http://harttle.com/2015/09/30/cpp-static-and-global.html

共有 人打赏支持
Lennie002
粉丝 8
博文 118
码字总数 63785
作品 0
大连
加载中

评论(1)

木叶粗眉
木叶粗眉
[4]
c++变量与函数的作用域与生存期

局部变量 范围:在一个函数内部定义的变量,作用范围只限于本函数体内。 生存期:程序执行到本函数才会给局部变量分配内存单元,函数执行完毕局部变量所占的存储单元就被释放 静态局部变量 ...

memristor
2014/05/06
0
0
C/C++中的static和extern关键字

一.C语言中的static关键字 在C语言中,static可以用来修饰局部变量,全局变量以及函数。在不同的情况下static的作用不尽相同。 (1)修饰局部变量 一般情况下,对于局部变量是存放在栈区的,并...

木木情深
2014/02/13
0
0
浅谈C/C++中的static和extern关键字

static是C++中常用的修饰符,它被用来控制变量的存贮方式和可见性。extern, "C"是使C++能够调用C写作的库文件的一个手段,如果要对编译器提示使用C的方式来处理函数的话,那么就要使用exter...

dclink
2014/03/31
0
2
C++中头文件、源文件之间的区别与联系

.h头文件和.cpp文件的区别 疑惑1:.h文件能够编写main函数吗? 实验: 编写test.h文件,里面包含main函数 若直接编译g++ test.h -o test,通过file命令 file test,得到如下结果test: GCC p...

风筝Fergus
2013/04/17
0
2
C语言关键---static

static 这个关键字,我感觉对很多初级程序源和刚接触程序的同学来说,可能都没有完全理解它的含义。至少对我来说对它的认识不是很清楚。因此对它在次总结一下。主要作为自己的备忘录。 1.st...

jackxu2015
2015/08/14
0
0
javascript引擎在c,c+中调用

JavaScript是一种广泛用于Web客户端开发的脚本语言,常用来控制浏览器的DOM树,给HTML网页添加动态功能。目前JavaScript遵循的web标准的是ECMAScript262。由于JavaScript提供了丰富的内置函数...

crossmix
2015/04/19
0
0
C语言编程学习中,全局变量与局部变量同名时,如何判断

C语言是面向过程的,而C++是面向对象的 C和C++的区别: C是一个结构化语言,它的重点在于算法和数据结构。C程序的设计首要考虑的是如何通过一个过程,对输入(或环境条件)进行运算处理得到...

小辰带你看世界
05/10
0
0
7、变量的生存期和作用域

在C语言中,程序可在任意花括号后定义变量,在C++中,允许在程 序中任何地方声明变量,它的优点是可在使用该变量附近定义他。当在一个块内定义了一个变量,这个变量的作用域是从定义点起到当...

hopegrace
2010/08/03
0
0
const、static关键字

最近一直在迷惑着这两个关键字,感觉大脑里一片糊涂,特此整理一下: 一、const 1、数据常量,不能修改,必须初始化; 1)非类的数据成员 a、指向const对象的指针(remember) b、const指针(...

EillenMe
2013/04/11
0
0
[C++再学习系列] 全局或静态变量(对象)的初始化

  对于C语言的全局和静态变量,不管是否被初始化,其内存空间都是全局的;如果初始化,那么初始化发生在任何代码执行之前,属于编译期初始化。由于内置变量无须资源释放操作,仅需要回收内...

技术小美
2017/11/05
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

expect脚本同步文件、expect脚本指定host和要同步的文件、构建文件分发系统

expect脚本同步文件 更改权限 执行脚本 查看执行结果 expect eof需要加上,作用是等脚本命令执行完再进行退出 expect脚本指定host和要同步的文件 更改权限,执行脚本 构建文件分发系统 需求背...

Zhouliang6
28分钟前
1
0
Hive应用:外部分区表

Hive应用:外部分区表 介绍 Hive可以创建外部分区表。创建表的时候,分区要在建表语句中体现。建完之后,你不会在表中看到数据,需要进行分区添加,使用alter语句进行添加。然后数据才会显示...

星汉
38分钟前
2
0
点击Enter登录

1. 效果 2. 实现过程(记得引入jq文件) //6.回车事件 登录 $(function() { document.onkeydown = function(event) { var e = event || window.event || arguments.callee.caller.arguments......

Lucky_Me
44分钟前
1
0
点击菜单内容切换

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <style> .menu{ height: 38px; background-color: #eeeeee; line-height: 38px; } .mao{ ......

南桥北木
今天
1
0
OSChina 周六乱弹 —— 妹子和游戏哪个更好玩

Osc乱弹歌单(2018)请戳(这里) 【今日歌曲】 @andonny :分享唐朝乐队的单曲《国际歌》 《国际歌》- 唐朝乐队 手机党少年们想听歌,请使劲儿戳(这里) @举个栗子- :日常祈雨 邪恶的大祭...

小小编辑
今天
522
6
流利阅读笔记32-20180721待学习

“人工智能”造假:只有人工,没有智能 Lala 2018-07-21 1.今日导读 当今社会,擅长单个方面的人工智能已经盛行,手机借助 AI 智慧防抖技术帮助大家拍出清晰照片,谷歌研发的 AI 助手将可以帮...

aibinxiao
今天
7
0
我的成长记录(一)

今天突然精神抖擞,在我的博客下新开一项分类>成长记录,专门记录每隔一段时间我的一点感悟吧。因为今天才专门花时间新开这样一个分类,所以以前有过的一些感悟没有记录下来,现在已经想不起...

dtqq
今天
1
0
机器学习管理平台 MLFlow

最近工作很忙,博客一直都没有更新。抽时间给大家介绍一下Databrick开源的机器学习管理平台-MLFlow。 谈起Databrick,相信即使是不熟悉机器学习和大数据的工程湿们也都有所了解,它由Spark的...

naughty
今天
15
0
idea tomcat 远程调试

tomcat 配置 编辑文件${tomcat_home}/bin/catalina.sh,在文件开头添加如下代码。    CATALINA_OPTS="-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=7829" Idea端配......

qwfys
今天
2
0
遍历目录下的文件每250M打包一个文件

#!/usr/bin/env python # -*- utf-8 -*- # @Time : 2018/7/20 0020 下午 10:16 # @Author : 陈元 # @Email : abcmeabc@163.com # @file : tarFile.py import os import tarfile import thr......

寻爱的小草
今天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部