初始化在Java编程中的重要性

发表时间: 2018-12-18 16:33

有人说,你应该关注时事、财经,甚至流行的电影、电视剧,才有可能趁着热点写出爆文;有人说,你别再写“无聊”的技术文了,因为程序员的圈子真的很小,即便是像圈中鸿祥那样的招牌大牛,文章是那么的干货,浏览量有多少?不到万吧;有人说,你别妄想在写作上面知识变现了,因为你写的文章真的很不优秀,我都不爱看!

我想说,你们说的话我都不爱听!我也懒得反驳,因为我没有成功,也就没有话语权,多说无益。我只想做一名执拗的程序员,静静地写文,哪怕只有一个读者——有时候,做一件事,并不是想要结果,仅仅只是因为心底那种狂热的喜欢。

今天,我打算聊聊Java编程中的初始化。

使用构造器来确保对象初始化

在Java编程中,无论是对象,还是基本类型,都不允许在未经初始化的情况下使用它们;否则,Java编译器就会热情地提醒你——请初始化后再使用。

那,Java是通过什么机制来确保对象初始化呢?

答案就是“构造器”——类的对象要被正确的初始化,就必须先过构造器这一关。

程序清单1-1:一个带有构造器的简单类

class Writer { public Writer() { System.out.println("我是一名写作爱好者"); } public static void main(String[] args) { new Writer(); }}

当使用关键字new来创建一个对象Writer时,就会调用构造器(与类名Writer相同的方法Writer())进行初始化,因此上述程序就会输出“我是一名写作爱好者”。

构造方法Writer()没有参数,因此被称为无参构造器;事实上,无参构造器是可以省略的——编译器会自动创建一个无参构造器,被称为“默认构造器”(Java设计者真的无比明智啊——帮助程序员省去了创建默认构造器的麻烦)。

程序清单1-2:默认构造器

class Writer { public static void main(String[] args) { new Writer(); }}

默认构造器并不会一直“默认”存在,如果已经定义了一个构造器,无论有参还是无参,编译器将不再自动创建默认构造器。

程序清单1-3:不会一直存在的默认构造器

class Writer { public Writer(String name) { System.out.println(name + "是一名写作爱好者"); } public static void main(String[] args) { new Writer(); new Writer("沉默王二"); }}

一旦定义了一个有参构造器,那么在创建对象的时候就必须传递构造器需要的参数,否则编译器会提示“The constructor Writer() is undefined”(使用new Writer()创建对象对)——这样做的好处就是,确保对象在初始化的时候符合类设计的初衷(上例中,Writer需要指定作者姓名,所以你在创建Writer对象时不能不传递作者姓名)。

参差不齐乃幸福本源

读王小波的《沉默的大多数》,我喜欢上了一句话:“参差不齐乃幸福本源”。王小波的意思可能是想说:一个能容忍不同观点与不同的生活方式的社会,才是一个幸福的社会。那么,在Java的世界里,也有一个幸福的社会。

由于构造器的特殊性(不能与其他成员方法的名字冲突),导致构造器的名字必须和类名保持一致,也就是说,一个类,只能有一个构造器名。这似乎局限了构造器的使用方式。但其实不然,Java允许方法重载——可以只有一个方法名,但方法的参数列表可不尽相同;哎,问题就这么巧妙的解决了。

程序清单2-1:构造方法的重载

class Writer { private String name; private String bookName; public Writer(String name) { this.name = name; System.out.println(name + "是一名写作爱好者"); } public Writer(String name, String bookName) { this.name = name; this.bookName = bookName; System.out.println(name + "不仅是一名写作爱好者,还出版了书籍" + bookName); } public static void main(String[] args) { new Writer("沉默王二"); new Writer("沉默王三", "《Web全栈开发进阶之路》"); }}

你看,沉默王二没出版书籍,可以是一名写作爱好者;沉默王三虽然出版了书籍《Web全栈开发进阶之路》,但仍然和王二是好兄弟,并没有看不起王二(从来没说过:“王二,你个渣渣,连本书都没有出版,好意思说自己是写作爱好者?”)。是不是很和谐?

Java该如何区分重载方法(毕竟参数名相同)呢?上例中,你也看到了,参数个数的不同就可以区分;另外,参数的类型和顺序(不建议使用,因为这样做会让代码难以维护,见下例)也可以用来作为区分的条件。

程序清单2-2:难以维护的方法重载(靠顺序,别这样!)

class Writer { private String name; private int age; public Writer(String name, int age) { this.name = name; this.age = age; } public Writer(int age, String name) { this.age = age; this.name = name; }}

返回当前对象引用的this关键字

在很长一段时间里,我对this关键字都避而不见,因为我搞不懂它到底在干嘛,我所使用它的场合仅限于程序清单2-1(this.name指的是类的成员变量,而name指的是当前方法的参数)。直到我遇见了jQuery(一个快速、简洁的JavaScript框架)的链式调用。

程序清单3-1:jQuery的链式调用

$("#canvas").append("我是一块抹布") .show();

方法后面还能再紧跟一个方法,就好像一个胯下运球再接一个后仰跳投,超自然超连贯超燃的一连串动作,令人心情感到愉悦。这背后是什么原理呢?

程序清单3-2:jQuery的链式调用的背后

var MyJQ = function(){ } MyJQ.prototype = { append : function(content){ console.log("添加内容"); return this; }, show : function(){ console.log("将元素显示"); return this; } }; var myjq = new MyJQ(); myjq.append("我是一块抹布").show();

看完程序清单3-2之后,你就会恍然大悟,原来方法的内部返回了一个this,而这个this就是当前对象的引用;也就是说,myjq.append("我是一块抹布").show();就相当于:myjq.append("我是一块抹布");myjq.show();。

理解了jQuery的链式调用,我们来模拟一下Java当中的链式调用(写作者做完运动后去睡觉)。

程序清单3-3:Java的链式调用

class Writer { public Writer sleep() { System.out.println("睡一觉"); return this; } public Writer ml() { System.out.println("做运动"); return this; } public static void main(String[] args) { new Writer().ml().sleep(); }}

理想国中的Java垃圾回收

对象要想被正确使用,必须先被初始化,这是一切的开端;那么,当对象不再被使用后,它就需要被清理掉,要善始善终。假如你遇到一个面试官,他要“强行”问你关于Java垃圾回收的一些问题,你可以提前做好下面这些准备。

Q:为什么要进行垃圾回收?

A:如果不进行垃圾回收,内存迟早都会被消耗空。除非内存无限大,我们可以任性的分配而不回收,但是事实并非如此。所以,垃圾回收是必须的。

Q:哪些内存需要回收?

A:所谓“要回收的垃圾”无非就是那些不可能再被任何途径使用的对象。

Q:Java是如何回收垃圾的?

A:深入理解 Java 垃圾回收机制(
https://www.cnblogs.com/andy-zcx/p/5522836.html)

不过,在我的印象里,有一副爆笑的动态图令我印象深刻,它隐喻的是Java的垃圾回收机制。

作者简介:沉默王二,CSDN 博客专家。

声明:本文为作者投稿,版权归其个人所有。