一、类与对象的初步认知.
C 语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题。
Java 是面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完成。
面向过程注重的是过程,在整个过程中所涉及的行为,就是功能。
面向对象注重的是对象,也就是参与过程所涉及到的主体,是通过逻辑将一个个功能实现连接起来。
面向对象概念
- 面向对象是思考问题的一种思考方式,是一种思想。
- 类就是一类对象的统称。对象就是这一类具体化的一个实例。
- 面向对象的好处,将复杂的事情变简单了,只要面对一个对象即可。
面向对象设计
面向对象设计把握一个重要的经验:谁拥有数据,谁对外提供操作这些数据的方法!
开发时:找对象,建对象,用对象,并且维护对象之间的关系。
后期学习过程中,我们会就这三点进行深入学习。
简而言之
面向对象就是用代码(类)来描述客观世界事务的一种方式.
一个类主要包含一个事务的属性和行为.
二、类和类的实例化
类就是一类对象的统称。对象就是这一类具体化的一个实例。
举个例子:我们建房子的图纸就是一个类,而通过这个图纸建设出的新房子就是一个实体(对象、实例)。一个模子可以实例化无数个对象。 一张图纸,可以建设无数个房子。
总的来说:类就是一个模板,对象是由模板产生的样本。一个类可以产生无数个对象。
声明一个类就是创建一个新的数据类型,而类在 Java 中属于引用类型,Java 使用关键字 class 来声明类。我们来简单声明一个类↓
基本语法
// 创建类
class <class_name>{
field;//成员属性
method;//成员方法
}
// 实例化对象
<class_name> <对象名> = new <class_name>();
class为定义类的关键字,ClassName为类的名字,{}中为类的主体。
类中的元素称为:成员属性。类中的函数称为:成员方法。
示例
class Person{
//成员属性
public int age;
public String name;
public String sex;
//成员方法
public void eat(){
System.out.println("吃饭");
}
public void sleep(){
System.out.println("睡觉");
}
}
注意事项
类和方法的定义还是不一样的,内部的成员方法不带 static 关键字.
类的实例化
使用类类型来创建对象的过程,称为类的实例化
- 类只是一个模型一样的东西,限定了类有哪些成员.
- 一个类可以实例化出很多对象,实例化出的对象,占用实际的物理空间(存储类成员变量).
示例
class Person{
public int age;
public String name;
public String sex;
public void eat(){
System.out.println("吃饭");
}
public void sleep(){
System.out.println("睡觉");
}
}
public class Test01 {
public static void main(String[] args) {
//通过new实例化对象
Person person = new Person();
//成员方法的调用,需要通过对象的引用调用
person.eat();
person.sleep();
//一份图纸可以实例化多个对象
Person person1 = new Person();
Person person2 = new Person();
}
}
//输出结果
吃饭
睡觉
注意事项
- new 关键字用于创建一个对象的实例.
- 使用 “.” 来访问对象中的属性和方法.
- 同一个类可以创建多个实例.
三、类的成员
类的成员可以包含以下:字段、方法、代码块、内部类和接口等.
我们主要介绍前三个.
字段/属性/成员变量
在类中,方法外部定义的变量.
这样的变量,我们称为 “字段” 或 “属性” 或 “成员变量”.
用于描述一个类中包含哪些数据.
class Person{
public int age;
public String name;
public String sex;
}
public class Test01 {
public static void main(String[] args) {
//通过new实例化对象
Person person = new Person();
System.out.println(person.name);
System.out.println(person.age);
}
}
//执行结果
null
0
注意事项
- 使用 “ . ” 访问对象的字段.
- “访问”,既包含读,也包含写.
- 一个对象的字段,如果没有显式设置初始值,那么会设置一个默认值.
- 默认值规则,数字类型默认值为 0,boolean 类型默认值为 false,引用类型(String,Array,自定义类)默认值为 null.
认识null
null 在 Java 中为“空引用”,表示不一样任何对象,类似于 C 语言的空指针.
如果对 null 进行 “ . ” 操作,会引发空指针异常.
class Person {
public String name;
public int age;
}
class Test {
public static void main(String[] args) {
Person person = new Person();
System.out.println(person.name.length()); // 获取字符串长度
}
}
// 执行结果
Exception in thread "main" java.lang.NullPointerException
字段就地初始化
很多时候,我们可以显式地设定初始值.
class Person {
public String name = "张三";
public int age = 18;
}
class Test {
public static void main(String[] args) {
Person person = new Person();
System.out.println(person.name);
System.out.println(person.age);
}
}
// 执行结果
张三
18
方法(method)
用于描述一个对象的行为( Person 类的 eat 方法,sleep 方法).
class Person {
public int age = 18;
public String name = "张三";
public void show() {
System.out.println("我叫" + name + ", 今年" + age + "岁");
}
}
class Test {
public static void main(String[] args) {
Person person = new Person();
person.show();
}
}
// 执行结果
我叫张三, 今年18岁
此时的 show 方法,表示 person 这个对象具有一个“自我展示”的行为.
这样的 show 方法是和 person 实例相关联的.
如果创建了其它实例,那么 show 的行为也会发生改变.
Person person2 = new Person();
person2.name = "李四";
person2.age = 20;
person2.show()
// 执行结果
我叫李四, 今年20岁
static 关键字
可以用于修饰属性、修饰方法、代码块、修饰类.
修饰属性
Java 静态属性和类相关,和具体的实例无关.
换句话说,同一个类的不同实例共用一个静态属性.
class TestDemo{
public int a;
public static int count;
}
public class Main{
public static void main(String[] args) {
TestDemo t1 = new TestDemo();
t1.a++;
TestDemo.count++;
System.out.println(t1.a);
System.out.println(TestDemo.count);
System.out.println("============");
TestDemo t2 = new TestDemo();
t2.a++;
TestDemo.count++;
System.out.println(t2.a);
System.out.println(TestDemo.count);
}
}
//输出结果:
1
1
============
1
2
count 被 static 所修饰,所有类共享。
不属于对象,访问方式为:类名.属性
修饰方法
如果在任何方法上应用 static 关键字,此方法称为静态方法.
- 静态方法属于类,而不属于类的对象.
- 可以之间调用静态方法,无需创建类的实例.
- 静态方法可以访问静态数据成员,并且可以改变静态数据成员的值.
class TestDemo{
public int a;
public static int count;
public static void change() {
count = 100;
//a = 10; error 不可以访问非静态数据成员
}
}
public class Main{
public static void main(String[] args) {
TestDemo.change();//无需创建实例对象 就可以调用
System.out.println(TestDemo.count);
}
}
//输出结果
100
注意事项 :静态方法和实例无关,而是和类相关. 因此导致了两个情况:
- 静态方法不能之间使用非静态数据成员或调用非静态方法(非静态数据成员和方法都是和实例相关的).
- this 和 super 两个关键字不能在静态上下文使用(this 是当前实例的引用,super 是当前实例父类实例的引用,也是和当前实例有关).
小结
实在记不住 static 关键字的具体效果,我们就直接记着:
- 带有 static 关键字的数据,都是在 Java 程序运行一开始就执行的代码.
- static 的变量和方法在内存中一块叫方法区的内存块,每个 static 都只存在一份.
四、封装
什么是封装?
<<代码大全>> 开篇就在讨论一个问题: 软件开发的本质就是对程序复杂程度的管理. 如果一个软件代码复杂程度太高, 那么就无法继续维护. 如何管理复杂程度? 封装就是最基本的方法.
在我们写代码的时候经常会涉及两种角色: 类的实现者和类的调用者.
封装的本质就是让类的调用者不必太多的了解类的实现者是如何实现类的, 只要知道如何使用类就行了.
private 实现封装
private / public 这两个关键字表示 “访问权限控制”.
- public 修饰的成员变量或者成员方法,可以直接被类的调用者使用.
- private 修饰的成员变量或者成员方法,不能被类的调用者使用.
直接使用 public
class Person {
public String name = "张三";
public int age = 18;
}
class Test {
public static void main(String[] args) {
Person person = new Person();
System.out.println("我叫" + person.name + ", 今年" + person.age + "岁");
}
}
// 执行结果
我叫张三, 今年18岁
这样有什么缺点?
这样的代码导致类的使用者(main方法的代码)必须要了解 Person 类内部的实现, 才能够使用这个类. 学习成本较高
一旦类的实现者修改了代码(例如把 name 改成 myName), 那么类的使用者就需要大规模的修改自己的代码, 维护成本较高.
使用private封装属性
并且提供 public 方法供类的调用者使用.
class Person {
private String name = "张三";
private int age = 18;
public void show() {
System.out.println("我叫" + name + ", 今年" + age + "岁");
}
}
class Test {
public static void main(String[] args) {
Person person = new Person();
person.show();
}
}
// 执行结果
我叫张三, 今年18岁
-
此时字段已经使用 private 来修饰. 类的调用者(main方法中)不能直接使用. 而需要借助 show 方法. 此时类的使用者就不必了解 Person 类的实现细节.
-
同时如果类的实现者修改了字段的名字, 类的调用者不需要做出任何修改(类的调用者根本访问不到 name, age这样的字段).
注意事项
- private 不光能修饰字段,也能修饰方法.
- 通常情况下我们会把字段设为 private 属性, 但是方法是否需要设为 public, 就需要视具体情形而定. 一般我们希望一个类只提供 “必要的” public 方法, 而不应该是把所有的方法都无脑设为 public.
getter 和 setter 方法
当我们使用 private 来修饰字段的时候,就无法直接使用这个字段了.
代码示例
class Person {
private String name = "张三";
private int age = 18;
public void show() {
System.out.println("我叫" + name + ", 今年" + age + "岁");
}
}
class Test {
public static void main(String[] args) {
Person person = new Person();
person.age = 20;
person.show();
}
}
// 编译出错
Test.java:13: 错误: age可以在Person中访问private
如果此时需要获取或者修改这个 private 属性,就需要使用 getter / setter 方法.
代码示例
class Person {
private String name;//实例成员变量
private int age;
public void setName(String name){
//name = name;//不能这样写
this.name = name;//this引用,表示调用该方法的对象
}
public String getName(){
return name;
}
public void show(){
System.out.println("name: "+name+" age: "+age);
}
}
public static void main(String[] args) {
Person person = new Person();
person.setName("caocao");
String name = person.getName();
System.out.println(name);
person.show();
}
// 运行结果
caocao
name: caocao age: 0
注意事项
- getName 即为 getter 方法,表示获取这个成员的值.
- setName 即为 setter 方法,表示设置这个成员的值.
- 当 set 方法的形参名字和类中的成员属性的名字一样的时候,如果不使用 this,相当于自赋值. this 表示当前实例的引用.
- 不是所有字段都需要提供 setter / getter 方法,而是要根据实际情况决定提供哪种方法.
五、构造方法
基本语法
构造方法是一种特殊方法,使用关键字 new 实例化新对象时会被自动调用,用于完成初始化操作.
new 执行过程
- 为对象分配内存空间
- 调用对象的构造方法
语法规则
- 方法名称必须与类名称相同
- 构造方法没有返回值类型声明
- 每一个类中至少存在一个构造方法
注意事项
- 如果类中没有提供任何构造方法,那么编译器会默认生成一个不带有参数的构造方法.
- 类中定义了构造方法,则默认的无参的构造方法将会无效.
- 构造方法支持重载,规则和普通方法的重载一致.
代码示例
class Person {
private String name;//实例成员变量
private int age;
private String sex;
//默认构造函数 构造对象
public Person() {
this.name = "caocao";
this.age = 10;
this.sex = "男";
}
//带有3个参数的构造函数
public Person(String name,int age,String sex) {
this.name = name;
this.age = age;
this.sex = sex;
}
public void show(){
System.out.println("name: "+name+" age: "+age+" sex: "+sex);
}
}
public class Main{
public static void main(String[] args) {
Person p1 = new Person();//调用不带参数的构造函数 如果程序没有提供会调用不带参数的构造函数
p1.show();
Person p2 = new Person("zhangfei",80,"男");//调用带有3个参数的构造函数
p2.show();
}
}
// 执行结果
name: caocao age: 10 sex: 男
name: zhangfei age: 80 sex: 男
this 关键字
this 表示当前对象的引用(注意不是当前对象).
可以借助 this 来访问对象的字段和方法.
class Person {
private String name;//实例成员变量
private int age;
private String sex;
//默认构造函数 构造对象
public Person() {
//this调用构造函数
this("bit", 12, "man");//必须放在第一行进行显示
}
//这两个构造函数之间的关系为重载。
public Person(String name,int age,String sex) {
this.name = name;
this.age = age;
this.sex = sex;
}
public void show() {
System.out.println("name: "+name+" age: "+age+" sex: "+sex);
}
}
public class Main{
public static void main(String[] args) {
Person person = new Person();//调用不带参数的构造函数
person.show();
}
}
// 执行结果
name: bit age: 12 sex: man
我们会发现在构造函数的内部,我们可以使用 this 关键字,构造函数时用来构造对象的,对象还没有构造好,我们就使用 this,那 this 还代表当前对象吗?
当然不是,this 代表的时当前对象的引用。
六、认识代码块
字段初始化的方式有:
- 就地初始化.
- 使用构造方法初始化.
- 使用代码块初始化.
前两种我们都介绍过了,接下来我们学习下代码块初始化.
什么是代码块
使用 { } 定义的一段代码.
根据代码块定义的位置以及关键字,又可以分为以下四种:
1. 普通代码块
2. 构造块
3. 静态快
4. 同步代码块
普通代码块
定义在方法中的代码块.
public class Main{
public static void main(String[] args) {
{ //直接使用{}定义,普通方法块
int x = 10 ;
System.out.println("x1 = " +x);
}
int x = 100 ;
System.out.println("x2 = " +x);
}
}
// 执行结果
x1 = 10
x2 = 100
这种用法比较少见.
构造代码块
定义在类中的代码块,也叫:实例代码块。
构造代码块一般用于初始化实例成员变量。
class Person{
private String name;//实例成员变量
private int age;
private String sex;
public Person() {
System.out.println("I am Person init()!");
}
//实例代码块
{
this.name = "bit";
this.age = 12;
this.sex = "man";
System.out.println("I am instance init()!");
}
public void show(){
System.out.println("name: "+name+" age: "+age+" sex: "+sex);
}
}
public class Main {
public static void main(String[] args) {
Person p1 = new Person();
p1.show();
}
}
// 运行结果
I am instance init()!
I am Person init()!
name: bit age: 12 sex: man
注意事项: 实例代码块优先于构造函数执行.
静态代码块
使用 static 定义的代码块。一般用于初始化静态成员属性
class Person{
private String name;
private int age;
private String sex;
private static int count = 0;
public Person(){
System.out.println("I am Person init()!");
}
{
this.name = "aaa";
this.age = 12;
this.sex = "man";
System.out.println("I am instance init()!");
}
static{
count = 10;
System.out.println("I am static init()!");
}
public void show(){
System.out.println("name: " + name);
System.out.println("age: " + age);
}
}
public class Test01{
public static void main(String[] args) {
Person person = new Person();
Person person1 = new Person();
}
}
//执行结果
I am static init()!
I am instance init()!
I am Person init()!
I am instance init()!
I am Person init()!
注意事项
- 静态代码块不管生成多少个对象,只会执行依次,且是最先执行的.
- 执行顺序是:静态代码块、构造代码块、构造方法.
七、补充说明
toString 方法
我们刚刚把对象的属性进行打印的时候,都自己实现了 show 方法 ↓
class Person {
private String name;
private int age;
public Person(String name,int age) {
this.age = age;
this.name = name;
}
public void show() {
System.out.println("name:"+name+" " + "age:"+age);
}
}
public class Main {
public static void main(String[] args) {
Person person = new Person("caocao",19);
person.show();
//我们发现这里打印的是一个地址的哈希值 原因:调用的是Object的toString方法
System.out.println(person);
}
}
// 执行结果
name:caocao age:19
Person@1c168e5
其实我们不必每次都自定义 show 方法,可以使用 toString 这样的方法来将对象自动转成字符串.
代码示例
class Person {
private String name;
private int age;
public Person(String name,int age) {
this.age = age;
this.name = name;
}
public void show() {
System.out.println("name:"+name+" " + "age:"+age);
}
//重写Object的toString方法
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
public class Main {
public static void main(String[] args) {
Person person = new Person("caocao",19);
person.show();
System.out.println(person);
}
}
// 执行结果
name:caocao age:19
Person{name='caocao', age=19}
注意事项
- toString 方法会在 println 的时候被自动调用.
- 将对象转成字符串这样的操作,我们称为 序列化.
- toString 是 Object 类提供的方法,我们自己创建的 Person 类默认继承自 Object 类,可以重写 toStirng 方法实现我们自己版本的转换字符串方法.
匿名对象
匿名只是表示没有名字的对象.
- 没有引用的对象称为匿名对象.
- 匿名对象只能在创建对象时候使用.
- 如果一个对象只使用依次,可以考虑使用匿名对象.
代码示例
class Person {
private String name;
private int age;
public Person(String name,int age) {
this.age = age;
this.name = name;
}
public void show() {
System.out.println("name:"+name+" " + "age:"+age);
}
}
public class Main {
public static void main(String[] args) {
new Person("caocao",19).show();//通过匿名对象调用方法
}
}
// 执行结果
name:caocao age:19
八、内容重点总结
- 一个类可以产生无数的对象,类就是模板,对象就是具体的实例.
- 类中定义的属性,大概分为几类:类属性,对象属性。其中被 static 所修饰的数据属性称为类属性,static 修饰的方法称为类方法,特点是不依赖于对象,我们只要通过类名就可以调用其属性或者方法.
- 静态代码块优先实例代码块执行,实例代码块优先构造函数执行.
- this 关键字代表的是当前对象的引用,并不是当前对象.