C++类成员默认初始值
有时候我们会不给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是(可能有const和volatile的)类类型,T的默认构造函数会被调用(如果没有可用的,初始化就是非法的)(默认构造函数可能会对成员执行默认初始化);如果
T是数组类型,每个元素被默认初始化(同样是递归的默认初始化);否则,不初始化。
从这些标准中的条款,可以得出结论:自动或动态存储期限的非类类型对象,无论是否是数组或是否有const修饰,如果不指定初始值,它的值就是未定的。
而整数类型、指针类型等都属于非类类型,如果我们希望这些类型的成员变量有确定的初始值,即使是看起来默认的0,也要自己写上初始化。
----------------分割线----------------
我在这个问题上栽过大跟头,这就是我要写这篇文章的原因。
那是一个单片机的C++程序,涉及到一个二重指针数组,类型是一个定义了虚函数的基类,通过指针调用虚函数。数据结构并不复杂,逻辑肯定不会出错,但是程序跑飞了。
一开始觉得是初始化的问题,就把所有全局变量换成单例模式创建,无果。然后加了许多调试语句,把问题定位到了解引用上。想了想指针解引用肯定不会出问题,就觉得问题在虚函数调用过程中的函数指针解引用上。把指针调用(动态绑定)换成对象调用(静态绑定),果然程序就正常了。但函数指针解引用本质上是修改PC(程序计数器)寄存器,这种编译器搞定的事情我也插不了手。换了新版本的编译器,也没有解决。
后面的探索就奇怪了起来。我发现给PCB加一个100uF的电容可以让程序在烧写后正常运行,但在重新上电后,一旦程序去调用那虚函数,单片机就会复位;如果把业务逻辑改简单一点,有助于重新上电后正常工作,但没有本质上解决问题。用了一些猥琐操作后,我发现不是单片机复位而是程序回到起始处。这可能是未注册而触发的中断导致的,但就算给所有中断都指定了空函数,也还是没有解决。
百思不得其解,调试了一整天都没有成功。
几个月后再来看这个中断的项目,重新读了一遍代码,发现了这个初始值的问题,终于解决了。后面就很顺了。
现在看来这个debug的过程一开始方向挺对的,后面的探索就慢慢差到十万八千里外去了。
烧写正常运行、上电不正常的奇怪现象也可以解释了。单片机内置复位电路会在上电时给寄存器赋初值,但内存中的数据还是随机的。程序烧写前后内存中的数据被保留,之前的程序不知怎么把那一块内存初始化好了,烧写后就可以正常工作;而上电后那一块内存里都是随机值,对随机数进行4次解引用(二重指针2次,虚函数调用2次),程序早就不知道跑哪里去了。可能是寻到合法地址空间以外去了,由于某些保护机制的存在,程序就回到了起始处。
C++类成员默认初始值的更多相关文章
- C++ 变量默认初始值不确定(代码测试)
C++ int变量默认初始值是不确定的,因此使用时初始化是很有必要的. 下面写个小程序测试一下int变量默认初始值. #include <iostream> #include <ve ...
- Java中在实例化一个类时,这个类中没有初始值的int类型成员变量i,i的值是不是0?
java中有两种类型一种是数值性,另一种是类变量数值性变量的初始值为0,类变量的初始化为null没做初始化成员变量int性变量是0, 在java中有这么一条规则,声明在方法中的变量在使用时必须要初始化 ...
- Java语言基础(六)char成员变量默认初始值 最简单的Java源文件 Java的main()方法
①char成员变量的初始值是:'\u0000' ②package用来指定该文件所处的包的名称,必须位于源文件的顶端. import java.util.*; package com.hyy.test; ...
- 【C++】不要依赖编译器的默认初始值
最好在定义的时候就给出初始值. 类和结构体给出构造函数. 比如int,在vs的debug和release模式下,初始化的值是不同的.
- Java未赋值变量的默认初始值
在 Java 程序中,任何变量都必须经初始化后才能被使用.当一个对象被创建时,实例变量在分配内存空间时按程序员指定的初始化值赋值,否则系统将按下列默认值进行初始化: 数据类型 初始值 byte 0 s ...
- 【转】Lombok Pojo默认初始值问题
Lombok以注解形式来简化java代码,提高开发效率.比如我们常用的@Builder.@Data.@AllArgsConstructor.@NoArgsConstructor.@ToString等. ...
- Lombok Pojo默认初始值问题
Lombok以注解形式来简化java代码,提高开发效率.比如我们常用的@Builder.@Data.@AllArgsConstructor.@NoArgsConstructor.@ToString等. ...
- 126、Java面向对象之引用传递实例四,修改类成员的属性值
01.代码如下: package TIANPAN; class Message { private String info = "此内容无用"; // 定义String类型属性 p ...
- Select2实现的带搜索的省市区三级联动代码 设置默认初始值
$(function() { $('#loc_province').select2('val','2456'); $('#loc_province').change(); $('#loc_city') ...
随机推荐
- 红帽学习笔记[RHCE]网络配置与路由转发
目录 网络配置基本的IPV4与IPV6 拓扑图 操作 新加一块网卡 将增加的网卡分别加到两台虚拟机上 在两台虚拟机上配置IPV4与 IPV6 配置域名访问 拓展路由转发 拓扑图 操作 关于网关设置 重 ...
- SpringBoot系列之i18n集成教程
目录 1.环境搭建 2.resource bundle资源配置 3.LocaleResolver类 4.I18n配置类 5.Thymeleaf集成 SpringBoot系统之i18n国际化语言集成教程 ...
- LaravelS - 基于Swoole加速Laravel/Lumen
LaravelS LaravelS是一个胶水项目,用于快速集成Swoole到Laravel或Lumen,然后赋予它们更好的性能.更多可能性.Github 特性 内置Http/WebSocket服务器 ...
- Alibaba Nacos 学习(一):Nacos介绍与安装
Alibaba Nacos 学习(一):Nacos介绍与安装 Alibaba Nacos 学习(二):Spring Cloud Nacos Config Alibaba Nacos 学习(三):Spr ...
- TraceID在AspNETCore日志排障中的应用
前言 .NetCore日志,相信大家多少都接触过,博客园有关 ① AspNetCore依赖注入第三方日志组件 ②第三方日志组件Nlog,Serilog 应用方法的博文层出不穷. 结合程序的部署结构 ...
- day20190904一号店网页HTML+CSS
1.知识要理一理.梳理.当天讲了什么内容?当天我学习到了什么内容?看懂.所有的学习型从模仿开始.1.看懂代码,看懂思路,学思路,多问自己问题,为什么要这么写?下一步为什么要这么写?因 果.2.多练多敲 ...
- Chapter 03—Getting Started with graphs
例01:一个简单的例子 一. 图形参数 1. 符号和线条 例02: plot(dose,drugA,type="b",lty=3,lwd=3,pch=1 ...
- 【MySQL】数据库课程实验
数据定义 #mysql --version 查版本号 #mysql -uroot -p #登录 show databases; #查询当前服务存在的数据库 #create database test ...
- obs命令行工具obsutil的使用测试
test1 批量复制,目标文件夹ggggg-zyx0809/data/tmp/a0文件夹已存在,不使用flat命令,目标路径包含a0文件夹 操作 从ggggg-zyx0809/data/g_cageg ...
- Ctrl + Shift + F7 ; F3、Shift + F3
pycharm 查找并高亮参数 选中某一参数,Ctrl + Shift + F7 高亮所有该文件中所有该参数 接下来, 按 F3 在所有高亮选择中向下移动一个, Shift + F3 在所有高亮选择 ...