面向过程和面向对象
- 面向过程思维
- 步骤清晰简单,第一步做什么,第二部做什么...
- 面向过程适合处理一些较为简单的问题
- 面向对象思维
- 物以类聚,分类的思维模式,思考问题首先会解决问题需要哪些分类,然后对这些分类进行单独思考。最后,才对某个分类下的细节进行面向过程的思考。
- 面向对象适合处理复杂的问题,适合处理需要多人协作的问题。
- 对于描述复杂的事物,为了从宏观上把握,从整体上合理分析,我们需要使用面向对象的思维来分析整个系统。具体到围观操作,仍然需要面向过程的思维去处理。
面向对象
- 面向对象编程(Object-Oriented Programming, OOP):以类的方式组织代码,以对象的组织(封装)数据。
- 抽象
- 三大特性:
- 封装
- 继承
- 多态
- 从认识论角度考虑是先有对象后有类。对象是具体的事物,类是抽象的,是对对象的抽象。
- 从代码运行角度考虑是先有类后有对象。类是对象的模板。
类与对象的关系
- 类是一种抽象的数据类型,它是对某一类事情整体描述/定义,但是并不能代表某一个具体的事物。
- 对象是抽象概念的具体实例。
创建与初始化对象
- 使用new关键字创建对象
- 使用new关键字创建对象的时候,除了分配内存空间以外,还会给创建好的对象进行默认的初始化以及对类中构造器的调用。
//类:只包括属性和方法。
public class Person {
//属性
//类是一个模板,不可赋值。
String name;//未赋值,默认为null。
int age;//默认为0。
//方法
public void method(){
System.out.println(this.name+"MiSuShen");
}
}
public class Demo01 {
//一个项目只有一个main方法
public static void main(String[] args) {
//Person类是抽象的,需要实例化
//person对象是Person类的具体实例。
Person person = new Person();//实例化后(new Person();),会返回一个对象。person这个名字随便起。
//同一个类可以产生不同的对象。
Person person01 = new Person();
person.name = "弥玊生";
person.age = 18;
System.out.println(person.name);//弥玊生
System.out.println(person.age);//18
}
}
- 类中的构造器也称为构造方法,是在进行创建对象的时候必须要调用的。并且构造器有以下两个特点:
- 必须和类的名字相同。
- 必须没有返回类型,也不能写void。
public class Person {
String name;
//构造器一般用来初始化对象的值
//无参构造(一个类即使什么也不写,也会存在一个无参构造)
public Person(){
}
//有参构造(一旦定义了有参构造,无参构造就必须显示定义)
public Person(String name){
this.name = name;
}
}
public class Demo01 {
public static void main(String[] args) {
//使用new关键字本质是在调用构造器
Person person = new Person("弥玊生");//若传了参数,就会调用有参构造器。
System.out.println(person.name);//弥玊身
}
}
- 创建构造快捷键alt+ins
创建对象内存分析简单分析
public class Person{
public String name;
public int age;
public void said(){
System.out.println("Hello");
}
}
public class Demo01{
public static void main(String[] args){
Person personOne = new Person();
personOne.name = "弥玊生";
personOne.age = "18";
personOne.said();
System.out.println(personOne.name);
System.out.println(personOne.age);
Person personTwo = new Person();
}
}
面向对象的三大特性
封装
- 程序设计追求“高内聚,低耦合”。高内聚:类的内部数据操作细节自己完成,不允许外部干涉;低耦合:仅暴露少量的方法给外部使用。
- 封装(隐藏的数据):通常,应禁止直接访问一个对象中数据的实际表示,而应通过操作接口来访问,这称为信息隐藏。
- 属性私有 get/set
- 封装的作用:
- 提高程序的安全性,保护数据。
- 隐藏代码的实现细节。
- 统一接口
- 系统可维护性提高
public class Person{
private String name; //已被私有(private),替换成public则变为公共的。
private int age;
//get 获得这个数据
public String getName(){
return this.name;
}
//set 给这个数据设置值
public void setName(String name){
this.name = name;
}
public int getAge(){
return age;
}
//
public void setAge(int age){
if (age < 150 && age > 0){
this.age = age;
}else{
System.out.println("输入超限");
}
}
//at + ins可自动生成get、set方发
}
public class Demo{
public static void main(String[] args){
Person person = new Person();
// (p1.name = )不能直接赋值,已被私有
person.setName("弥玊生");
System.out.println(person.getName());
person.setAge(18);
System.out.println(person.getAge());
}
}
继承
- 继承的本质是对某一批类的抽象,从而实现对现实世界更好的建模。
- extends“拓展”。子类是父类的扩展。
- Java中类只有单继承,没有多继承。
- 继承是类和类之间的一种关系。除此之外,类和类之间的关系还有依赖、组合、聚合等。
- 继承关系的两个类,一个为子类(派生类),一个为父类(基类)。子类继承父类,使用关键字extends来表示。
- 子类和父类之间,从意义上讲应该具有“is a”的关系。
- ctrl + H可查看继承关系
- Java中,所有的类,都默认直接或间接继承Object类
public class Father{ //父类
public void said(){
System.out.println("弥玊生");
}
}
public class Son extends Father{ //子类
}
public class Demo{
public static void main(String[] args){
Son son = new Son();
son.said(); //弥玊生,若修饰符未private(封装)则不能直接继承(见封装笔记)
}
}
super
public class Father{
protected String name = "a";
public void print(){
System.out.println("Son!");
}
}
public class Son extends Father{
private String name = "b";
public void print(){
System.out.println("Father!")
}
public void test01(String name){
System.out.println(name); //c
System.out.println(this.name); //b
System.out.println(super.name); //a
public void test02(){
print(); // Father!
this.print(); // Father!
super.print(); // Son!
}
}
}
public class Demo{
public static void main(String[] args){
Son son = new Son();
son.test01("c")
son.test02();
}
}
- 父类和子类的构造器
- 子类默认调用父类无参构造。
- 调用父类的构造器,必须要在子类构造器的第一行。
- 若父类没有无参构造,子类无参构造可以调用父类有参构造。(要显式调用,否则默认调用无参会报错)
- super和this不能同时调用构造方法。
public class Father{
public Father(){
}
}
public class Son extends Father{
public Son(){
super(); //调用父类的构造器,必须要在子类构造器的第一行。(默认,不用写出)
}
}
方法重写
- 方法重写需要有继承关系,子类重写父类的方法。
- 方法名需相同。
- 参数列表需相同。
- 修饰符范围可以扩大,但不能缩小:public > protected > Default > private
- 抛出异常:范围可以缩小,但不能扩大。
- 方法重写的作用:父类的功能子类不一定需要或满足。
public class Father{
public void test(){
System.out.println("Son!");
}
}
public class Son extends Father{
@Override
public void test(){ // 快捷键ctrl+ins-->Override Methods
System.out.println("Father!");}
public class Demo{
public static void main(String[] args){
Son son = new Son();
son.test(); //Father!
Father father = new Son(); //子类重写了父类的方法
father.test(); //Father!
}
}
- 若用static静态方法,则不能构成重写。(final,,private也不能重写)
public class Father{
public static void test(){
System.out.println("Son!");
}
}
public class Son extends Father{
public static void test(){
System.out.println("Father!");
}
public class Demo{
public static void main(String[] args){
Son son = new Son();
son.test(); //Father!
Father father = new Son(); //父类的引用指向了子类,若用static,方法的调用只和左边定义的数据类型有关。
father.test(); //Son!
}
}
多态
-
即同一方法可以根据发送对象的不同而采用多种不同的行为方式。(方法的多态,属性没有多态)
-
一个对象的实际类型是确定的,但可以指向对象的引用的类型有很多。
-
多态存在的条件
- 有继承关系
- 子类重写父类方法
- 父类引用指向子类对象
public class Father{
public void method01(){
System.out.println("Father");
}
}
public class Son extends Father{
@Override
public void method01(){
System.out.println("Son");
public void method02(){
System.out.println("son");
}
}
}
public class Demo{
public static void main(String[] args){
//对象的实际类型(new Son();)是确定的,但指向的引用类型(Son/Father)有很多种。
Son p1 = new Son(); //可以调用子类或继承父类的方法。
Father p2 = new Son(); //父类引用指向子类对象。(不能调用子类独有的方法)
Object p3 = new Son();
//对象能执行哪些方法,主要看对象左边的类型。
p1.method01(); //Son
p2.method01(); //Son,子类重写了父类的方法,执行子类的方法。若子类没有重写则调用父类,Father。
//p2.method02();父类没有该方法,不能调用。
((Son)p2).method02(); //强制转换
}
}
instanseof
public class Father{
}
public class Son extends Father{
}
public class GrandSon extends Father{
}
public class Demo{
public static void main(String[] args){
//Object > String
//Object > Father > Son
//Object > Father > GrandSon
Object object = new Son();
System.out.println(object instanceof Son); //Ture
System.out.println(object instanceof Father); //Ture
System.out.println(object instanceof Object); //Ture
System.out.println(object instanceof GrandSon); //False
System.out.println(object instanceof String); //False
Father father = new Son();
System.out.println(father instanceof Son); //Ture
System.out.println(father instanceof Father); //Ture
System.out.println(father instanceof Object); //Ture
System.out.println(father instanceof GrandSon); //false
// System.out.println(father instanceof String); 编译报错 Father类与String类没有父子关系
Son son = new Son();
System.out.println(Son instanceof Son); //Ture
System.out.println(Son instanceof Father); //Ture
System.out.println(Son instanceof Object); //Ture
// System.out.println(Son instanceof GrandSon); 编译报错
// System.out.println(Son instanceof String); 编译报错
}
}
类型转换
public class Father{
public void method01();
System.out.println("method01");}
public class Son extends Father{
public void method02();
System.out.println("method02");}
public class Demo{
public static void main(String[] args){
//高转低
Father obj = new Son(); //只能父类引用指向子类的对象
((Son)obj).method02(); //父类转为子类(强制转换,方便方法调用,减少重复代码),Son将这个对象转换为Son类型,就可以使用Son类型的方法
/*也可写成:
Son son = (Student) obj;
son.method02();
*/
//低转高
Son son = new Son();
son.method02();
Father father = son; //子类转为父类,可能丢失一些方法
}
}