java与C++变量初始化的对比

java尽力保证:所有变量在使用前都能得到恰当的初始化

①函数/方法局部变量的初始化

在C/C++中,变量的初始化还是得依赖于程序员的自觉性。对于函数局部变量,编译器不会为基本类型赋予默认初始值,新手经常会使用未初始化的指针访问内存,导致程序崩溃。对于类对象,编译器将使用类的默认构造函数对对象进行初始化。而在java中,对于方法的局部变量,java以编译时错误来保证变量在使用前都能得到恰当的初始化。

void f(){
    int i ;
    i ++ ; //Error- - i not initialized
}

尽管java编译器也可以为方法的局部变量赋予一个初值,但局部变量未初始化更有可能是程序员的疏忽,采用默认值反而会掩盖这种错误。

②类数据成员的初始化

C++程序员刚接触到java的类时可能会很不习惯,java类的数据成员居然可以在定义时就初始化:

public class InitialValues{
boolean bool = true;
char ch = ‘x’;
int i = 999;
double d = 3.14;
Depth d = new Depth();
}

这种方式在java中称为指定初始化。在指定初始化之前,编译器还会为这些数据成员进行默认初始化,实际上是把刚分配的对象内存都置零。

// java数据成员的默认初始化
public class InitialValues{
    boolean t;  //flase
    char c;     //[]
    short s;    //0
    byte b;   //0
    int i; //0
    long l; //0
    float f; //0.0
    double d; //0.0
}

在对象里定义一个引用,且不将其初始化时,默认初始化为null。这种默认初始化的实现是,在创建(new)一个对象时,在堆上对对象分配足够的空间之后,这块存储空间会被清零,这样就自动把基本类型的数据成员都设置成了默认值。默认初始化动作之后,才执行指定初始化。也就是说下面的i经历过被初始化为0后,再赋值为999的过程。

public class InitialValues{
int i = 999;
}

java也可以使用构造函数来进行初始化,但构造函数的初始化无法阻止指定初始化和默认初始化的进行,而且总是在它们之后,才会执行构造函数初始化。总结起来说,java中数据成员的初始化过程是:

  • ① 先默认初始化
  • ② 进行定义处的初始化(指定初始化)
  • ③ 构造函数初始化

C++禁止在定义数据成员时就进行指定初始化,而且C++也没有默认初始化。有人会问,下面这段代码不是默认初始化了吗?

class Test {
public:
    int i;
    double b;
    char ch;
};
int main()
{
    Test *t = new Test();
    cout << t->b; //输出0
    cout << t->i; //输出0
    cout << t->ch; //输出[]
    return 0;
}

这实际上是C++的默认构造函数进行的构造函数初始化。当类没有构造函数时,编译器会为类声明并实现一个默认构造函数,默认构造函数将数据成员初始化为默认值。所以C++数据成员的初始值,只能依赖:

  • 成员初始化列表
  • 构造函数

成员初始化列表与java的指定初始化相似,也是在进入构造函数函数体之前,对数据成员进行的初始化。在数据成员的初始化顺序上,java与C++倒是一致的,定义的顺序决定了初始化的顺序。

关于static成员的初始化

在java中不允许有static的局部变量,只能够有static的域,如static数据成员。static数据成员在对象被第一次创建时才会被实例化,而且只实例化一次。例如:

class StaticTest{
    int _a ;
    StaticTest (int a ){
        _a = a;
        System.out.println("StaticTest("+a+")");
    }
}

class Test{
    static StaticTest st1= new StaticTest(1);
    StaticTest nonSt = new StaticTest(0);     
    static StaticTest st2= new StaticTest(2);
}

public class Main {
    public static void main(String[] args) throws Throwable {
        Test t = new Test(); //直到这个时候, st1与st2才会被实例化
    }
}

//程序输出
StaticTest(1)
StaticTest(2)
StaticTest(0)

从输出可以看出在java中,初始化顺序是这样的:先初始化静态数据成员,再初始化非静态数据成员。

在C++中,static数据成员必须在类之外初始化。关于C++的static,http://www.cnblogs.com/QG-whz/p/4473384.html 我以前的总结放在这里挺合适的。

回到篇首的话。

java尽力保证:所有变量在使用前都能得到恰当的初始化(《java编程思想》)

java在变量初始化上,普通变量以编译错误、成员变量以默认初始化等手段,尽力使所有的变量在使用前都可得到初始化,在安全性上大大强于C++。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏顶级程序员

一文读懂Python可迭代对象、迭代器和生成器

序列可以迭代的原因:iter 函数。解释器需要迭代对象 x 时,会自动调用 iter(x)。内置的 iter 函数有以下作用:

1023
来自专栏xingoo, 一个梦想做发明家的程序员

第四章 数据抽象 ----《C++编程思想》

1 袖珍C库 2 相应问题 3 基本对象   C++允许任何类型的指针赋给void* 4 什么是对象 5 抽象数据类型 6 对象细节 7 头文件形式   可能问...

1658
来自专栏前端儿

JS 生成26个大小写字母

主要用到 str.charCodeAt()和 String.fromCharCode()方法

721
来自专栏青枫的专栏

代码块练习题:看代码写程序的执行结果。

641
来自专栏小白客

Python基础学习-字典

一:使用字典:在Python中,字典是一系列键-值对,与键相关联的值可以是数字、字符串、列表乃至字典。字典用放在花括号{}中的一系列键-值对表示。键与值之间用冒...

3309
来自专栏Python小屋

Python版24点游戏

24点游戏是指随机选取4张扑克牌(不包括大小王),然后通过四则运算来构造表达式,如果表达式的值恰好等于24就赢一次。下面的代码定义了一个函数用来测试随机给定的4...

2966
来自专栏程序员互动联盟

【编程基础第十二讲】web开发编程基础--回调函数

存在问题: 我们都知道C语言存在回调函数,那么JavaScript中的回调函数是如何实现的? 如何实施: a.非参数回调函数 这类回调函数大多比较简单,往往传一...

2295
来自专栏LinkedBear的个人空间

唠唠SE的集合-01——Collection接口

当集合中存储的对象类型不同时,那么会导致程序在运行的时候的转型异常,所以jdk1.5加入了泛型机制。

552
来自专栏一个会写诗的程序员的博客

JavaScript 获取 2 天之后的日期代码实例

791
来自专栏LanceToBigData

JS中toString()、toLocaleString()、valueOf()的区别

Array、Boolean、Date、Number等对象都具有 toString()、toLocaleString()、valueOf()三个方法,那这三个方...

541

扫码关注云+社区