面向对象概念
- 面向对象三大特征:封装,继承,多态
- 面向对象编程(OOP,Object Oriented Programing)是相对于面向过程编程说的,之前写的代码基本都是纯的面向过程编程的,当项目复杂了,那么纯面向过程代码实现会很复杂,面向对象可以简化代码的结构和组织关系。面向对象不是替代面向过程的,宏观是面向对象,微观仍然是面向过程。
- 类(class)和对象(Object):人、王力宏
类的定义
class Person{ private int age;//成员变量 private String name; public void setAge(int age) { this.age=age; } public void setName(String name) { this.name=name; } public void sayHello() { System.out.println(“大家好,我是”+name+“我今年”+age+“岁了”); } }
- 一个Java文件中只能定义一个public的class,且文件名必须和public类一样
对象的实例化
Person yzk=new Person(); yzk.setName("杨中科"); yzk.setAge(18); yzk.sayHello();
Person lzy=new Person(); lzy.setName("林志颖"); lzy.setAge(80); lzy.sayHello(); //两个对象的内存分配:根据模板拷贝两份。 yzk.sayHello(); 每new出的每个对象都是一个单独的实例,两个对象之间的成员变量是独立的两份。new出来的叫类对象或者实例(Instance)
类对象是引用传递 Person weiren=yzk; weiren.setAge(18); weiren.sayHello(); yzk.sayHello(); 解释:“Person weiren=yzk;”weiren指向yzk当前所指向的对象;
成员变量和局部变量 public void setAge(int age) { this.age=age; } 1、局部变量必须初始化,成员变量声明时默认初始化,基本数值类型默认初始化为int,String等非基本类型初始化为null。why?涉及到栈内存和堆内存,以后讲。 2、当成员变量和局部变量(函数参数也可以看做局部变量)重名的时候,被看做局部变量,因此为了避免混乱,建议访问成员变量的时候加上“this”,this代表当前对象。 3、new Person().sayHello();创建一个匿名对象,然后调用。
private/public
- 我们可以把age成员变量声明为public,也可以把setAge声明为private,声明为private,这样就只能在类内部调用private成员(再写一个方法调用private的setAge)
- 结论:public成员可以被类内部或者外部访问,private成员只能被类内部访问,这样可以保护不希望外界调用的内部成员(Member,包含字段Field/变量、方法)不被外界访问。
- 直接通过public的age设置年龄,不通过setAge赋值,这样有什么坏处?
- 字段(Field)/成员变量(Member Variable)一般声明为private
构造函数(Constructor)
- 构造函数是创建类对象,并且在创建完成前对类进行初始化的特殊函数。如果定义类时没有声明构造函数,默认会给出一个无参构造函数,如果定义了任意一个构造函数,将不会提供默认的无参构造函数。
- 构造方法格式及特点:
方法名必须和类名一致 没有返回值类型 3、构造函数可以重载,Person(String name,int age) 4、用构造函数初始化和new之后通过set***初始化的区别:语意,有的对象生来就要有一些成员变量被赋值的,否则就是怪胎
静态static
- 一些场景下会要求一个类的多个实例共享一个成员变量;有时候要定义“常量”,但是Java没有提供常量;有时候想定义一些不和具体对象关联、不需要new就调用方法
- static方法不需要new就可以直接通过类名调用。
- static变量不需要new就可以直接通过类名调用。
- static 方法中无法使用this关键字,因为static独立于对象存在,不是任何人的唯一。
- static成员中只能访问static成员,不能直接访问非static成员。非static成员可以访问static成员。
单例模式
- 有的类在系统中只能有一个对象(*,资源管理器,缓存管理器等),这时就要使用“单例模式”(singleton)。实现单例模式有很多方法,先介绍最简单,最实用的“饿汉式”
- 构造函数声明为private,这样避免外界访问
- 定义一个private final static的对象实例
- 定义一个public static的getInstance方法,返回唯一实例。
类的静态代码块 class MyTest { static { System.out.println(“zi 静态代码块”) } public MyTest() { System.out.println(“zi 构造方法”) } } MyTest t1=new MyTest(); MyTest t2=new MyTest(); 静态代码块在类第一次被使用的时候执行一次,在构造函数执行之前执行,只要用到类,哪怕没new对象(比如只是声明变量)也会被执行,且只执行一次,一般用于对类进行初始化。 包(package)
- 用文件系统的文件夹解释避免文件名重复的问题。
- 包语法:
通过package定义在页面最顶部; 在磁盘上的保存路径和package一致,可以单级,可以多级 3、包名“潜规则” 包名小写 ”公司域名反着写.产品名.模块名”:com.rupeng.crm.user、com.sum.media.sound 4、当前包中的类无需引用;其他包中的类: import com.rupeng.crm.user.*(不推荐,一次引用太多,容易冲突) import com.rupeng.crm.user.Person;(推荐) 使用的时候java.sql.Array,不用import,适用于同时使用多个重名的类; 5、java.lang下的内容不用手动import 继承(inherit)
- java中一个类可以“继承自其他类,如果A继承自B”,则A叫做B的子类,B叫做A的父类(基类)。子类会从父类继承所有非private成员。子类还可以有子类。
- java中一个类只能有一个父类,如果没指定父类,则java内置的Object为父类
class FuLei { private void method1() { } private void method2() { } } class ZiLei extends FuLei { private void method3() { } private void method4() { } } ZiLei zi1=new ZiLei(); zi1.nethod1(); zi1.nethod2(); zi1.nethod4(); zi1.toString(); 继承中的构造函数调用顺序
- 子类的构造方法默认都去访问了父类的无参构造方法:在子类中的构造方法中都有一行默认语句:super()
class Fu { public Fu() { System.out.println("fu"); } } class Zi extends Fu { public Zi() { super();//不管是否显示调用,控制台都会输出fu System.out.println("zi") } } Zi z=new Zi(); 先执行父类的构造函数把父类初始化完成,再初始化子类的 private/public/protected
- private成员无法被子类、外界访问,这样保证了private成员的安全性
- protected成员只能被自己以及子类(直接或者间接)以及兄弟(同package包中)访问,无法被“外姓人”访问
枚举
- 有些数据是开放性范围的,比如int、float、String。有些数据可选值是有限取值范围的,比如星座、月份、方向,如果用1/2/3/4代表东南西北,那万一设置了8怎么办?
- 枚举是一种定义确定取值范围的特殊类型(可选值一般大写)。(JDK5之后的语法)
enum Dir { EAST,WEST,NORTH,SOUTH; } 使用Dir d=Dir.EAST; 3、(*)枚举类型中还可以定义方法:在所有枚举定义结束后写“;”比如:getIntValue(得到代表的整数)、parse(把整数解析为枚举对象)。常见用法:给枚举定义一个确定的值。构造函数要定义为private(why)
多态
- 面向对象三大特征:封装,继承,多态。多态是面向对象最强大的一个特征,也是最难的一个特征,设计模式等都是多态的体现。大型项目架构也大量应用多态。
- 子类中定义和父类中一样的方法就叫“重写(Override)或覆盖”
class DiQiuRen { pubic void speak(){System.out.println("我是地球人");} } class Chinese extends DiQiuRen { @Override public void speak(){System.out.println("我是中国人");} public void baiNian(){System.out.println(过年好);} } DiQiuRen dqr1=new DIQiuRen(); dqr1.speak(); Chinese zgr1=new Chinese(); zgr1.speak(); 下面的执行结果是什么? Chinese zgr2=new DiQiuRen();//地球人不一定是中国人 DiQiuRen dqr2=new Chinese(); dqr2.speak(); DiQiuRen dqr2=zgr1; dqr2.speak();
- 父类变量可以指向子类对象,内存中也还是子类对象,有且只有这一个对象。变量是什么类型没有关系,到底执行谁的方法主要取决于内存中的对象是什么类型。
- 变量类型是“把对象看成什么”,DiQiuRen dqr2=new Chinese();是把Chinese对象看成是“地球人”,因为“地球人”不一定有baiNian方法。因此dqr2.baiNian()编译失败。
final
- 如果父类中的某些方法不希望被子类Override,比如kill,那么标记被final即可。好处:安全,可以保证父类定义的行为的执行(必须“被杀死”);效率高,避免了查询虚方法表(*)。
- 如果某些类不希望被继承,类也可以声明为final。
- final类的好处主要是安全(不会有冒名顶替的“不肖子孙”)。String类就是final。
抽象类
- 复习:抽象类被标注abstract,抽象类无法new。
- 地球人的sayHello输出“我是地球人”显然不合理,因为无法确定怎么说,也就是地球人不知道如何sayHello,只有具体到中国人、日本人、美国人才知道如何sayHello
- 把地球人的sayHello的方法体去掉,并且方法增加abstract修饰,类也修饰为abstract:
abstract class DiQiuRen { public abstract void speak(); } 抽象方法没有方法体;一旦类中定义了抽象方法,类必须被修饰为抽象;抽象类无法实例化(new) 抽象类子类
- 抽象类的抽象方法主要用来限制子类"必须实现这些抽象方法";子类也可以不实现,那么子类也要是抽象类,由子类的子类.....去实现
- Eclipse中可以通过Alt+/或者主菜单“源码”——>“覆盖/实现方法”实现覆盖/实现父类方法。
接口
- 接口是一种用来声明“能力”的类型,不提供具体实现不提供实现方法,连{ }都不能有。
- 接口一般以“able”(英文表示“能力”的词一般以able结尾)结尾。
- 接口无法实例化,只能被类“实现”(implements)
public interface Speakable { public void speak(); } public class TeacherCang implements Speakable {
} 4、既可以使用接口类型变量又可以使用类类型变量调用speak接口的意义是定义“做什么”,类定义“怎么做” 接口深入
- 复习:接口用interface声明,接口中定义的方法不能提供实现,{ }空方法都不行。接口无法new,只能被类实现(implements)。
- 接口一般以“able”(英文表示“能力”的词一般以able结尾)结尾。Ctrl+Shift+T看一下系统额接口名验证一下。
- 接口中只能声明static final成员变量,不能声明普通成员变量。因为static final可以被外界用,一个没有实现代码的接口中声明普通成员变量没意义。
- 接口中可以定义多个方法。也可以不定义任何方法(*标识接口)。
- 接口只是“能力”不是“实现”,因此不能也不必要定义构造函数。
- 类只能有一个父亲,类可以实现多个接口。
- 接口可以“实现”其他接口,也是使用implements,不是extends,因此不能称作“继承接口”。
- 接口的多态、类型转换。
|