封装的概述和好处:
封装原则:
好处:
- public class Student {
- String name;
- //int age;
- Private int age;
- Public void setAge(inta) {
- if (a < 0 || a > 200) {
- System.out.println("你给的年龄有误");
- } else {
- age = a;
- }
- }
- Public int getAge() {
- returnage;
- }
- Public void show() {
- System.out.println("姓名是:" + name + ",年龄是:" + age);
- }
- }
- public class DemoStudent {
- public static void main(String[] args) {
- //创建学生对象
- Student s = new Student();
- s.show(); ? ????????????
- s.name = "柳岩";
- //s.age = 18;
- //s.age = -18;
- //s.setAge(-18);
- s.setAge(28);
- s.show();
- }
- }
在程序中,如果想声明一个类继承另一个类,需要使用 extends 关键字。
格式:
class 子类 extends 父类 {
}
父类内容
, 子类可以使用
子类特有内容, 可以写在子类中.
- class Employee {
- String name; // 定义name属性
- // 定义员工的工作方法
- publicvoid work() {
- System.out.println("尽心尽力地工作");
- }
- }
- /*
- * 定义讲师类Teacher继承员工类Employee
- */
- classTeacherextends Employee {
- // 定义一个打印name的方法
- publicvoidprintName() {
- System.out.println("name=" + name);
- }
- }
- /*
- * 定义测试类
- */
- publicclass Example01 {
- publicstaticvoid main(String[] args) {
- Teachert = newTeacher(); // 创建一个讲师类对象
- t.name = "小明"; // 为该员工类的name属性进行赋值
- t.printName(); // 调用该员工的printName()方法
- t.work(); // 调用Developer类继承来的work()方法
- }
- }
在上述代码中,Teacher 类通过 extends 关键字继承了 Employee 类,这样 Teacher 类便是 Employee 类的子类。从运行结果不难看出,子类虽然没有定义 name 属性和 work() 方法,但是却能访问这两个成员。这就说明,子类在继承父类的时候,会自动拥有父类的成员。
小结:
继承是面向对象的核心特性,是面向对象的学习重点。
继承是代码复用的重要方式,是类与类之间的一种关系。
从类与类之间的设计关系来看, 子类必须属于父类的一种时,才会继承。
父类抽取出了共性的内容,子类可以在父类基础上扩展新的属性与行为。
子类拥有父类的所有属性与方法,无需重新定义。并且可以直接使用非私有的父类成员。 Fu 类中的成员变量是非私有的,子类中可以直接访问,若 Fu 类中的成员变量私有了,子类是不能直接访问的。 当子父类中出现了同名成员变量时,在子类中若要访问父类中的成员变量,必须使用关键字 super 来完成。super 用来表示当前对象中包含的父类对象空间的引用。
在子类中,访问父类中的成员变量格式:
super. 父类中的成员变量
继承 - 子父类中成员方法特点 当在程序中通过对象调用方法时,会先在子类中查找有没有对应的方法,若子类中存在就会执行子类中的方法,若子类中不存在就会执行父类中相应的方法。
- class Fu
- {
- //Fu中的成员变量。
- intnum = 5;
- }
- classZi extends Fu
- {
- //Zi中的成员变量
- intnum = 6;
- void show()
- {
- //子父类中出现了同名的成员变量时
- //在子类中需要访问父类中非私有成员变量时,需要使用super关键字
- //访问父类中的num
- System.out.println("Fu num="+super.num);
- //访问子类中的num2
- System.out.println("Zi num2="+this.num);
- }
- }
- class Demo5
- {
- public static void main(String[] args)
- {
- Zi z = new Zi(); //创建子类对象
- z.show(); //调用子类中的show方法
- }
- }
- class Fu {
- publicvoid show() {
- System.out.println("Fu类中的show方法执行");
- }
- }
- classZi extends Fu {
- public void show2() {
- System.out.println("Zi类中的show2方法执行");
- }
- }
- public class Test {
- public static void main(String[] args) {
- Zi z = new Zi();
- z.show(); //子类中没有show方法,但是可以找到父类方法去执行
- z.show2();
- }
- }
继承特点
Java 支持单继承
Java 支持多层继承
父类定义了继承树中共性内容,子类定义了该类个性内容。
在结合多态后,能使用父类时尽量使用父类,提高程序扩展性。
继承的注意点:
1.java 中不支持类的多继承
- class Fu{
- }
- class Zi extends Fu{
- }
- class Zi2 extends Zi, Object{// 错误写法
- }
2. java 中支持多层继承
- class Grand {}
- class Fu extends Grand {}
- class Zi extends Fu {}
缺点:
1. 不支持多继承 (但是保留了多继承的机制, 通过接口体现)
2. 增强类与类之间的耦合性 (通过接口的方式可以解决)
this:
含义:
谁调用了 this 所在的方法, this 就代表谁.
//1. 使用在方法中, 解决了成员变量和局部变量重名的问题
- public void setName(String name) {
- this.name = name;
- }
//2. 可以调用构造方法, 构造方法之间的相互调用 (了解即可)
- public Student(){
- }
- public Student(String name){
- this();
- }
//3. 调用普通的成员方法, 学了继承之后, 可以区分子父类的方法
- public void show(){
- }
- public void print(){
- this.show();
- }
匿名对象:
有名字的对象
Student stu = new Student();
匿名对象:
new Student();
使用场景:
1. 用来临时调用一个方法
2. 可以作为方法参数
3. 可以作为方法的返回值
多态
多态是继封装、继承之后,面向对象的第三大特性。
现实事物经常会体现出多种形态,如学生,学生是人的一种,则一个具体的同学张三既是学生也是人,即出现两种形态。
Java 中多态的代码体现在一个子类对象 (实现类对象) 既可以给这个子类 (实现类对象) 引用变量赋值,又可以给这个子类 (实现类对象) 的父类 (接口) 变量赋值。
如 Student 类可以为 Person 类的子类。那么一个 Student 对象既可以赋值给一个 Student 类型的引用,也可以赋值给一个 Person 类型的引用。
^^^^ 最终多态体现为父类引用变量可以指向子类对象。!!!!!!!!!!!!!!!!!!!
多态的前提是必须有子父类关系或者类实现接口关系,否则无法完成多态。
在使用多态后的父类引用变量调用方法时,会调用子类重写后的方法。
具体格式如下:
父类引用指向子类对象就是多态的定义格式。同一个父类的方法会被不同的子类重写为各自的具体实现。在调用方法时,调用的为各个子类重写后的方法。
父类类型 变量名 = new 子类类型 ();
变量名. 方法名 ();
此时,虽然该变量指向的是子类对象,但表现为一个父类的形态,可以调用一切父类的方法,子类特有的方法将不能调用。
1 成员变量赋值、局部变量赋值
2 方法传参 (最常用最能体现出多态优点的应用)
3 返回返回值
当变量名指向不同的子类对象时,由于每个子类重写父类方法的内容不同,所以会调用不同的方法。
所以多态的存在意义 (优点) 为:
配合继承与方法重写提高了代码的复用性与扩展性,如果没有方法重写,则多态同样没有意义。
多态本身是子类类型向父类类型向上转型的过程。
多态的转型分为向上转型与向下转型两种:
1 向上转型:当有子类对象赋值给一个父类引用时,便是向上转型,多态本身就是向上转型的过程。
使用格式:
父类类型 变量名 = new 子类类型 ();
如:Person p = new Student();
2 向下转型:一个已经向上转型的子类对象可以使用强制类型转换的格式,将父类引用转为子类引用,这个过程是向下转型。如果是直接创建父类对象,是无法向下转型的!
使用格式:
子类类型 变量名 = (子类类型) 父类类型的变量;
如: Student stu = (Student) p; // 变量 p 实际上指向 Student 对象接口:
接口是功能的集合,同样可看做是一种数据类型,是比抽象类更为抽象的 "类"。
接口只描述所应该具备的方法,并没有具体实现,具体的实现由接口的实现类 (相当于接口的子类) 来完成。这样将功能的定义与实现分离,优化了程序设计。
请记住:一切事物均有功能,即一切事物均有接口。
与定义类的 class 不同,接口定义时需要使用 interface 关键字。
定义接口所在的仍为. java 文件,虽然声明时使用的为 interface 关键字的编译后仍然会产生. class 文件。这点可以让我们将接口看做是一种只包含了功能声明的特殊类。
定义格式:
public interface 接口名 {
抽象方法 1;
抽象方法 2;
抽象方法 3;
}
使用 interface 代替了原来的 class,其他步骤与定义类相同:
1 接口中的方法均为公共访问的抽象方法
2 接口中无法定义普通的成员变量
类与接口的关系为实现关系,即类实现接口。实现的动作类似继承,只是关键字不同,实现使用 implements。
其他类 (实现类) 实现接口后,就相当于声明:"我应该具备这个接口中的功能"。实现类仍然需要重写方法以实现具体的功能。
格式:
class 类 implements 接口 {
重写接口中方法
}
在类实现接口后,该类就会将接口中的抽象方法继承过来,此时该类需要重写该接口的所有抽象方法,完成具体的逻辑。
1 接口中定义功能,当需要具有该功能时,可以让类实现该接口,只声明了应该具备该方法,是功能的声明。
2 在具体实现类中重写方法,实现功能,是方法的具体实现。
于是,通过以上两个动作将功能的声明与实现便分开了。(此时请重新思考:类是现实事物的描述,接口是功能的集合。)
1、接口中可以定义变量,但是变量必须有固定的修饰符修饰,public static final 所以接口中的变量也称之为常量,其值不能改变。后面我们会讲解 static 与 final 关键字
2、接口中可以定义方法,方法也有固定的修饰符,public abstract
3、接口不可以创建对象。
4、子类必须覆盖掉接口中所有的抽象方法后,子类才可以实例化。否则子类是一个抽象类。
- interface Demo { ///定义一个名称为Demo的接口。
- public static final int NUM = 3; // NUM的值不能改变
- public abstract void show1();
- public abstract void show2();
- }
- //定义子类去覆盖接口中的方法。类与接口之间的关系是 实现。通过 关键字 implements
- class DemoImpl implements Demo { //子类实现Demo接口。
- //重写接口中的方法。
- public void show1() {}
- public void show2() {}
- }
1 接口可以继承接口
如同类继承类后便拥有了父类的成员,可以使用父类的非私有成员。A 接口继承 B 接口后,A 接口便拥有了 A、B 两个接口中所有的抽象方法。
2 Java 支持一个类同时实现多个接口,或一个接口同时继承多个接口。
3 类可以在继承一个类的同时,实现多个接口。
4 接口与父类的功能可以重复,均代表要具备某种功能,并不冲突
相同点:
1 都位于继承的顶端, 用于被其他类实现或继承;
2 都不能直接实例化对象;
3 都包含抽象方法, 其子类都必须覆写这些抽象方法;
区别:
1 抽象类为部分方法提供实现, 避免子类重复实现这些方法, 提高代码重用性; 接口只能包含抽象方法;
2 一个类只能继承一个直接父类 (可能是抽象类), 却可以实现多个接口;(接口弥补了 Java 的单继承) 二者的选用:
1 优先选用接口, 尽量少用抽象类;
2 需要定义子类的行为, 又要为子类提供共性功能时才选用抽象类;
来源: http://www.bubuko.com/infodetail-2436423.html