有时候我们会不给C++类成员变量赋初始值,或是因为忘记在构造函数中指定(C++11可以写在类内),或是觉得没有必要写。然而,因为觉得编译器会把变量赋成0而不写是错误的。本文通过C++标准来解释这个问题。

本文基于N3337(C++11草案)标准。

关于没有初始化器的对象,在8.5-11中有提及:

If no initializer is specified for an object, the object is default-initialized; if no initialization is performed, an object with automatic or dynamic storage duration has indeterminate value.

没有初始化器的对象会被默认初始化;没有初始化的自动(局部变量)或动态存储期限(new出来的)对象的值是未定的。

这里涉及到了两种“无初始化”的概念,没有初始化器与没有初始化,注意区分。8.5-6对默认初始化(default-initialize)的定义是:

To default-initialize an object of type T means:

  • if T is a (possibly cv-qualified) class type (Clause 9), the default constructor for T is called (and the initialization is ill-formed if T has no accessible default constructor);

  • if T is an array type, each element is default-initialized;

  • otherwise, no initialization is performed.

默认初始化T类型的对象是指:

  • 如果T是(可能有constvolatile的)类类型,T的默认构造函数会被调用(如果没有可用的,初始化就是非法的)(默认构造函数可能会对成员执行默认初始化);

  • 如果T是数组类型,每个元素被默认初始化(同样是递归的默认初始化);

  • 否则,不初始化。

从这些标准中的条款,可以得出结论:自动或动态存储期限的非类类型对象,无论是否是数组或是否有const修饰,如果不指定初始值,它的值就是未定的。

而整数类型、指针类型等都属于非类类型,如果我们希望这些类型的成员变量有确定的初始值,即使是看起来默认的0,也要自己写上初始化

----------------分割线----------------

我在这个问题上栽过大跟头,这就是我要写这篇文章的原因。

那是一个单片机的C++程序,涉及到一个二重指针数组,类型是一个定义了虚函数的基类,通过指针调用虚函数。数据结构并不复杂,逻辑肯定不会出错,但是程序跑飞了。

一开始觉得是初始化的问题,就把所有全局变量换成单例模式创建,无果。然后加了许多调试语句,把问题定位到了解引用上。想了想指针解引用肯定不会出问题,就觉得问题在虚函数调用过程中的函数指针解引用上。把指针调用(动态绑定)换成对象调用(静态绑定),果然程序就正常了。但函数指针解引用本质上是修改PC(程序计数器)寄存器,这种编译器搞定的事情我也插不了手。换了新版本的编译器,也没有解决。

后面的探索就奇怪了起来。我发现给PCB加一个100uF的电容可以让程序在烧写后正常运行,但在重新上电后,一旦程序去调用那虚函数,单片机就会复位;如果把业务逻辑改简单一点,有助于重新上电后正常工作,但没有本质上解决问题。用了一些猥琐操作后,我发现不是单片机复位而是程序回到起始处。这可能是未注册而触发的中断导致的,但就算给所有中断都指定了空函数,也还是没有解决。

百思不得其解,调试了一整天都没有成功。

几个月后再来看这个中断的项目,重新读了一遍代码,发现了这个初始值的问题,终于解决了。后面就很顺了。

现在看来这个debug的过程一开始方向挺对的,后面的探索就慢慢差到十万八千里外去了。

烧写正常运行、上电不正常的奇怪现象也可以解释了。单片机内置复位电路会在上电时给寄存器赋初值,但内存中的数据还是随机的。程序烧写前后内存中的数据被保留,之前的程序不知怎么把那一块内存初始化好了,烧写后就可以正常工作;而上电后那一块内存里都是随机值,对随机数进行4次解引用(二重指针2次,虚函数调用2次),程序早就不知道跑哪里去了。可能是寻到合法地址空间以外去了,由于某些保护机制的存在,程序就回到了起始处。

C++类成员默认初始值的更多相关文章

  1. C++ 变量默认初始值不确定(代码测试)

    C++ int变量默认初始值是不确定的,因此使用时初始化是很有必要的. 下面写个小程序测试一下int变量默认初始值. #include <iostream> #include <ve ...

  2. Java中在实例化一个类时,这个类中没有初始值的int类型成员变量i,i的值是不是0?

    java中有两种类型一种是数值性,另一种是类变量数值性变量的初始值为0,类变量的初始化为null没做初始化成员变量int性变量是0, 在java中有这么一条规则,声明在方法中的变量在使用时必须要初始化 ...

  3. Java语言基础(六)char成员变量默认初始值 最简单的Java源文件 Java的main()方法

    ①char成员变量的初始值是:'\u0000' ②package用来指定该文件所处的包的名称,必须位于源文件的顶端. import java.util.*; package com.hyy.test; ...

  4. 【C++】不要依赖编译器的默认初始值

    最好在定义的时候就给出初始值. 类和结构体给出构造函数. 比如int,在vs的debug和release模式下,初始化的值是不同的.

  5. Java未赋值变量的默认初始值

    在 Java 程序中,任何变量都必须经初始化后才能被使用.当一个对象被创建时,实例变量在分配内存空间时按程序员指定的初始化值赋值,否则系统将按下列默认值进行初始化: 数据类型 初始值 byte 0 s ...

  6. 【转】Lombok Pojo默认初始值问题

    Lombok以注解形式来简化java代码,提高开发效率.比如我们常用的@Builder.@Data.@AllArgsConstructor.@NoArgsConstructor.@ToString等. ...

  7. Lombok Pojo默认初始值问题

    Lombok以注解形式来简化java代码,提高开发效率.比如我们常用的@Builder.@Data.@AllArgsConstructor.@NoArgsConstructor.@ToString等. ...

  8. 126、Java面向对象之引用传递实例四,修改类成员的属性值

    01.代码如下: package TIANPAN; class Message { private String info = "此内容无用"; // 定义String类型属性 p ...

  9. Select2实现的带搜索的省市区三级联动代码 设置默认初始值

    $(function() { $('#loc_province').select2('val','2456'); $('#loc_province').change(); $('#loc_city') ...

随机推荐

  1. websocket可以做什么

    本篇介绍的是websocket,但是并不介绍它的协议格式,一般能看明白http头也能明白websocket在协议切换前的协商,能看明白IP报头也就对websocket在协议切换后通讯格式不陌生.web ...

  2. pycharm设置python脚本模板

    PyCharm PyCharm是一个有名的Python IDE,带有一整套可以帮助用户在使用Python语言开发时提高其效率的工具,比如调试.语法高亮.Project管理.代码跳转.智能提示.自动完成 ...

  3. electron——dialog(实现导出excel)

    背景 前端点击导出excel按钮后,请求完需要导出的数据后发送给主进程electron,由主进程保存到本地 dialog 显示用于打开和保存文件.警报等的本机系统对话框. dialog 模块提供了ap ...

  4. SpringSecurity动态加载用户角色权限实现登录及鉴权

    很多人觉得Spring Security实现登录验证很难,我最开始学习的时候也这样觉得.因为我好久都没看懂我该怎么样将自己写的用于接收用户名密码的Controller与Spring Security结 ...

  5. 用Helm3构建多层微服务

    Helm是一款非常流行的k8s包管理工具.以前就一直想用它,但看到它产生的文件比k8s要复杂许多,就一直犹豫,不知道它的好处能不能抵消掉它的复杂度.但如果不用,而是用Kubectl来进行调式真的很麻烦 ...

  6. day 23 面向对象中类的成员 和嵌套

    1.类的成员? 变量.方法.属性 class Foo: # 方法 def __init__(self,name): # 实例变量/字段 self.name = name # 方法 def func(s ...

  7. Mybatis拦截器实现原理深度分析

    1.拦截器简介 拦截器可以说使我们平时开发经常用到的技术了,Spring AOP.Mybatis自定义插件原理都是基于拦截器实现的,而拦截器又是以动态代理为基础实现的,每个框架对拦截器的实现不完全相同 ...

  8. master节点的部署介绍和前置工作

    目录 组件介绍 组件介绍 kubernetes master节点运行组件如下: kube-apiserver.kube-scheduler.kube-controller-manager.kube-n ...

  9. Linux进阶文档丨阿里架构师十年Linux心得,全在这份文档里面

    Linux是什么 Linux就是个操作系统: 它和Windows XP.Windows 7.Windows 10什么的一样就是一个操作系统而已! Linux能干什么: 它能当服务器,服务器上安装者各种 ...

  10. 教你如何关闭IIS服务

    由于IIS服务器和Apache的默认端口号都是80端口,有时我们需要关闭IIS服务,下面讲讲关闭IIS服务的方法.   方法如下:   1.services.msc,在里面找到一个“World Wid ...