在企业面试中,经常问到,面向对象有哪些基本特性?答案应该是:封装、继承和多态。这里简要介绍一下封装。
封装的目的是简化编程和增强安全性。
- 简化编程是指,封装可以让使用者不必了解具体类的内部实现细节,而只是要通过提供给外部访问的方法来访问类中的属性和方法。例如 Java
API 中的 Arrays.sort()方法,该方法可以用于给数组进行排序操作,开发者只需要将待排序的数组名放到
Arrays.sort()方法的参数中,该方法就会自动的将数组排好序。可见,开发者根本不需要了解
Arrays.sort()方法的底层逻辑,只需要简单的将数组名传递给方法即可实现排序。 - 增强安全性是指,封装可以使某个属性只能被当前类使用,从而避免被其他类或对象进行误操作。例如在 Student.java
的程序中,Student 的 stuAge 属性是 public 的形式体现的,但这样做实际存在着安全隐患:TestStudent
类(或在访问修饰符可见范围内的其他类)完全可以随意的对 stuAge 进行修改,如以下程序。
public class TestStudent {
public static void main(String[] args) {
Student wangYun = new Student();
wangYun.stuAge = -10;
...
}
}
如上,给 stuAge 赋了一个不符合逻辑的值,但语法是却正确的。因此这种做法,实际会给程序造成了一定程度上的安全问题
在初识封装中我们提到 public 的形式体现的属性可以随意进行修改,那么如何避免此类问题呢?我们可以使用 private 修饰符来修饰 stuAge 属性,以此禁止 Student 以外的类对 stuAge 属性的修改。但这么做未免显得“过犹不及”,为了保证安全,也不至于让其他类无法访问吧!有没有一种办法,既能让其他类可以访问 Student 类中的 stuAge 属性,又能保证其他类始终是在安全的数值范围内修改 stuAge 值呢?
有,我们可以先用 private 修饰 stuAge 属性,然后再给该属性提供两个 public 修饰的、保证属性安全的访问方法(setter 方法和 getter 方法),即:
- 用 private 禁止其他类直接访问属性;
- 给 1 中的属性新增两个 public 修饰的 setter 和 getter 方法,供其他类安全的访问。
setter 方法用于给属性赋值,而 getter 访问用于获取属性的值。并且一般而言,setter 方法的名字通常是 set+属性名,getter 方法的名字通常是 get+属性名。
根据以上描述,先用 private 修饰 stuAge,禁止 TestStudent 类对 stuAge 的直接访问,以此保证 stuAge 安全性;然后新增 setStuAge()和 getStuAge()方法,一方面供 TestStudent 类间接的访问 stuAge 属性,另一方面也保证了 stuAge 的数据安全,详见以下程序。
public class Student {
private int stuAge ;
//获取stuAge的值
public int getStuAge() {
return stuAge;
}
//给stuAge赋一个合法的值
public void setStuAge(int age) {
//如果年龄在合理范围内,则正常赋值
if( age >0 && age <110)
stuAge = age;
else //如果对年龄的赋值不合理,则设置为默认值0
age = 0 ;
}
}
后续,其他类只需要调用 setStuAge()和 getStuAge()方法,就能对 stuAge 属性进行安全的赋值或取值,代码如下。
public class TestStudent3 {
public static void main(String[] args) {
Student wangYun = new Student();
/* 如果将stuAge赋值负数,setStuAge()就会将stuAge设置为默认值0,防止出现安全问题
wangYun.setStuAge(-10); */
wangYun.setStuAge(22);
int age = wangYun.getStuAge() ;
}
}
实际上,使用 setter 和 getter 的解决方案用到了一个程序设计的基本原则:逻辑代码不能写在变量中,而必须写在方法或代码块中。