目录
Java是一门纯面向对象的语言(Object Oriented Program,简称OOP),在面向对象的世界里,一切皆为对象。面向对象是解决问题的一种思想,主要依靠对象之间的交互完成一件事情。用面向对象的思想来设计程序,更符合人们对事物的认知,对于大型程序的设计、扩展以及维护都非常友好。
1. 传统洗衣服过程
传统的方式:
传统的方式:注重的是洗衣服的过程,少了一个环节可能都不行。
而且不同衣服洗的方式,时间长度,拧干方式都不同,处理起来就比较麻烦。如果将来要洗鞋子,那就是另一种方式。
按照该种方式来写代码,将来扩展或者维护起来会比较麻烦。
2. 现代洗衣服过程
以面向对象方式来进行处理,就不关注洗衣服的过程,具体洗衣机是怎么来洗衣服,如何来甩干的,用户不用去关心,只需要将衣服放进洗衣机,导入洗衣粉,启动开关即可,通过对象之间的交互来完成的。
注意:面向过程和面向对象并不是一门语言,而是解决问题的方法,没有那个好坏之分,都有其专门的应用场景。
那么对象从何而来?
比如张三这个人, 就是一个对象. 我们就可以用一些口头语言去描述张三的形象特征, 也就是描述张三这个对象. 于是我们也可以通过代码来描述张三这个人.
- class Person {
- // 描述 人的属性
- public int age; // 年龄
- public String name; // 姓名
-
- // 描述 人的行为
- public void eat() {
- System.out.println("吃饭~ ");
- }
- }
以上代码简单的素描了一下人的属性, 但注意这个代码并不等于有了一个"真实的张三".
那么描述一个对象我们就会通过这样的一个"类"去描述.
类是用来对一个实体(对象)来进行描述的,主要描述该实体(对象)具有哪些属性(外观尺寸等),哪些功能(用来干啥),描述完成后计算机就可以识别了。
比如:洗衣机,它是一个品牌,在Java中可以将其看成是一个类别。
属性:产品品牌,型号,产品重量,外观尺寸,颜色...
功能:洗衣,烘干、定时....
一个类由两部分组成: 属性 + 行为 (也叫 成员属性 + 成员方法)
在java中定义类时需要用到class关键字: class 类名{}
(注: 类名使用大驼峰)
- // 创建洗衣机类
- class WashMachine {
- // 属性[字段] -> 成员属性
- public String brand; // 品牌
- public String type; // 型号
- public double weight; // 重量
- public double length; // 长
- public double width; // 宽
- public double height; // 高
- public String color; // 颜色
- // 行为[方法] -> 成员方法
-
- public void washClothes(){ // 洗衣服
- System.out.println("洗衣功能");
- }
-
- public void dryClothes(){ // 脱水
- System.out.println("脱水功能");
- }
-
- public void setTime(){ // 定时
- System.out.println("定时功能");
- }
- }
注意事项:
1. 类名使用大驼峰命名; 方法名/变量使用小驼峰命名.
2. 成员前写法 这里 统一为public.
3. 这里 写的方法不带static关键字.
定义 狗 这个类.
- class Dog {
- // 成员变量
- public String name;
- public String color;
-
- // 行为
- public void barks() {
- System.out.println(name + "汪汪叫!");
- }
- public void wag() {
- System.out.println(name + "摇尾巴!");
- }
- }
注意事项:
1. 一个Java文件一般只定义一个类
2. main方法所在的类一般要使用public修饰
3. public修饰的类必须要和文件名相同
4. 不要轻易去修改public修饰的类的名称,如果要修改,通过开发工具修改
有了类之后, 应该怎么用? 应该怎么样拥有一个对象?
我们现在已经有了一个类, 但是它不是一个具体的实体, 比如上文中的Dog类, 它并不是一个具体的狗, 只是一个类似于模板/图纸一样的东西, 那么要想让这个Dog成为"真正的狗", 那怎么办?我们来看.
Dog dog = new Dog();
这句代码我们称作: 实例化一个Dog对象.
那么只要new一个Dog, 就真正意义上有一只狗了. 这就是由类变成一个真正的实体.
定义了一个类,就相当于在计算机中定义了一种新的类型.
就像int a = 10;
,类型 变量 = 值;
, 我们定义的类Dog也叫做类型, 是我们自己定义的一种类型,类型 变量 = 实例化一个对象;
与int,double类似,只不过int和double是java语言自带的内置类型,而类是用户自定义了一个新的类型. 类(一种新定义的类型)有了这些自定义的类型之后,就可以使用这些类来定义实例(或者称为对象)。
用类类型创建对象的过程,称为类的实例化,在java中采用new关键字,配合类名来实例化对象。
那么new了对象后, 类中的属性怎么访问?
通过.
操作符
- dog.name = "来福";
- dog.color = "黑色";
于是:
- System.out.println(dog.name);
- System.out.println(dog.color);
当给对象赋值了之后, 就可以通过 引用.方法
来访问方法.
- dog.barks();
- dog.wag();
1. 类只是一个模型一样的东西,用来对一个实体进行描述,限定了类有哪些成员.
2. 类是一种自定义的类型,可以用来定义变量.
3. 一个类可以实例化出多个对象,实例化出的对象 占用实际的物理空间,存储类成员变量
4. 做个比方。类实例化出对象就像现实中使用建筑设计图建造出房子,类就像是设计图,只设计出需要什么东西,但是并没有实体的建筑存在,同样类也只是一个设计,实例化出的对象才能实际存储数据,占用物理空间
代码示例: 通过一个类可以实例化无数个对象.
- Dog dog2 = new Dog();
- dog2.name = "小灰";
- dog2.color = "灰色";
-
- Dog dog3 = new Dog();
- Dog dog4 = new Dog();
先看一个日期类的例子:
- public class TestDate {
- public int year; // 年
- public int month; // 月
- public int day; // 日
-
- public void setDate(int y, int m, int d) {
- year = y;
- month = m;
- day = d;
- }
-
- public void printDate() {
- System.out.println(year + "-" + month + "-" + day);
- }
-
- public static void main1(String[] args) {
- TestDate testDate = new TestDate();
- System.out.println("fda");
- }
- }
此时打断点,可以看到:
在testDate中yeah, month, day都为0.
我们现在要给它们赋值, 于是就要通过testDate这个引用单独的去访问成员属性.
- public static void main(String[] args) {
- TestDate testDate = new TestDate();
- testDate.year = 2023;
- testDate.month = 10;
- testDate.day = 26;
- System.out.println("fda");
- }
此时再运行Debug:
可以看到, 三个成员属性被赋值了.
也可以通过printDate()
访问三个成员属性:
如果不是在main中一个一个赋值的, 那么就可以调用setDate()
传入参数进行赋值.
我们进行Debug, 进入setDate()
:
可以看到, 现在y, m, d这些形参的值是有的, 然后yeah, month, day这些默认都是0, 往下走会发现会被赋值.
然后再运行, 打断点的这一行代码就执行完毕了.
这里要明白, yeah, month, day都是这个testDate对象里面的.
对于前面的代码, 如果我们把setDate()
的形参变一下:
- public void setDate(int year, int month, int day) {
- year = year;
- month = month;
- day = day;
- }
运行之后会看到, 结果变成了都是0, 三个成员属性没有被赋值.
原因就在于:
局部变量优先原则.
也就是说, 在setDate()
中, 它认为yeah, month, day都是局部变量.
相当于在这里只是形参自己给自己赋值了, 所以它根本没有赋值到成员变量中.
那么怎么样才能赋值到成员变量中?
这里就要使用一个东西叫做this
关键字.
把this
加到setDate()
中就可以了
- public void setDate(int year, int month, int day) {
- this.year = year;
- this.month = month;
- this.day = day;
- }
可以看到, 加上this之后, 运行结果就没有什么问题了.
为什么加上this就可以了, this又代表什么?
再来看下面这个例子:
- public static void main(String[] args) {
- TestDate testDate1 = new TestDate();
- TestDate testDate2 = new TestDate();
- TestDate testDate3 = new TestDate();
-
- testDate1.setDate(2023, 10, 26);
- testDate2.setDate(2024, 10, 26);
- testDate3.setDate(2025, 10, 26);
-
- testDate1.printDate();
- testDate2.printDate();
- testDate3.printDate();
- }
执行结果:
并且问题是在setDate中就算没有加this, 结果也是一样的.
三个对象都在调用setDate和printDate函数,但是这两个函数中没有任何有关对象的说明,setDate和
printDate函数如何知道打印的是那个对象的数据呢?
this引用指向当前对象(成员方法运行时调用该成员方法的对象),在成员方法中所有成员变量的操作,都是通过该引用去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。
- public class Date {
- public int year;
- public int month;
- public int day;
- public void setDay(int year, int month, int day){
- this.year = year;
- this.month = month;
- this.day = day;
- }
- public void printDate(){
- System.out.println(this.year + "/" + this.month + "/" + this.day);
- }
- }
注意:this引用的是调用成员方法的对象。
- public static void main(String[] args) {
- Date d = new Date();
- d.setDay(2020,9,15);
- d.printDate();
- }
this有三种使用方法:
this.成员变量
this.访问成员方法
this();访问构造方法
第三个具体后续讲解.
1. this的类型:对应类类型引用,即哪个对象调用就是哪个对象的引用类型
2. this只能在"成员方法"中使用
3. 在"成员方法"中,this只能引用当前对象,不能再引用其他对象
4. this是“成员方法”第一个隐藏的参数,编译器会自动传递,在成员方法执行时,编译器会负责将调用成员方法对象的引用传递给该成员方法,this负责来接收
在代码层面来简单演示--->注意:下图右侧中的Date类也是可以通过编译的
写一个学生类 有姓名 年龄 等属性, 然后通过一个方法设置这些属性的值。其次通过写2个方法,在其中一个方法当中,使用this调用另一个方法.
- public class Student {
- public String name;
- public int age;
-
- public void setStudent(String name, int age) {
- this.name = name;
- this.age = age;
- }
-
- public void print() {
- System.out.println(this.name + " => " + this.age);
- }
-
- public static void main(String[] args) {
- Student student = new Student();
- student.setStudent("Zhangsan", 18);
- student.print();
- }
- }
通过new可以实例化一个对象, 在对象中有很多的成员, 如果是方法中的一个局部变量, 我们知道没有初始化时一定会报错, 但是如果是对象, 却是能通过编译的.
- public static void main(String[] args) {
- Student student = new Student();
- // 没有给成员变量赋值, 直接访问成员变量, 不会报错, 因为它是一个引用变量
- System.out.println(student.name);
- }
不会报错的原因就在于默认初始化. 也就是没有给成员变量赋值的时候, 它们有自己的默认值, 这就是默认初始化.
写类的时候直接给成员变量赋值.
- public class Student {
- public String name = "Zhangsan";
- public int age = 18;
-
- // ..
- }
然而这种写法不好, 因为类本身就是一个模板, 对于上例来说不是所有的学生都叫做yy, 年龄都是18.
就地初始化的使用取决于代码的需求.
不同数据类型作为成员变量时所对应的默认值如下表:
数据类型 | 默认值 |
---|---|
byte | 0 |
char | '\u0000' |
short | 0 |
int | 0 |
long | 0L |
boolean | false |
float | 0.0f |
double | 0.0 |
reference | null |
那么除了setXXX方法还有其他的方式可以用于对象的初始化
通过构造方法给对象中的成员进行初始化.
构造方法(也称为构造器)是一个特殊的成员方法,没有返回值, 方法名与类名相同,在创建对象时,由编译器自动调用,并且在整个对象的生命周期内只调用一次。
- // 对于 Student类
- Student() {
- System.out.println("不带参数的构造方法");
- }
-
- // 可以通过构造方法传参并对成员变量赋值
- Student(String name,int age) {
- System.out.println("带2个参数的构造方法");
- this.name = name;
- this.age = age;
- }
虽然构造方法写好了, 但是构造方法如何被调用?
我们运行代码:
可以看到, 我们写的Student()方法被调用了. 并且可以知道, 这个方法是在Student student = new Student();
这一行代码被调用的.
构造方法的调用是在new后的Student()
. 我们通过new实例化对象, 在实例化对象的时候一定会调用构造方法.
在前面我们没有写两个Student()
构造方法的时候, 并没有编译报错, Student student = new Student();
也没有提供参数? 为什么说一定会调用构造方法?
当我们没有主动提供构造方法的时候, 编译器会默认帮我们提供一个不带参数的构造方法.
所以理论上来说, 我们在Student student = new Student();
什么都没写, 实际上会有默认调用Student()
的存在. 那么只要我们主动写了, 就会调用我们写的构造方法.
注: 一般我们写构造方法会在前面都加上public, 具体原因后续说.
- // 对于 Student类
- public Student() {
- System.out.println("不带参数的构造方法");
- }
-
- // 可以通过构造方法传参并对成员变量赋值
- public Student(String name,int age) {
- System.out.println("带2个参数的构造方法");
- this.name = name;
- this.age = age;
- }
那么我们也可以调用带两个参数的构造方法, 在main中加上:
Student student2 = new Student("Sun",18);
可以看到, 不带参数的和带2个参数的都已经被打印.
由此可以说明, 在实例化对象的时候才会调用构造方法, 而且在调用构造方法的时候可以传参, 对成员属性进行赋值. 所以要想实例化对象, 就一定要调用构造方法.
于是我们可以理解为, 当构造方法调用完成之后, 对象才实际意义上的产生了.
同时我们可以看到, 构造方法是可以被重载的,
小结: 构造方法的特性:
1. 名字必须与类名相同
2. 没有返回值类型,设置为void也不行
3. 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次(相当于人的出生,每个人只能出生一次)
4. 构造方法可以重载(用户根据自己的需求提供不同参数的构造方法)
5. 如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的。
注意:一旦用户定义,编译器则不再生成。
6. 构造方法中,可以通过this调用其他构造方法来简化代码
注意:
this(...)必须是构造方法中第一条语句
不能形成环
7. 绝大多数情况下使用public来修饰,特殊场景下会被private修饰(后序讲单例模式时会遇到)
- // 关于构造方法中的 this 的使用
- public Student() {
- // 调用 本类当中 其他的构造方法
- this("zhangsan", 23); // 走到这里, 代码会调到Student(name, age)方法执行
- System.out.println("不带参数的构造方法");
- }