包的本质 实际上就是创建不同的文件夹来保存类文件
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dUKyUYQ7-1645795496832)(C:\Users\xiaban\Pictures\java学习\包的本质分析.png)]
package com.xxxx;
说明
package com.use; import com.xiaoqiang.Dog; public class Test { public static void main(String[] args) { Dog dog = new Dog(); System.out.println(dog);//com.xiaoqiang.Dog@1b6d3586 com.xiaoming.Dog dog1 = new com.xiaoming.Dog(); System.out.println(dog1);//com.xiaoming.Dog@4554617c } }
**命名规则:**只能包含数字、字母、下划线、小圆点,但不能用数字开头,不能是关键字或者保留字
**命名规范:**一般是小写字母+小一点
一般是 com.公司名.项目名.业务模块名
例如:com.sina.crm.user
一个包下,包含很多的类,java中常用的包由:
java.lang.* //lang包是基本包,默认引入,不需要再引入
java.util.* //util包,系统提供的工具包,工具类
java.net.* //网络包,网络开发
java.awt.* //是坐java的界面开发,GUI
语法:import 包;
我们引入一个包的主要目的是要使用该包下的类
比如:
import java.util.Scanner;就只是引入一个类Scanner
import java.util.*; 表示将java.util包所有类都引入
package com.xiaban; import java.util.Arrays; //注意: //我们需要用到哪个类,就导入哪个类 //import java.util.Scanner;//表示只会引入一个类Scanner //import java.util.*;//表示将java.util包所有类都引入 public class Import01 { public static void main(String[] args) { int[] arr = {-1,20,2,13,3}; Arrays.sort(arr); for (int i = 0; i < arr.length; i++) { System.out.print(arr[i] + " ");//-1 2 3 13 20 } } }
java提供四种访问修饰符号,用于控制方法和属性(成员变量)的访问权限(范围):
封装(encapsulation)就是把抽象出的数据[属性]和对数据的操作[方法]封装在一起,数据被保护在内部,程序的其它部分只有通过被授权的操作[方法],才能对数据进行操作——对电视机的操作就是典型封装
package com.xiaban.encap; public class Encapsulation01 { public static void main(String[] args) { Person person = new Person(); person.setName("jack"); person.setAge(30); person.setSalary(30000); System.out.println(person.info()); } } class Person{ public String name;//名字公开 private int age;//age 私有化 private double salary;//.. public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { if(age >= 1 && age <= 120){ this.age = age; }else{ System.out.println("年龄需要在1-120之间"); this.age = 18;//给一个默认值 } } public double getSalary() { return salary; } public void setSalary(double salary) { this.salary = salary; } public String info() { return "消息为name = " + name + " age = " + age + " 薪水 = " + salary; } }
将封装和构造器结合
package com.xiaban.encap; public class Encapsulation01 { public static void main(String[] args) { Person person = new Person(); person.setName("jack"); person.setAge(30); person.setSalary(30000); System.out.println(person.info()); Person smith = new Person("smith", 18, 50000); System.out.println(smith.info()); } } class Person{ public String name;//名字公开 private int age;//age 私有化 private double salary;//.. public Person(){ } public Person(String name, int age, double salary) { this.name = name; this.age = age; this.salary = salary; setName(name); setAge(age); setSalary(salary); } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { if(age >= 1 && age <= 120){ this.age = age; }else{ System.out.println("年龄需要在1-120之间"); this.age = 18;//给一个默认值 } } public double getSalary() { return salary; } public void setSalary(double salary) { this.salary = salary; } public String info() { return "消息为name = " + name + " age = " + age + " 薪水 = " + salary; } }
继承可以解决代码复用,让我们的编程更加靠近人类思维,当多个类存在相同的属性(变量)和方法时,可以从这些类中抽象出父类,在父类中定义这些相同的属性和方法,所有的子类不需要重新定义这些属性和方法,值需要通过extends来声明继承父类即可
class 子类 extends 父类{
}
注意
父类
package com.xiaban.extend_.improve_; public class Student { String name; int age; private double score; public void setScore(double score) { this.score = score; } public void showInfo(){ System.out.println("学生名"+name+"年龄"+age+"得了"+score+"分"); } }
子类1
package com.xiaban.extend_.improve_; public class Pupil extends Student{ public void testing(){ System.out.println("小学生"+name+"正在考式"); } }
子类2
package com.xiaban.extend_.improve_; public class Graduate extends Student{ public void testing(){ System.out.println("大学生"+name+"正在考式"); } }
子类继承了所有的属性和方法,但是私有属性不能再子类直接访问,要通过公共的方法访问
(即父类的公共方法里返回一个属性,然后在子类调用父类的公共方法)
子类必须调用父类的构造器,完成父类的初始化
当创建子类对象时,不管使用子类的哪个构造器,默认情况下总会去调用父类的无参构造器,如果父类没有提供无参构造器,则必须在子类构造器中用super去指定使用父类的哪个构造器完成对父类的初始化工作
如果希望指定去调用父类的某个构造器,则显式的调用一下:super(参数列表)
public Sub(String name){ super("pick"); System.out.println("Sub(String name)构造器被调用"); }
super()和this()都只能放在构造器的第一行,因此俩个方法不能共存在一个构造器
java所有类都是Object类的子类
IDEA ctrl+H查看继承关系
父类构造器的调用不限于直接父类,将一直往上追溯直到Object类(顶级父类)
子类最多只能继承一个父类(指直接继承),即java中是单继承机制
不能滥用继承,子类和父类之间必须满足is-a的逻辑关系(规范)
package com.xiaban.extend_; public class ExtendsTheory { public static void main(String[] args) { Son son = new Son(); System.out.println(son.name);//"大头蛾子" } } class GrandPa{ String name = "大头爷爷"; String hobby = "旅游"; } class Father extends GrandPa{ String name = "大头爸爸"; int age = 39; } class Son extends Father { String name = "大头蛾子"; }
super代表父类的引用,用于访问父类的属性、方法、构造器
package com.xiaban.super_; public class B extends A{ public void hi(){ System.out.println(super.n1 + " " + super.n2 + " " + super.n3); } public void ok(){ super.test100(); super.test200(); super.test300(); } public B() { //super(); //super("smith"); super("smith",6); } }
No. | 区别点 | this | super |
---|---|---|---|
1 | 访问属性 | 访问本类中的属性,如果本类没有此属性,则从父类中继续查找 | 从父类开始查找属性 |
2 | 调用方法 | 访问本类中的方法,如果本类没有此方法则从父类继续查找 | 从父类开始查找方法 |
3 | 调用构造器 | 调用本类构造器 | 调用父类构造器 |
4 | 特殊 | 表示当前对象 | 子类中访问父类对象 |
简单来说,方法重写(覆盖)就是子类的一个方法,和父类的某个方法的名称、返回类型、参数一样,那么我们就说子类的这个方法重写/覆盖了父类的那个方法
父类
package com.xiaban.override_; public class Animal { public void cry(){ System.out.println("动物叫唤"); } }
子类
package com.xiaban.override_; public class Dog extends Animal{ public void cry(){ System.out.println("旺旺旺"); } }
方法调用
package com.xiaban.override_; public class Override01 { public static void main(String[] args) { Dog dog = new Dog(); dog.cry();//ctrl+b查找cry //输出 旺旺旺 } }
方法重写也叫方法覆盖,需要满足下面的条件
子类的方法的形参列表、方法名称要和父类方法的形参列表名称完全一样
子类方法的返回类型和父类方法返回类型一样,或者是父类返回类型的子类
public Object m1(){} public String m1(){} //上面这样也能构成方法重写
但是返回类型为Object的必须为父类
子类方法不能缩小父类方法的访问权限
名称 | 发生范围 | 方法名 | 形参列表 | 返回类型 | 修饰符 |
---|---|---|---|---|---|
重载(overload) | 本类 | 必须一样 | 类型,个数或者顺序至少有一个不同 | 无要求 | 无要求 |
重写(override) | 父子类 | 必须一样 | 相同 | 子类重写的方法的返回类型和父类方法的返回类型一致或者是其子类 | 子类方法不能缩小父类方法的访问范围 |
方法或对象具有多种形态。是面向对象的第三大特征,多态是建立在封装和继承基础之上的
多态的具体表现
package com.xiaban.poly_; public class PloyMethod { public static void main(String[] args) { A a = new A(); //方法重载 System.out.println(a.sum(10,20));//30 System.out.println(a.sum(10,20,30));//60 //多态 B b = new B(); a.say();//A say()方法被调用 b.say();//B say()方法被调用 } } class B{ public void say(){ System.out.println("B say()方法被调用"); } } class A extends B { public int sum(int n1, int n2){ return n1 + n2; } public int sum(int n1,int n2, int n3){ return n1 + n2 + n3; } public void say(){ System.out.println("A say()方法被调用"); } }
一个对象的编译类型和运行类型可以不一致
Animal animal = new Dog();//animal编译类型是Animal,运行类型是Dog
编译类型在定义对象时,就确定了,不能改变
运行类型是可以变化的
package com.xiaban.poly_.PolyObject_; public class PolyObject { public static void main(String[] args) { Animal animal = new Dog(); animal.cry();//运行类型是Dog 所以调用的是Dog 的cry() animal = new Cat(); animal.cry();//运行类型是Cat 所以调用的是Cat 的cry() } }
多态的前提是:两个对象(类)存在继承关系
多态的向上转型
package com.xiaban.poly_.detail; public class PolyDetail { public static void main(String[] args) { Animal animal = new Cat(); Object object = new Cat(); //animal.catchMouse();//catchMouse是Animal的子类Cat的方法,所以不能用 animal.eat();//猫吃鱼 animal.run();//跑 animal.show();//hello animal.sleep();//睡 } } public class Animal { String name = "动物"; int age = 10; public void sleep(){ System.out.println("睡"); } public void run(){ System.out.println("跑"); } public void eat(){ System.out.println("吃"); } public void show(){ System.out.println("hello"); } } public class Cat extends Animal{ public void eat(){//方法重写 System.out.println("猫吃鱼"); } public void catchMouse(){ System.out.println("猫抓老鼠"); } }
多态的向下转型
Cat cat = (Cat) animal; cat.catchMouse();//猫抓老鼠
注意
package com.xiaban.poly_.dynamic_; public class Dynamic { public static void main(String[] args) { A a = new B(); System.out.println(a.sum());//30 System.out.println(a.sum1());//110 System.out.println(a.getI());//10 } } class A { public int i = 10; public int sum(){ return getI() + 10; } public int sum1(){ return i + 100; } public int getI(){ return i; } } class B extends A{ public int i = 20; // public int sum(){ // return i + 20; // } public int getI(){ return i; } // public int sum1(){ // return i + 10; // } }
数组的定义类型为父类类型,里面保存的实际元素类型为子类类型
public class PloyArr { public static void main(String[] args) { Person[] persons = new Person[5]; persons[0] = new Person("jack",20); persons[1] = new Student("caoc",20,80); persons[2] = new Student("smith",19,50); persons[3] = new Teacher("scott",30,200000); persons[4] = new Teacher("king",29,100000); for (int i = 0; i < persons.length; i++){ System.out.println(persons[i].say()); } } } class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String say(){ return name + "\t" + age; } } class Student extends Person{ private double score; public Student(String name, int age, double score) { super(name, age); this.score = score; } public double getScore() { return score; } public void setScore(double score) { this.score = score; } public String say(){ return super.say() + "score = " + score; } } class Teacher extends Person{ private double salary; public Teacher(String name, int age, double salary) { super(name, age); this.salary = salary; } public double getSalary() { return salary; } public void setSalary(double salary) { this.salary = salary; } public String say(){ return super.say() + "salary = " + salary; } }
上面输出的结果为
jack 20
caoc 20 score = 80.0
smith 19 score = 50.0
scott 30 salary = 200000.0
king 29 salary = 100000.0
方法定义的形参类型为父类类型,实参类型允许为子类类型