Java入门(五)之 类和对象
类与对象的初步认知
C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题。而JAVA是基于面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完成。
面向过程注重的是过程,在整个过程中所涉及的行为,就是功能。面向对象注重的是对象,也就是参与过程所涉及到的主体。是通过逻辑将一个个功能实现连接起来。
面向对象小结:
- 面向对象是思考问题的一种思考方式,是一种思想。比如:概念与实例。理论与实践。名和实等等。
- 类就是一类对象的统称。对象就是这一类具体化的一个实例。
- 面向对象的好处:将复杂的事情变简单了,只要面对一个对象就行。
类和类的实例化
类是一类对象的统称。
对象是这一类具体化的一个实例。
基本语法
// 创建类
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("睡觉!");
}
public static void func() {
//静态成员方法
}
}
和之前写的方法不同,此处写的方法不带 static 关键字. 类的成员中会详细解释 static 是什么作用。
关键字 public 访问修饰限定符。规定了当前它所修饰变量的权限。
访问修饰限定符共四个:public,private,protected,default。
public void eat() 和public void sleep() 为实例成员方法。
public static void func() 为静态成员方法。
注:
- new 关键字用于创建一个对象的实例。
- 使用 . 来访问对象中的属性和方法。
- 同一个类可以创建对个实例。
类的成员
类的成员可以包含以下:字段、方法、代码块、内部类和接口等。
此处我们重点介绍前三个。
字段/属性/成员变量
在类中,但是方法外部定义的变量,这样的变量我们称为 “字段” 或 “属性” 或 “成员变量”(三种称呼都可以, 一般不会严格区分)。
如果没有初始化,那么它的值就是一个默认值(默认的零值)。
- 如果是引用类型,那么默认值就是null
- 如果是简单类型,那么默认值就是对应类型的默认值。
byte——》0;short——》0; int——》0;long——》0L
float——》0.0f;double——》0.0
char——》 ‘\u0000’(十六进制)
boolean——》false
验证如下
class Person {
public String name; // 字段
public int age;
public float grade;
public char ch;
public boolean flg;
}
class Test {
public static void main(String[] args) {
Person person = new Person();
System.out.println(person.name);
System.out.println(person.age);
System.out.println(person.grade);
System.out.println(person.ch);
System.out.println(person.flg);
}
}
执行结果
再次认识 null
null 在 Java 中为 “空引用”,表示不引用任何对象。类似于 C 语言中的空指针。如果对 null 进行 . 操作就会引发异常。
方法 (method)
就是我之前总结过的方法。用于描述一个对象的行为。
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();
}
}
该代码执行如下
此处的 show 方法, 表示 Person 这个对象具有一个 “展示自我” 的行为。
这样的 show 方法是和 person 实例相关联的。如果创建了其他实例,那么 show 的行为就会发生变化。
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();
Person person2 = new Person();
person2.name = "李四";
person2.age = 20;
person2.show();
}
}
此代码的执行结果就变为了下图
方法中还有一种特殊的方法称为 构造方法 (construction method)。
在实例化对象的时候会被自动调用到的方法,方法名字和类名相同,用于对象的初始化。
虽然我们前面已经能将属性就地初始化,但是有些时候可能需要进行一些更复杂的初始化逻辑,那么就可以使用构造方法。
同样,构造方法本期也会介绍到。
static 关键字
1.修饰属性
修饰属性,Java静态属性和类相关,和具体的实例无关。换句话说,同一个类的不同实例共用同一个静态属性。
class TestDemo{
public int a;
public static int count;
}
class Test{
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);
}
}
执行结果
count被static所修饰,所有类共享。定义在方法区上,且不属于对象,访问方式为:类名 . 属性。
实例代码内存解析
2.修饰方法
如果在任何方法上应用 static 关键字,此方法称为静态方法。
- 静态方法属于类,而不属于类的对象。
- 可以直接调用静态方法,而无需创建类的实例。
- 静态方法可以访问静态数据成员,并可以更改静态数据成员的值。
class TestDemo{
public int a;
public static int count;
public static void change() {
count = 100;
//a = 10; error 不可以访问非静态数据成员
}
}
class Test{
public static void main(String[] args) {
TestDemo.change();//无需创建实例对象 就可以调用
System.out.println(TestDemo.count);
}
}
执行结果
注意事项1: 静态方法和实例无关, 而是和类相关. 因此这导致了两个情况:
- 静态方法不能直接使用非静态数据成员或调用非静态方法(非静态数据成员和方法都是和实例相关的)。
- this和super两个关键字不能在静态上下文中使用(this 是当前实例的引用, super是当前实例父类实例的引用,也是和当前实例相关的)。
注意事项2
- 之前所写的方法为了简单,都统一加上了 static。但实际上一个方法具体要不要带 static,都需要是情形而定。
- main 方法为 static 方法。
代码块
下面总结代码块时详细介绍
修饰类
后面在内部类会详细介绍
小结
以下代码, 分析内存布局
class Person {
public int age;//实例变量存放在对象内
public String name;//实例变量
public String sex;//实例变量
public static int count;//类变量也叫静态变量,编译时已经产生,属于类本身,且只有一份。存放在方法区
public final int SIZE = 10;//被final修饰的叫常量,也属于对象。 被final修饰,后续不可更改
public static final int
COUNT = 99;//静态的常量,属于类本身,只有一份被final修饰,后续不可更改
//实例成员函数
public void eat() {
int a = 10;//局部变量
System.out.println("eat()!");
}
//实例成员函数
public void sleep() {
System.out.println("sleep()!");
}
//静态成员函数
public static void staticTest(){
//不能访问非静态成员
// sex = "man"; error
System.out.println("StaticTest()");
}
}
class Test {
public static void main(String[] args) {
//产生对象 实例化对象
Person person = new Person();//person为对象的引用
System.out.println(person.age);//默认值为0
System.out.println(person.name);//默认值为null
//System.out.println(person.count);//会有警告!
//正确访问方式:
System.out.println(Person.count);
System.out.println(Person.COUNT);
Person.staticTest();
//总结:所有被static所修饰的方法或者属性,全部不依赖于对象。
person.eat();
person.sleep();
}
}
输出结果为
内存布局为
封装
写代码的时经常会涉及两种角色: 类的实现者和类的调用者。
封装的本质就是让类的调用者不必太多的了解类的实现者是如何实现类的,只要知道如何使用类就行了。这样就降低了类使用者的学习和使用成本,从而降低了复杂程度。
private实现封装
private/ public 这两个关键字表示 “访问权限控制” 。
- 被 public 修饰的成员变量或者成员方法,可以直接被类的调用者使用。
- 被 private 修饰的成员变量或者成员方法,不能被类的调用者使用。
使用 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();
}
}
该代码执行结果如下
- 此时字段已经使用 private 来修饰。类的调用者(main方法中)不能直接使用。而需要借助 show 方法. 此时类的使用者就不必了解 Person 类的实现细节。
- 同时如果类的实现者修改了字段的名字,类的调用者不需要做出任何修改(类的调用者根本访问不到 name, age这样的字段)。
注意事项
- private 不光能修饰字段,也能修饰方法。
- 通常情况下我们会把字段设为 private 属性,但是方法是否需要设为 public,就需要视具体情形而定。一般我们希望一个类只提供 “必要的” public 方法,而不应该是把所有的方法都无脑设为 public。
getter和setter方法
当我们使用 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);
}
}
class Test{
public static void main(String[] args) {
Person person = new Person();
person.setName("晓晓");
String name = person.getName();
System.out.println(name);
person.show();
}
}
getName 即为 getter 方法,表示获取这个成员的值。
setName 即为 setter 方法,表示设置这个成员的值。
当set方法的形参名字和类中的成员属性的名字一样的时候,如果不使用this,相当于自赋值。this 表示当前实例的引用。
不是所有的字段都一定要提供 setter / getter 方法,而是要根据实际情况决定提供哪种方法。
在 IDEA 中可以使用 alt + insert (或者 alt + F12) 快速生成 setter / getter 方法;在 VSCode 中可以使用鼠标右键菜单 -> 源代码操作 中自动生成 setter / getter 方法。
构造方法
构造方法是一种特殊方法,使用关键字new实例化新对象时会被自动调用,用于完成初始化操作。
基本语法
new 执行过程:
- 为对象分配内存空间。
- 调用对象的构造方法。
语法规则:
- 方法名称必须与类名称相同。
- 构造方法没有返回值类型声明。
- 每一个类中一定至少存在一个构造方法(没有明确定义,则系统自动生成一个无参构造)。
注意事项:
- 如果类中没有提供任何的构造函数,那么编译器会默认生成一个不带有参数的构造函数。
- 若类中定义了构造方法,则默认的无参构造将不再生成。
- 构造方法支持重载。规则和普通方法的重载一致。
class Person {
private String name;//实例成员变量
private int age;
private String sex;
//默认构造函数 构造对象
public Person() {
this.name = "小明";
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);
}
}
class Test{
public static void main(String[] args) {
Person p1 = new Person();//调用不带参数的构造函数 如果程序没有提供会调用不带参数的构造函数
p1.show();
Person p2 = new Person("小红",8,"女");//调用带有3个参数的构造函数
p2.show();
}
}
上述代码执行结果如下
this关键字
this表示当前对象引用(注意不是当前对象)。可以借助 this 来访问对象的字段和方法。
class Person {
private String name;//实例成员变量
private int age;
private String sex;
//默认构造函数 构造对象
public Person() {
//this调用构造函数
this("敏敏", 21, "woman");//必须放在第一行进行显示
}
//这两个构造函数之间的关系为重载。
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);
}
}
class Test {
public static void main(String[] args) {
Person person = new Person();//调用不带参数的构造函数
person.show();
}
}
上述代码的执行结果如下
在构造函数的内部,可以使用this关键字,构造函数是用来构造对象的,对象还没有构造好,就使用了this,那this还代表当前对象吗?
当然不是,this代表的是当前对象的引用。
认识代码块
字段的初始化方式有:
- 就地初始化
- 使用构造方法初始化
- 使用代码块初始化
前两种方式前面已经学习过了, 接下来我们介绍第三种方式, 使用代码块初始化.
什么是代码块
使用 { } 定义的一段代码。
根据代码块定义的位置以及关键字,又可分为以下四种:
- 普通代码块
- 构造块
- 静态块
- 同步代码块(后续到多线程部分再谈)
普通代码块
普通代码块:定义在方法中的代码块。
class Test{
public static void main(String[] args) {
{
//直接使用{}定义,普通方法块
int i = 10 ;
System.out.println("i = " +i);
}
int j = 100 ;
System.out.println("j = " +j);
}
}
这种用法较少见
构造代码块
构造块:定义在类中的代码块(不加修饰符)(也叫实例代码块)。构造代码块一般用于初始化实例成员变量。
class Person{
private String name;//实例成员变量
private int age;
private String sex;
public Person() {
System.out.println("I am Person init()!");
}
//实例代码块
{
this.name = "莉莉";
this.age = 21;
this.sex = "woman";
System.out.println("I am instance init()!");
}
public void show(){
System.out.println("name: "+name+" age: "+age+" sex: "+sex);
}
}
class Test {
public static void main(String[] args) {
Person p1 = new Person();
p1.show();
}
}
运行结果
注意: 实例代码块优先于构造函数执行。
静态代码块
使用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 = "bit";
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+" age: "+age+" sex: "+sex);
}
}
class Test {
public static void main(String[] args) {
Person p1 = new Person();
Person p2 = new Person();//静态代码块不会被执行。
}
}
静态代码块不管生成多少个对象,其只会执行一次,且是最先执行的。
静态代码块执行完毕后, 实例代码块(构造块)执行,再然后是构造函数执行。
补充说明
toString方法
- toString 方法会在 println 的时候被自动调用。
- 将对象转成字符串这样的操作我们称为 序列化。
- toString 是 Object 类提供的方法,我们自己创建的 Person 类默认继承自 Object 类,可以重写 toString 方法实现我们自己版本的转换字符串方法。
- @Override 在 Java 中称为 “注解”,此处的 @Override 表示下面实现的 toString 方法是重写了父类的方法。
- IDEA快速生成Object的toString方法快捷键:alt+f12(insert)
匿名对象
只是表示没有名字的对象。
- 没有引用的对象称为匿名对象。
- 匿名对象只能在创建对象时使用。
- 如果一个对象只是用一次,后面不需要用了。可以考虑使用匿名对象。
Java入门(五)之 类和对象,到这里就基本结束了。今天的内容,较以往来说,更抽象一些,同时,体现着Java语言面向对象的特点。
下期带来Java入门(六)之 时间和空间复杂度。各位铁铁,我们下期见。