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 条评论
登录 后参与评论

相关文章

来自专栏Esofar 开发日记

JavaScript权威指南 - 函数

函数本身就是一段JavaScript代码,定义一次但可能被调用任意次。如果函数挂载在一个对象上,作为对象的一个属性,通常这种函数被称作对象的方法。用于初始化一个...

783
来自专栏IMWeb前端团队

标准的Promise

序言 不同项目下lib里的promise/deferred往往是差异化最多的,用起来和自己的习惯相比经常是缺胳膊少腿多屁眼有卵用,因此聊聊标准的Promise的...

1935
来自专栏海天一树

小朋友学C++(12):多态

(一) 先编写函数: #include <iostream> using namespace std; class Shape { protected:...

3106
来自专栏Python爬虫实战

Python数据类型之列表(后续)

如图所示,有list1和list2两个列表,我们可以发现,原来列表竟然可以比较大小,在这里肯定有读者会说,123肯定小于234,但是如果我们往列表里面多添加几个...

932
来自专栏技术/开源

一道javascript面试题

下面表达式比较的结果分别是什么? 1. []=="0" 2. []==0 3. "0"==0 4. []==false 5. []==[] 大家可以...

22210
来自专栏专注 Java 基础分享

关于类的对象创建与初始化

今天,我们就来解决一个问题,一个类实例究竟要经过多少个步骤才能被创建出来,也就是下面这行代码的背后,JVM 做了哪些事情? Object obj = new ...

3325
来自专栏叁金大数据

自学Python二 Python中的屠龙刀(续)

秉承着一切皆对象的理念,函数作为对象,可以为其赋值新的对象名,也可以作为参数传递给其他函数!

1113
来自专栏Python小屋

Python类中公开方法、私有方法和特殊方法的继承原理

在Python中,对象的私有成员在类外部、派生类或程序中无法直接访问,但是可以通过“对象名._类名__私有成员名”这样一种特殊的形式来访问。 基类的构造方法和公...

3775
来自专栏河湾欢儿的专栏

第九节 js里的new方法

要创建 Person 的新实例,必须使用 new 操作符。以这种方式调用构造函数实际上会经历以下 4 个步骤: (1) 创建一个新对象; (2) 将构造函...

1161
来自专栏从流域到海域

JavaScript闭包详解

JavaScript闭包详解 闭包就是由函数创造的一个词法作用域,里面创建的变量被引用后,可以在这个词法环境之外自由使用(维基百科)。 闭包,官方对闭包...

2058

扫码关注云+社区