现在的程序在使用类的时候一般都按照如下的步骤进行:
- 声明并实例化对象这个时候实例化对象中的属性并没有任何的数据存在,都是其对应数据类型的默认值
- 需要通过一些列的setter方法为类中的属性设置内容
等于现在想要真正获得一个可以正常使用的实例化对象,必须经过两个步骤才能完成
public class JavaDemo{
public static void main(String args []){
//对象初始化准备
Person per = new person();//声明并实例化对象
per.setName("张三");
per.setAge(18);
//对象的使用
per.tell();//方法的调用
}
}
但是如果按照这个方法思考,就会发现一个问题:
假设说现在类中的属性很多个,按照之前的做法就需要调用很多次setter方法,这样的调用太啰嗦了,为了考虑到这样的问题,专门提供了构造方法,即:可以通过构造方法实现实例化对象中的属性初始化处理。只有在关键字new的时候使用构造方法。
在Java程序里构造方法的定义如下:
- 方法名称必须与类的名称保持一致
- 构造方法不允许设置任何的返回值类型,即:没有返回值定义
- 构造方法是在使用关键字new实例化对像的时候自动调用的。
定义构造方法:
class Person{//定义一个类
private String name;//姓名
private int age;//年龄
//方法名称与类名称相同,并且无返回值定义
public Person(String n,int a){
name = n;//为类中的属性赋值(初始化)
age = a;
}
public void tell(){
System.out.println("姓名"+name+"、年龄"+age);
}
}
public class JavaDemo{
public static void main(String args []){
//对象初始化准备
Person per = new person("张三",18);//声明并实例化对象
//对象的使用
per.tell();//方法的调用
}
}
下面针对当前的对象实例化格式与之前的对象实例化格式做一个比较:
- 之前的对象实例化格式:①Person ②per = ③new ④person();
- 当前的对象实例化格式:①Person ②per = ③new ④person(“张三”,18);
①Person:主要是定义对象的所属类型,类型决定了你可以调用的方法
②per:实例化对象的名称,所有的操作通过对象来进行访问
③new:开辟一块新的堆内存空间
④person(“张三”,18):调用有参构造。④person():调用无参构造
在Java程序里考虑到程序结构的完整性,所以所有的类都会提供有构造方法,也就是说,如果现在你的类中没有定义任何的构造方法,那么一定会默认提供有一个无参的,什么都不做的构造方法。这个构造方法是在程序编译的时候自动编译的。如果已经在类中明确的定义有一个构造方法的时候,那么这个默认的构造方法将不会被自动创建。
结论:一个类至少存在一个构造方法,永恒存在!
疑问:为什么构造方法上不能设置返回值类型?
既然构造方法是一个方法,那么为什么不让它定义返回值类型呢?
既然构造方法不会返回数据,为什么不使用void定义呢?
分析:程序编译器是根据代码结构来进行编译处理的,执行的时候也是根据代码结构来处理的
如果在构造方法上使用了void,那么此结构就与普通方法的结构完全相同了,这样编译器就会认为此方法是一个普通方法,普通方法和构造方法最大的区别在于:构造方法是在类对象实例化的时候调用的,而普通方法是在类对象实例化产生之后调用的。
既然构造方法是一个方法,那么方法就有重载的特点,而构造方法重载的时候只需要考虑参数的类型及个数即可。
构造方法重载:
class Person{//定义一个类
private String name;//姓名
private int age;//年龄
public Person(){
name = "无名氏";
age = -1;
}
public Person(String n){
name = n;
}
//方法名称与类名称相同,并且无返回值定义
public Person(String n,int a){
name = n;//为类中的属性赋值(初始化)
age = a;
}
public void tell(){
System.out.println("姓名"+name+"、年龄"+age);
}
}
public class JavaDemo{
public static void main(String args []){
//对象初始化准备
Person per = new person("张三",18);//声明并实例化对象
//对象的使用
per.tell();//方法的调用
}
}
在进行多个构造方法的时候强烈建议大家有一些定义的顺序,例如:可以按照参数的个数降序或升序排列。
经过发现,构造方法可以进行数据的设置,setter也可以进行数据的设置,这个时候一定要清楚,构造方法是在对像实例化的时候为属性设置初始化内容,而setter除了拥有设置数据的功能之外还有修改数据的功能。
经过分析之后发现,利用构造方法可以传递属性数据,进一步分析对像的产生格式:
- 定义对象名称:类名称 对象名称 = null;
- 实例化对象:对象名称 = new 类名称();
如果这个时候只是通过实例化对象来进行类的操作也是可以的,而这种形式的对象由于没有名字就叫匿名对象。
public calss JavaDemo{
public static void main(String args []){
new Person("张三",10).tell();//进行方法的调用
}
}
此时依然通过了对象进行了类中tell()方法的调用,但是由于此对象没有任何的引用名称,所以该对象使用一次之后就将成为垃圾,而所有的垃圾将被GC进行回收与释放。
发现此时程序里面已经有构造方法了,那么下面通过一个程序来利用构造方法来进行内存分析:
class Massage{
private String titile;
public Message(String t){
title = t;
}
public String getTitle(){
retrun title;
}
public void setTitle(string t){//具有修改功能
title = t;
}
}
class Person{//定义一个类
private String name;//姓名
private int age;//年龄
public Person(Massage msg,int a){
name = msg.getTitle();
age = a;
}
public Massage getInfo(){
return new massage(name+":"+age);
}
public void tell(){
System.out.println("姓名"+name+"、年龄"+age);
}
}
public class JavaDemo{
public static void main(String args []){
//对象初始化准备
Massage msg = new massage(new Message("midn"));
Person per = new person(msg,20);
msg = per.getInfo();
System.out.println(msg.getTitle());
}
}
通过此程进行内存分析:
只要是方法都可以传递任意的数据类型(基本数据类型和引用数据类型)



