Java中的类---接上篇

 

一、用户自定义类

1、写先出一个简单的Employee类作为例子说明。

代码如下:

import java.time.LocalDate;

/**
 * 自定义方法练习--测试 这个程序中包含了两个类Employee类和带有public修饰的EmployeeTest类
 * 
 * @author:Archer-LCY
 * @date:2018年2月3日下午8:08:51
 */
public class EmployeeTest {
	public static void main(String[] args) {
		// 定义Employee类得数组并赋值
		Employee staff[] = new Employee[3];
		System.out.println("加薪前");
		staff[0] = new Employee("Herry", 70000, 1988, 12, 3);
		staff[1] = new Employee("Marry", 20000, 1990, 1, 31);
		staff[2] = new Employee("Amy", 10000, 1989, 11, 23);

		// 给每个人加薪5%,并打印
		System.out.println("\n加薪后");
		for (Employee em : staff) {
			em.raiseSalary(5);
			System.out.println("姓名:" + em.getname() + "\t薪水:" + em.getSalary() + "\t入职时间:" + em.getHireDay());
		}
	}
}

/**
 * 员工类
 * 
 * @author:Archer-LCY
 * @date:2018年2月3日下午8:28:40
 */
class Employee {
	private String name;
	private double salary;
	private LocalDate hireDay;// LocalDate需要导入包java.time.LocalDate;

	public Employee(String name, double salary, int year, int month, int day) {
		super();
		this.name = name;
		this.salary = salary;
		this.hireDay = LocalDate.of(year, month, day);
		output();
	}

	// 输出
	public void output() {
		System.out.println("姓名:" + name + "\t薪水:" + salary + "\t入职时间:" + hireDay);
	}

	public String getname() {
		return name;
	}

	public double getSalary() {
		return salary;
	}

	public LocalDate getHireDay() {
		return hireDay;
	}

	public void raiseSalary(double byPercent) {
		double raise = this.salary * byPercent / 100;
		salary += raise;
	}
}

 

 

 

注意:

(1)在这个示例中包含两个类,一个是Employee类,一个是带有public访问修饰符的EmployeeTest类。其中EmployeeTest类中包含了main方法。

(2)源文件名是EmployeeTest.java ,这是因为文件名必须与public类的名字相匹配。在一个源文件中,只能有一个共有类,但非共有类的数目可以任意。

(3)当编译这段源代码时,编译器将在目录下创建两个类文件:EmployeeTest.class和Employee.class 。

(4)程序中包含main方法的类名提供字节码解释器,以便启动这个程序:java EmployeeTest  ,字节码解释器开始运行EmployeeTest类的main方法的代码。在这段代码中,定义了Employee类的数组,里面有三组,先后构造了三个新的Employee对象,并显示它们的状态。

2、多个文件的使用

在EmployeeTest.java一个源文件中包含了两个类。但一般情况下,我们习惯于每一个类放在一个单独的源文件中。例如,将EmployeeTest类放在EmployeeTest.java中,而将Employee类放在Employee.java中。

如果习惯上述所说的分开组织文件的方法,可以采用下面三种编译源程序的方法:

注:每次在命令提示符窗口使用前记得先用dir,查看当前目录,这样可以避免不必要的尴尬错误。还需要注意的一点是javac和java时必须是在文件所在目录,如下图:

 

(1)使用通配符调用java编译器:javac EmployeeTest*.java ,此时,所有的与通配符匹配的源文件都将被翻译成类文件。编译成功结果如下:

 

(2)第二种是使用:javac EmployeeTest.java ,编译成功结果如下:

 

(3)第三种:如果想独立测试使用:java EmployeeTest    这种方法可以将结果打印在命令提示符窗口,结果如下图:

 

一个更大应用程序的一部分,可以用):java Application

3、对Employee类剖析

(1)这个类里面有一个构造器和5方法:

/**构造器*/

public Employee(String name, double salary, int year, int month, int day)

/**5个方法*/

public void output()

public String getname()

public double getSalary()

public LocalDate getHireDay()

public void raiseSalary(double byPercent)

(2)所有方法都被public标记。关键字public意味着任何类的任何方法都可以调用这些方法。

(3)在Employee类中有3个示例域用来存放将要操作的数据:

private String name;

private double salary;

private LocalDate hireDay;

(4)关键字private确保只有Employee类自身才能访问这些实例域,而其他类的方法不可以。

(5)注:可以有public来标记实例域,但是,这是一张极不为提倡的方法。public数据域允许程序中的任何方法都能对其进行读取和修改,这就破坏了封装。任何类任何方法都可以修改public与,这意味着某些代码将可以使用这种存取权限,这是我们不希望看到的。

(6)注:有两个实例域本身就是对象:name域是String类的对象,hireDay域是LocalDate类的对象。其实,这种情形十分常见:类通常包括类型属于某个类类型的实例域。

4、构造函数的使用

A.从Employee类的构造器开始分析

public Employee(String name, double salary, int year, int month, int day) {

super();

this.name = name;

this.salary = salary;

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

output();

}

(1)可以看到构造器与类名相同,在构造Employ类的对象时,构造器会运行,以便将实例域初始化为希望的状态。

(2)例如:new Employee("Herry", 70000, 1988, 12, 3); 这条代码在创建Employee类实例时,将会把实例域设置为:name=Herry;salary=70000;hireDay=LocalDate.of(1988,12,3);

(3)构造与其他方法不一样的是,构造器总是会伴随着New操作符的执行而被调用,而对一个已存在的对象不能调用构造器来达到从新调用的目的。例如:Herry.Employee("Herry", 70000, 1988, 12, 3); 会产生编译错误。

 

 

 

4、隐式参数与显式参数

隐式参数:出现在方法名之前的。

显式参数:位于方法名后面的括号中的。

方法用于操作对象以及存取它们的实例域。将调用这个方法的对象的salary实例域设置为新值。例如:方法:

public void raiseSalary(double byPercent) {

double raise = salary * byPercent / 100;

salary += raise;

}

 

在每一个方法中,关键字this表示隐式参数。可以用下面方式编写上面的方法:(这样的优点:这样可以将实例域与局部变量区分开来)

public void raiseSalary(double byPercent) {

double raise = this.salary * byPercent / 100;

salary += raise;

}

 

注意隐式构造和参数化构造不能共存:尽量把带参构造和默认构造都写出来,当你不把默认构造显示出来时,带参构造会将其覆盖,后面第二次使用时就会出错。

如下图:左边两组是带默认参数,不会报错;右边是不带默认参数,会报错(红色波浪线部分)

 

在Java中,所有的方法都必须在类的内部定义,但并不是内联(inline)方法。是否将某个变量设置为内联方法是Java虚拟机的任务

5、封装

封装(Encapsulation)是面向对象方法的重要原则,就是把对象的属性和操作(或服务)结合为一个独立的整体,并尽可能隐藏对象的内部实现细节。

封装是把过程和数据包围起来,对数据的访问只能通过已定义的接口。面向对象计算始于这个基本概念,即现实世界可以被描绘成一系列完全自治、封装的对象,这些对象通过一个受保护的接口访问其他对象。封装是一种信息隐藏技术,在java中通过关键字private,protected和public实现封装。

优点:(1)一旦在构造器中设定完毕,就没有任何一个办法可以对其修改,这样确保其不会被破坏,安全系数增加。

(2)更改器可以执行错误检查,而直接对域进行赋值就不会有这些处理。

6、静态方法

下面的情况适合使用静态方法:

(1)一个方法不需要访问对象状态,其所需参数都是通过显式参数提供(例如:Math.pow)

(2)一个方法只需要访问类的静态域

Java中的静态域和静态方法在功能上与c++相同,不同的是语法书写上不一样。Java中用 . 如:Math.pow;而c++中是用::操作符访问,如:Math::PI。

关于“static”:起初,C 引入关键字 static 是为了表示退出一个块后依然存在的局部变量在这种情况下, 术语“ static” 是有意义的;变量一直存在,当再次进入该块时仍然存在。随后, static 在 C 中有了第二种含义, 表示不能被其他文件 访问的全局变量和函数。 为了避免引入一个新的关键字, 关键字 static 被重用了。最后, C++ 第三次重用了这个关键字,与前面赋予的含义完全不一样, 这里将其解释为:属于类且不属于类对象的变量和函数。这个含义与 Java 相同。

-------摘自《Java核心技术》

 

二、对象构造

构造器:

(1)特点:没有返回值(void不是)、方法名和类名一致(方法名一样,参数不一样,参数可以有0个或者1个或者多个构造方法的重载)、每个类可以有一个以上的构造器、构造器总是伴随着new操作一起调用

(2)可以指定参数及实现重载

(3)注:java构造器的工作方式和C++一样。但是,所有java对象都是在堆中构造的,构造器总是伴随着new操作符一起使用。Employee number("Herry", 70000, 1988, 12, 3);//在c++中是正确的

(4)警告!!:不要在构造器中定义和实例域重名的局部变量。例如

public Employee(String n, double s, . . ){

String name = n;

double salary = s;

 }//是错误的

在上述错例中,在这个构造器中声明了局部变量name和salary。这些变量只能在构造器内部访问。这些变量屏蔽了同名的实例域。

 

 

在构造器内调用同一个类的构造器,可以用到this关键字,this(...),如:

 

构造器小结注意事项:

(1)一个类中可以有多个构造器,这些构造与类名相同,当存在多个构造方法时,这些方法传递的参数不一样。

(2)如果一个类中没有构造器也没有main方法,这时候会默认一个无参构造;这个构造器会将所有实例域设置为默认值,即实例域里,数值型为0,布尔型为false,对象变量为null。

(3)如果类中至少有一个构造器,并且这个构造器不带有参数,这时,如果在构造对象时没有提供参数就会视为不合法的。

(4)如果构造器中没有显示地给实例域赋值,这时会自动地赋为默认值:数值为0,布尔值为false,对象引用为null。这样,会影响代码的可读性。

(5)注:在c++中,不能直接初始化类的实例域,所有的域都必须在构造器里面设置。

 

三、包

使用包的主要原因是保证类名的唯一性,当两个程序员各自建立了两个类名相同的类时,只要将类放在不同的包中,就不会产生冲突。

为保证包的绝对唯一性,sun公司建议将公司的因特网域名(这是独一无二的)以逆序形式作为包名。例如:banana.com ,逆序变为com.banana。

从编译器角度来看,嵌套的包之间没有任何联系。

一个类可以使用包中的所有类,以及其他包中的公有类(public class)。访问另一个包中的公有类的两种方法:

(1)每个类名之前添加完整的包名(繁琐),例如:java.time.LocalDate today =java.time.LocalDate.now();

(2)我们常用的方法是使用import导入包,import语句应位于源文件的顶部(位于package语句后面),这样就无需在前面加前缀了。例如:import java.time.LocalDate;这时候就可以使用:LocalDate today=LocalDate.now();

 

当两个包中含有相同的类名时,在使用时,编译器无法识别要使用哪个类,这时候我们可以在使用的语句加前缀。例如,两个不同的类中都含有Date类时,可以这样表示:

注:在 C-H■ 中, 与 包 机 制 类 似 的 是 命 名 空 间(namespace)。 在 Java 中, package 与 import 语句类似于 C+H■ 中的 namespace 和 using 指令。

静态导入:如:import static java.lang.System.*;这时候就可以这样写了out.println();但是,这样降低了程序的可读性。

 

四、类设计的技巧

1、保证数据的私有。

2、数据一定要初始化。

3、不要在类中使用过多的私有类型。

4、一个类的职责不应该太多或者太少。

5、类名和方法名尽可能体现其职责。

6、优先使用不可变的类。

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值