当前位置: 移动技术网 > IT编程>开发语言>Java > Java基础(面向对象继承、修饰符、内部类)

Java基础(面向对象继承、修饰符、内部类)

2020年08月17日  | 移动技术网IT编程  | 我要评论
文章目录1. 继承1.1 继承的概念1.2 继承的好处和弊端2. 继承中的成员访问特点2.1 继承中变量的访问特点2.2 super关键字2.3 继承中构造方法的访问特点2.4 继承中成员方法的访问特点2.5 super内存图2.6 方法重写2.7 方法重写的注意事项2.8. Java中继承的注意事项3. 修饰符3.1 package3.2 import3.3 权限修饰符3.4 final3.5 final修饰局部变量3.6 static3.7 static访问特点1. 继承1.1 继承的概念继承



1. 继承

1.1 继承的概念

  • 继承是面向对象三大特征之一,子类继承父类的属性和行为,使得子类对象具有与父类相同的属性、相同的行为。子类可以直接 访问父类中的非私有的属性和行为

  • 实现继承的格式

    • 继承通过extends实现
    • 格式:class 子类 extends 父类 { }
      • 举例:class Dog extends Animal { }
 class 父类 { ... } class 子类 extends 父类 { ... } 
  • 继承带来的好处

    • 继承可以让类与类之间产生关系,子父类关系,产生子父类后,子类则可以使用父类中非私有的成员。
    • 提高代码的复用性。
  • 示例代码

 public class Fu { public void show() { System.out.println("show方法被调用"); } } public class Zi extends Fu { public void method() { System.out.println("method方法被调用"); } } public class Demo { public static void main(String[] args) { //创建对象,调用方法 Fu f = new Fu(); f.show(); Zi z = new Zi(); z.method(); z.show(); } } 

1.2 继承的好处和弊端

  • 继承好处
    • 提高了代码的复用性(多个类相同的成员可以放到同一个类中)
    • 提高了代码的维护性(如果方法的代码需要修改,修改一处即可)
  • 继承弊端
    • 继承让类与类之间产生了关系,类的耦合性增强了,当父类发生变化时子类实现也不得不跟着变化,削弱了子类的独立性
  • 继承的应用场景:
    • 使用继承,需要考虑类与类之间是否存在is…a的关系,不能盲目使用继承
      • is…a的关系:谁是谁的一种,例如:老师和学生是人的一种,那人就是父类,学生和老师就是子类

2. 继承中的成员访问特点

2.1 继承中变量的访问特点

在子类方法中访问一个变量,采用的是就近原则。

  1. 子类局部范围找
  2. 子类成员范围找
  3. 父类成员范围找
  4. 如果都没有就报错(不考虑父亲的父亲…)
  • 示例代码
 class Fu { int num = 10; } class Zi { int num = 20; public void show(){ int num = 30; System.out.println(num); } } public class Demo1 { public static void main(String[] args) { Zi z = new Zi(); z.show(); // 输出show方法中的局部变量30 } } 

2.2 继承中构造方法的访问特点

注意:子类中所有的构造方法默认都会访问父类中无参的构造方法

子类会继承父类中的数据,可能还会使用父类的数据。所以,子类初始化之前,一定要先完成父类数据的初始化,原因在于,每一个子类构造方法的第一条语句默认都是:super()

问题:如果父类中没有无参构造方法,只有带参构造方法,该怎么办呢?

  1. 通过使用super关键字去显示的调用父类的带参构造方法
  2. 在父类中自己提供一个无参构造方法

推荐方案:

自己给出无参构造方法

2.3继承中成员方法的访问特点

通过子类对象访问一个方法

  1. 子类成员范围找
  2. 父类成员范围找
  3. 如果都没有就报错(不考虑父亲的父亲…)

2.4 super关键字

  • this&super关键字:
    • this:代表本类对象的引用
    • super:代表父类存储空间的标识(可以理解为父类对象引用)
  • this和super的使用分别
    • 成员变量:
      • this.成员变量 - 访问本类成员变量
      • super.成员变量 - 访问父类成员变量
    • 成员方法:
      • this.成员方法 - 访问本类成员方法
      • super.成员方法 - 访问父类成员方法
  • 构造方法:
    • this(…) - 访问本类构造方法
    • super(…) - 访问父类构造方法

2.5super内存图

  • 对象在堆内存中,会单独存在一块super区域,用来存放父类的数据

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0TYCpNsQ-1597496582010)(F:\TeachRes\2019\JavaSe\day01\笔记\笔记资源包\图片1.png)]

2.6 方法重写

  1. 方法重写概念
    • 子类出现了和父类中一模一样的方法声明(方法名一样,参数列表也必须一样)
  2. 方法重写的应用场景
    • 当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法,这样,即沿袭了父类的功能,又定义了子类特有的内容
  3. Override注解
    • 用来检测当前的方法,是否是重写的方法,起到【校验】的作用

2.7 方法重写的注意事项

  • 方法重写的注意事项
  1. 私有方法不能被重写(父类私有成员子类是不能继承的)
  2. 子类方法访问权限不能更低(public > 默认 > 私有)
  • 示例代码
public class Fu { private void show() { System.out.println("Fu中show()方法被调用"); } void method() { System.out.println("Fu中method()方法被调用"); } } public class Zi extends Fu { /* 编译【出错】,子类不能重写父类私有的方法*/ @Override private void show() { System.out.println("Zi中show()方法被调用"); } /* 编译【出错】,子类重写父类方法的时候,访问权限需要大于等于父类 */ @Override private void method() { System.out.println("Zi中method()方法被调用"); } /* 编译【通过】,子类重写父类方法的时候,访问权限需要大于等于父类 */ @Override public void method() { System.out.println("Zi中method()方法被调用"); } } 

2.8. Java中继承的注意事项

  • Java中继承的注意事项

    1. Java中类只支持单继承,不支持多继承
      • 错误范例:class A extends B, C { }
    2. Java中类支持多层继承
  • 多层继承示例代码:

 public class Granddad { public void drink() { System.out.println("爷爷爱喝酒"); } } public class Father extends Granddad { public void smoke() { System.out.println("爸爸爱抽烟"); } } public class Mother { public void dance() { System.out.println("妈妈爱跳舞"); } } public class Son extends Father { // 此时,Son类中就同时拥有drink方法以及smoke方法 } 

3. 修饰符

3.1 package

  • 1、包的概念
    • 包就是文件夹,用来管理类文件的
  • 2、包的定义格式
    • package 包名; (多级包用.分开)
    • 例如:package com.heima.demo;
  • 3、带包编译&带包运行
    • 带包编译:javac –d . 类名.java
      • 例如:javac -d . com.heima.demo.HelloWorld.java
    • 带包运行:java 包名+类名
      • 例如:java com.heima.demo.HelloWorld

3.2 import

  • 导包的意义

    使用不同包下的类时,使用的时候要写类的全路径,写起来太麻烦了

    为了简化带包的操作,Java就提供了导包的功能

  • 导包的格式

    格式:import 包名;

    范例:import java.util.Scanner;

  • 示例代码(没有使用导包,创建的Scanner对象)

package com.heima; public class Demo { public static void main(String[] args) { // 1. 没有导包,创建Scnaner对象 java.util.Scanner sc = new java.util.Scanner(System.in); } } 
  • 示例代码(使用导包后,创建的Scanner对象)
package com.heima; import java.util.Scanner; public class Demo { public static void main(String[] args) { // 1. 没有导包,创建Scnaner对象 Scanner sc = new Scanner(System.in); } } 

3.3 权限修饰符

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZYXCNkGI-1597496582018)(F:\TeachRes\2019\JavaSe\day01\笔记\笔记资源包\图片2.png)]

3.4 final

  • fianl关键字的作用
    • final代表最终的意思,可以修饰成员方法,成员变量,类
  • final修饰类、方法、变量的效果
    • fianl修饰类:该类不能被继承(不能有子类,但是可以有父类)
    • final修饰方法:该方法不能被重写
    • final修饰变量:表明该变量是一个常量,不能再次赋值

3.5 final修饰局部变量

  • fianl修饰基本数据类型变量

    • final 修饰指的是基本类型的数据值不能发生改变
  • final修饰引用数据类型变量

    • final 修饰指的是引用类型的地址值不能发生改变,但是地址里面的内容是可以发生改变的

    • 举例:

 public static void main(String[] args){ final Student s = new Student(23); s = new Student(24); // 错误 s.setAge(24); // 正确 } 

3.6 static

  • static的概念
    • 关于 static 关键字的使用,它可以用来修饰的成员变量和成员方法,被修饰的成员是属于类的,而不是单单是属 于某个对象的。也就是说,既然属于类,就可以不靠创建对象来调用了
    • static关键字是静态的意思,可以修饰【成员方法】,【成员变量】
  • static修饰的特点
    1. 被类的所有对象共享,这也是我们判断是否使用静态关键字的条件
    2. 可以通过类名调用当然,也可以通过对象名调用**【推荐使用类名调用】**
  • 示例代码:
class Student { public String name; //姓名 public int age; //年龄 public static String university; //学校	共享数据!所以设计为静态! public void show() { System.out.println(name + "," + age + "," + university); } } public class StaticDemo { public static void main(String[] args) { // 为对象的共享数据赋值 Student.university = "传智大学"; Student s1 = new Student(); s1.name = "林青霞"; s1.age = 30; s1.show(); Student s2 = new Student(); s2.name = "风清扬"; s2.age = 33; s2.show(); } } 

3.7 static访问特点

  • static的访问特点
    • 非静态的成员方法
      • 能访问静态的成员变量
      • 能访问非静态的成员变量
      • 能访问静态的成员方法
      • 能访问非静态的成员方法
    • 静态的成员方法
      • 能访问静态的成员变量
      • 能访问静态的成员方法
    • 总结成一句话就是:
      • 静态成员方法只能访问静态成员

4. 内部类

4.1 内部类的基本使用

  • 内部类概念

    • 在一个类中定义一个类。举例:在一个类A的内部定义一个类B,类B就被称为内部类
  • 内部类定义格式

    • 格式&举例:
 /*
    	格式:
        class 外部类名{
        	修饰符 class 内部类名{
        	
        	}
        }
    */ class Outer { public class Inner { } } 
  • 内部类的访问特点

    • 内部类可以直接访问外部类的成员,包括私有
    • 外部类要访问内部类的成员,必须创建对象
  • 示例代码:

 /*
      内部类访问特点:
          内部类可以直接访问外部类的成员,包括私有
          外部类要访问内部类的成员,必须创建对象
   */ public class Outer { private int num = 10; public class Inner { public void show() { System.out.println(num); } } public void method() { Inner i = new Inner(); i.show(); } } 

4.2 成员内部类

  • 成员内部类的定义位置

    • 在类中方法,跟成员变量是一个位置
  • 外界创建成员内部类格式

    • 格式:外部类名.内部类名 对象名 = 外部类对象.内部类对象;
    • 举例:Outer.Inner oi = new Outer().new Inner();
  • 成员内部类的推荐使用方案

    • 将一个类,设计为内部类的目的,大多数都是不想让外界去访问,所以内部类的定义应该私有化,私有化之后,再提供一个可以让外界调用的方法,方法内部创建内部类对象并调用。
  • 示例代码:

    class Outer { private int num = 10; private class Inner { public void show() { System.out.println(num); } } public void method() { Inner i = new Inner(); i.show(); } } public class InnerDemo { public static void main(String[] args) { //Outer.Inner oi = new Outer().new Inner(); //oi.show(); Outer o = new Outer(); o.method(); } } 

4.3 局部内部类

  • 局部内部类定义位置

    • 局部内部类是在方法中定义的类
  • 局部内部类方式方式

    • 局部内部类,外界是无法直接使用,需要在方法内部创建对象并使用
    • 该类可以直接访问外部类的成员,也可以访问方法内的局部变量
  • 示例代码

 class Outer { private int num = 10; public void method() { int num2 = 20; class Inner { public void show() { System.out.println(num); System.out.println(num2); } } Inner i = new Inner(); i.show(); } } public class OuterDemo { public static void main(String[] args) { Outer o = new Outer(); o.method(); } } 

4.4 匿名内部类

  • 匿名内部类的前提

    • 存在一个类或者接口,这里的类可以是具体类也可以是抽象类
  • 匿名内部类的格式

    • 格式:new 类名 ( ) { 重写方法 } new 接口名 ( ) { 重写方法 }

    • 举例:

 new Inter(){ @Override public void method(){} } 
  • 匿名内部类的本质

    • 本质:是一个继承了该类或者实现了该接口的子类匿名对象
  • 匿名内部类的细节

    • 匿名内部类可以通过多态的形式接受
 Inter i = new Inter(){ @Override public void method(){ } } 
  • 匿名内部类直接调用方法
 interface Inter{ void method(); } class Test{ public static void main(String[] args){ new Inter(){ @Override public void method(){ System.out.println("我是匿名内部类"); } }.method(); // 直接调用方法 } } 

匿名内部类在开发中的使用

  • 匿名内部类在开发中的使用

    • 当发现某个方法需要,接口或抽象类的子类对象,我们就可以传递一个匿名内部类过去,来简化传统的代码
  • 示例代码:

 interface Jumpping { void jump(); } class Cat implements Jumpping { @Override public void jump() { System.out.println("猫可以跳高了"); } } class Dog implements Jumpping { @Override public void jump() { System.out.println("狗可以跳高了"); } } class JumppingOperator { public void method(Jumpping j) { //new Cat();   new Dog(); j.jump(); } } class JumppingDemo { public static void main(String[] args) { //需求:创建接口操作类的对象,调用method方法 JumppingOperator jo = new JumppingOperator(); Jumpping j = new Cat(); jo.method(j); Jumpping j2 = new Dog(); jo.method(j2); System.out.println("--------"); // 匿名内部类的简化 jo.method(new Jumpping() { @Override public void jump() { System.out.println("猫可以跳高了"); } }); // 匿名内部类的简化 jo.method(new Jumpping() { @Override public void jump() { System.out.println("狗可以跳高了"); } }); } } 

本文地址:https://blog.csdn.net/qq_36102598/article/details/108028253

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网