java类的主体是_Java核心技术 对象与类

对象与对象变量:

要想使用对象,就必须首先构造对象,并指定其初始状态,然后,对对象应用方法。

在Java中,使用构造器构造新实例。构造器是一种特殊的方法,用来构造并初始化对象。

在实际开发中,通常需要将对象存放在一个变量中来达到多次使用的目的。

在这里,birth指向了Date对象的存储的地方的引用。

Date birthday = new Date();

f4c4ff125ee4d16b24fa963ed24fc03d.png

在对象与对象变量直接存在着一个重要的区别,例如:

Date deadline; //deadline dosen't refer to any object;

定义了一个对象变量,它可以引用Date类型的对象,但是,一定要认识到:变量deadline不是一个对象,实际上也没有引用一个对象。并且不能用任何Date方法应用与这个变量上

s = deadline.toString(); //not yet

在这里,将会产生编译错误。

首先,必须初始化变量deadline,可以用新构造的对象初始化,也可以引用一个已存在的变量。

deadline = new Date() //新构造对象//

//或者

Date birthday = newDate();

deadline= birthday; //引用一个已存在对象

当引用一个已存在的变量时,两个变量引用同一个对象:

dbc5a13d540f3a73384f5254ee0e077c.png

在Java开发中,一定要认识到,一个对象变量并没i月实际包含一个对象,而仅仅使用一个对象,任何对象变量的值都是对存储在另外一个地方的一个对象的引用。

构造器:

首先来看一个Employee类:

classEmployee

{//instance fields

privateString name;private doublesalary;privateLocalDate hireDay;//constructor 构造器

public Employee(String n, double s, int year, int month, intday)

{

name=n;

salary=s;

hireDay=LocalDate.of(year, month, day);

}//a method

publicString getName()

{returnname;

}//more methods

. . .

}

该类的构造器与类同名,在构造Employee类的对象时,构造器会运行,并将实例初始化为所希望的状态。

例如,当使用下面代码实例化时:

new Employee("James Bond", 100000, 1950, 1, 1);

将会把实例域设置为:

name = "James Bond";

salary= 100000;

hireDay= LocalDate.of(1950, 1, 1); //January 1, 1950

构造器与其他方法有一个区别,即构造器伴随着new操作符被调用并且没有返回类型,而不能通过一个对象调用构造器,如下将会产生编译错误。

james.Employee("James Bond", 250000, 1950, 1, 1) //ERROR

重载:

一个类可以拥有多个构造器。

如果在一个类中,拥有多个方法,每个方法具有相同的名字,不同的参数,便产生了重载。编译器必须挑选出具体执行那个方法。

通常,多个具有相同名字不同参数的方法,是通过它们的签名来辨识的。

什么是方法的签名?

一个方法的签名即为它的方法名称,以及参数类型。注意,返回类型并不属于一个方法的签名!也就是说,不能具有相同名字,相同参数,而返回类型不同的两个方法。

默认域初始化:

如果在构造器中没有显式的给域赋予初始值,那么域会被自动地赋予默认值,数值为0,布尔值为false,对象引用为null。

这是域与局部变量的主要不同点。局部变量必须明确地初始化。

无参数的构造器:

很多类都包含一个无参数的构造函数,对象由构造函数创建时,状态会被设置为适当的默认值,如下为Employee类的无参数构造函数:

publicEmployee()

{

name= "";

salary= 0;

hireDay=newDate();

}

如果一个类没有编写构造器的话,那么系统就会提供一个无参数构造器。

调用另一个构造器:

如果构造器的第一个语句形如this(...),这个构造器将会调用同一个类的另一个构造器,下面是一个典型的例子:

public Employee(doubles){this("Employee ",s);

}

当调用new Employee(60)时,Employee(double)构造器将调用Employee(String,double)构造器。

初始化块:

通常,初始化数据域有三种方法,一种是在构造器设置,一种是在声明中赋值,还有一种是初始化话块,只要构造类的对象,这些块就会被执行。

classEmployee

{private static intnextId;private intid;privateString name;private doublesalary;//object initialization block 初始化块

{

id=nextId;

nextId++;

}public Employee(String n, doubles)

{

name=n;

salary=s;

}publicEmployee()

{

name= "";

salary= 0;

}

. . .

}

由于初始化数据域有多种途径,所以可能会有些混乱,下面是调用构造器的具体处理步骤:

1.所有数据域被初始化默认值。

2.按照类声明中出现的次序,依次执行所有域初始化语句和初始化块。

3.如果构造器第一行调用了第二个构造器,则执行第二个构造器的主体。

4.执行这个构造器的主体。

不要编写返回引用可变对象的访问器方法:

例如:

classEmployee

{privateDate hireDay;

. . .publicDate getHireDay()

{return hireDay; //Bad

}

. . .

}

这样会破坏封装性,即可以不通过Hireday的设置器方法就能改变Hireday,具体如下代码:

Employee harry =. . .;

Date d=harry.getHireDay();double tenYearsInMilliSeconds = 10 * 365.25 * 24 * 60 * 60 * 1000;

d.setTime(d.getTime()- (long) tenYearsInMilliSeconds);//let's give Harry ten years of added seniority

出错的原因即为d和harry.hireDay引用同一个对象,参见图:

71b03220b83f6be9c7564d1c7ff8c1aa.png

实际上,返回对象应对其进行克隆,并返回该克隆对象。

final实例域:

可以将实例域定义为final,构建对象时必须初始化这样的域,也就是说,确保在每一个构造器执行之后,这个域的值被设置,并且在后面的从早中,不能够在对它进行修改。

通常,不推荐是哦那个final修饰符应用于可变的类,如下:

private final Date hiredate;

这仅仅意味着,存储在hiredate变量的对象引用不可变,也就是说永远指向同一个对象引用,但任何方法都可以对hiredate引用的对象调用setTime更改器来改变该对象。

静态域与静态方法:

如果将域定义为static,每个类中只有一个这样的域。而每一个对象对于所有的实例域却都有一份自己的拷贝。静态域是所有对象共享的,如这里给Employee类添加一个实例域id和一个静态域nextId:

classEmployee

{private static int nextId = 1;private intid;

. . .

}

换句话说,如果有1000个Emplyee对象,那么就有1000个id实例域,但却只有一个nextId静态域。即使没有实例对象时,静态域也是存在的,它是属于类,而不是属于对象。

静态方法同理。

方法参数:

Java中的参数是一种值引用。

首先,观察一下基本数据类型(数字、布尔值)。

public static void tripleValue(double x) //doesn't work

{

x= 3 *x;

}double percent = 10;

tripleValue(percent);

当我们将percent作为参数给一个方法时,经过执行后,percent本身并没有改变,这个很容易理解。

c56ecb841fc53007c19b6e085a203c44.png

接下来看一下对象引用:

public static void tripleSalary(Employee x) //works

{

x.raiseSalary(200);

}

当调用如下语句时:

harry = newEmployee(. . .);

tripleSalary(harry);

具体的操作过程如下:

1.x被初始化harry值的拷贝,这是一个对象的引用。

2.raiseSalary方法应用于这个对象的引用,x和harry同时引用的那个Employee对象的薪金提高了200%。

3.方法结束后,参数变量x不再使用。当然,对象变量harry继续引用那个薪金增3倍的对象。

696d60fb32edcfeed5a6d9f377cee05a.png

看到这里,或许会认为Java的对象参数是引用调用,实际上,并不是。

依然是按值调用,传给tripleSalary函数的实际上是harry值的拷贝,而harry值指向了一个对象引用,所以x也指向了该引用,所以才会引起变化。

可以通过下面这个例子得出结果:

public static void swap(Employee x, Employee y) //doesn't work

{

Employee temp=x;

x=y;

y=temp;

}

如果Java是按引用调用的话,那么执行下面语句后,a与b应该互换。

Employee a = new Employee("Alice", . . .);

Employee b= new Employee("Bob", . . .);

swap(a, b);//does a now refer to Bob, b to Alice?

但在测试中,并没有互换,所以传入的实际上是a,b的拷贝,也就是一个对象引用,这个方法交换的是这两个拷贝。

最终,函数结束时,变量x,y被丢弃了,原来的ab对象变量依然引用之前的对象。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值