Java之类与对象(图文结合)
目录
一、面向对象的初步认知
1、什么是面向对象
2、面向对象与面向过程
二、类定义和使用
1、简单认识类
2、类的定义格式
3、练习
(1)定义一个狗类
(2)定义一个学生类
三、类的实例化
1、什么是实例化
2、类和对象的说明
四、this引用
1、为什么要有this引用
2、什么是this引用
3、this引用的特性
五、对象的构造及初始化
1、如何初始化对象
2、构造方法
(1)概念
(2)特性
3、默认初始化
4、就地初始化
六、封装
1、封装的概念
2、访问限定符
3、封装扩展之包
(1)包的概念
(3)自定义包
(4)包的访问权限控制
(5)常见的包
七、static成员
1、再谈学生类
2、static修饰成员变量
3、static修饰成员方法
4、static成员变量初始化
(1) 就地初始化
(2)静态代码块初始化
八、代码块
1、代码块概念以及分类
2、普通代码块
3、构造代码块
4、静态代码块
十、对象的打印
一、面向对象的初步认知
1、什么是面向对象
2、面向对象与面向过程
我们用洗衣服来举例子,了解一下面向对象和面向过程
传统的方式:注重的是洗衣服的过程,少了一个环节可能都不行。
二、类定义和使用
1、简单认识类
类是用来对一个实体(对象)来进行描述的 ,主要描述该实体 ( 对象 ) 具有哪些属性 ( 外观尺寸等 ) ,哪些功能 ( 用来干啥) ,描述完成后计算机就可以识别了
在Java语言中,如何对上述的洗衣机类来进行定义呢?
我们接下来就来学习一下类的定义格式
2、类的定义格式
// 创建类
class ClassName{
field; // 字段(属性) 或者 成员变量
method; // 行为 或者 成员方法
}// 创建类
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 关键字 .4、 一个Java文件中可以定义多个class类,但只能一个类是public修饰,而且public修饰的类名必须成为代码文件名。 实际开发中建议还是一个文件定义一个class类。
3、练习
(1)定义一个狗类
class PetDog {
public String name;//名字
public String color;//颜色
// 狗的属性
public void barks() {
System.out.println(name + ": 旺旺旺~~~");
}
// 狗的行为
public void wag() {
System.out.println(name + ": 摇尾巴~~~");
}
}
(2)定义一个学生类
public class Student{
public String name;
public String gender;
public short age;
public double score;
public void DoClass(){}
public void DoHomework(){}
public void Exam(){}
}
1. 一般一个文件当中只定义一个类2. main 方法所在的类一般要使用 public 修饰( 注意: Eclipse 默认会在 public 修饰的类中找 main 方法 )3. public修饰的类必须要和文件名相同4. 不要轻易去修改 public 修饰的类的名称,如果要修改,通过开发工具修改 。
三、类的实例化
1、什么是实例化
public class Main{
public static void main(String[] args) {
PetDog dogh = new PetDog(); //通过new实例化对象
dogh.name = "阿黄";
dogh.color = "黑黄";
dogh.barks();
dogh.wag();
PetDog dogs = new PetDog();
dogs.name = "阿黄";
dogs.color = "黑黄";
dogs.barks();
dogs.wag();
}
}
输出结果:
阿黄: 旺旺旺~~~
阿黄: 摇尾巴~~~
赛虎: 旺旺旺~~~
赛虎: 摇尾巴~~~
通过这段代码,我们可以发现,在实例化对象的时候,我们利用new来完成了实例化对象,然后通过.操作符来访问这个对象中的属性和方法,由此,我们可以得到以下实例化对象中的注意事项:
注意事项:
1、new 关键字用于创建一个对象的实例 .2、使用 . 来访问对象中的属性和方法 .3、同一个类可以创建多个实例 .
2、类和对象的说明
此外,我们还要注意一个类可以用new关键词实例化多个对象,并且这些对象指向的内存空间是不同的:
四、this引用
1、为什么要有this引用
同样的,我们通过代码的例子来进行讲解。
首先,我们先创建一个日期类,并用方法实现日期的打印:
public class Date {
public int year;
public int month;
public int day;
public void setDay(int y, int m, int d){
year = y;
month = m;
day = d;
}
public void printDate(){
System.out.println(year + "/" + month + "/" + day);
}
接下来,我们在main方法中创建三个对象,并通过Date这个类来实现对日期的打印操作
public static void main(String[] args) {
// 构造三个日期类型的对象 d1 d2 d3
Date d1 = new Date();
Date d2 = new Date();
Date d3 = new Date();
// 对d1,d2,d3的日期设置
d1.setDay(2020,9,15);
d2.setDay(2020,9,16);
d3.setDay(2020,9,17);
// 打印日期中的内容
d1.printDate();
d2.printDate();
d3.printDate();
}
此时,为了更好地观看代码,我们将它们合起来看看:
public class Date {
public int year;
public int month;
public int day;
public void setDay(int y, int m, int d){
year = y;
month = m;
day = d;
}
public void printDate(){
System.out.println(year + "/" + month + "/" + day);
}
public static void main(String[] args) {
// 构造三个日期类型的对象 d1 d2 d3
Date d1 = new Date();
Date d2 = new Date();
Date d3 = new Date();
// 对d1,d2,d3的日期设置
d1.setDay(2020,9,15);
d2.setDay(2020,9,16);
d3.setDay(2020,9,17);
// 打印日期中的内容
d1.printDate();
d2.printDate();
d3.printDate();
}
}
我们可以发现,这串代码整体逻辑非常简单,没有任何问题,但是细思之下有以下两个疑问:
1. 形参名不小心与成员变量名相同:
public void setDay(int year, int month, int day){
year = year;
month = month;
day = day;
}
当我们把形参设置的和类中的成员变量名一致的时候,那函数体中到底是谁给谁赋值?成员变量给成员变量?参数给参数?参数给成员变量?成员变量参数?估计自己都搞不清楚了。
2、什么是this引用
public void setDay(int year, int month, int day){
year = year;
month = month;
day = day;
}
我们可以将这个先放到IDEA中执行试一下,这时我们会发现,当方法中的形参和类的成员对象名一致的时候,year\month\day的打印结果均为0
这是因为:
这里的year是给局部变量自己赋值,但是并没有给到成员变量赋值
那么这个时候,就轮到我们的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表示对当前对象的引用,也就是说:谁调用了setDay,那么这个this就是谁,那么为了防止以后出现类似这样的问题,我们推荐习惯使用this,这样就可以规避错误了
public static void main(String[] args) {
Date d = new Date();
d.setDay(2020,9,15);
d.printDate();
}
我们可以发现,在形参中本身就存在一个this ,只是我们的编译器帮我们隐藏起来了,因此在平时我们是看不到的。
3、this引用的特性
1. this 的类型:对应类类型引用,即哪个对象调用就是哪个对象的引用类型2. this 只能在 " 成员方法 " 中使用 (静态成员方法中不可使用)3. 在 " 成员方法 " 中, this 只能引用当前对象,不能再引用其他对象4. this 是 “ 成员方法 ” 第一个隐藏的参数,编译器会自动传递,在成员方法执行时,编译器会负责将调用成员方法对象的引用传递给该成员方法,this 负责来接收
那么接下来,在代码层面来简单演示--->注意:下图右侧中的Date类也是可以通过编译的
五、对象的构造及初始化
1、如何初始化对象
public static void main(String[] args) {
int a;
System.out.println(a);
}
这是由于可能未对变量a进行初始化导致的,要让上述代码通过编译,非常简单,只需在正式使用a之前,给a设置一个初始值即可。如果是对象:
public static void main(String[] args) {
Date d = new Date();
d.printDate();
d.setDate(2021,6,9);
d.printDate();
}
我们会发现:需要调用之前写的SetDate方法才可以将具体的日期设置到对象中。
通过上述例子发现两个问题:
2、构造方法
(1)概念
构造方法 ( 也称为构造器 ) 是一个特殊的成员方法, 名字必须与类名相同,在创建对象时,由编译器自动调用,并且在整个对象的生命周期内只调用一次。
那么有了构造方法,我们的Date类应该是这个样子的:
public class Date {
public int year;
public int month;
public int day;
// 构造方法:
// 名字与类名相同,没有返回值类型,设置为void也不行
// 一般情况下使用public修饰
// 在创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次
public Date(int year, int month, int day){
this.year = year;
this.month = month;
this.day = day;
System.out.println("Date(int,int,int)方法被调用了");
}
public void printDate(){
System.out.println(year + "-" + month + "-" + day);
}
public static void main(String[] args) {
// 此处创建了一个Date类型的对象,并没有显式调用构造方法
Date d = new Date(2021,6,9); // 输出Date(int,int,int)方法被调用了
d.printDate(); // 2021-6-9
}
}
(2)特性
1. 名字必须与类名相同2. 没有返回值类型,设置为 void 也不行3. 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次 ( 相当于人的出生,每个人只能出生一次 )4. 构造方法可以重载 ( 用户根据自己的需求提供不同参数的构造方法 )5. 如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的。6. 构造方法中,可以通过this调用其他构造方法来简化代码7. 绝大多数情况下使用public来修饰,特殊场景下会被private修饰
public class Date {
public int year;
public int month;
public int day;
// 无参构造方法
public Date(){
this.year = 1900;
this.month = 1;
this.day = 1;
}
// 带有三个参数的构造方法
public Date(int year, int month, int day) {
this.year = year;
this.month = month;
this.day = day;
}
public void printDate(){
System.out.println(year + "-" + month + "-" + day);
}
public static void main(String[] args) {
Date d = new Date();
d.printDate();
}
}
上述两个构造方法:名字相同,参数列表不同,因此构成了方法重载。
public class Date {
public int year;
public int month;
public int day;
public void printDate(){
System.out.println(year + "-" + month + "-" + day);
}
public static void main(String[] args) {
Date d = new Date();
d.printDate();
}
}
在这段代码中,我们可以发现此时我们并没有创建一个构造方法,但是编译器会给我们默认生成一个不带任何参数的构造方法,只不过被编译器隐藏起来了,所以我们看不到。
但是一旦我们自己定义了一个构造方法的情况下,编译器便不会再帮助我们定义一个构造方法了
public class Date {
public int year;
public int month;
public int day;
public Date(int year, int month, int day) {
this.year = year;
this.month = month;
this.day = day;
}
public void printDate(){
System.out.println(year + "-" + month + "-" + day);
}
public static void main(String[] args) {
// 如果编译器会生成,则生成的构造方法一定是无参的
// 则此处创建对象是可以通过编译的
// 但实际情况是:编译期报错
Date d = new Date();
d.printDate();
}
}
/*
Error:(26, 18) java: 无法将类 extend01.Date中的构造器 Date应用到给定类型;
需要: int,int,int
找到: 没有参数
原因: 实际参数列表和形式参数列表长度不同
*/
在上面这个代码中,我们可以发现编译器进行了报错,这是因为我们在这个类中已经创建好了一个含有三个参数的构造方法,因此编译器便不会再为我们创建一个不含参数的构造方法,但是此时我们在main方法中调用了不含参数的构造方法,因此导致了编译器报错
class Date {
public int year;
public int month;
public int day;
// 无参构造方法--内部给各个成员赋值初始值,该部分功能与三个参数的构造方法重复
// 此处可以在无参构造方法中通过this调用带有三个参数的构造方法
// 但是this(1900,1,1);必须是构造方法中第一条语句
public Date(){
//System.out.println(year); 注释取消掉,编译会失败
this(1900, 1, 1);
System.out.println("调用了无参数的构造方法");
}
// 带有三个参数的构造方法
public Date(int year, int month, int day) {
this.year = year;
this.month = month;
this.day = day;
System.out.println(year + " " + month + " " + day);
}
}
public class Test {
public static void main(String[] args) {
Date date = new Date();
}
}
我们观察这段代码,会发现在其中一个构造方法中,有这么一行代码:
this(1900, 1, 1);
这段代码表示的其实就是调用了另一个包含三个参数的构造方法,并且在该方法调用完成后,还会继续执行原来的方法。
因此这段代码的执行情况是这个样子的:
2、不能形成环
public Date(){
this(1900,1,1);
}
public Date(int year, int month, int day) {
this();
}
/*
无参构造器调用三个参数的构造器,而三个参数构造器有调用无参的构造器,形成构造器的递归调用
编译报错:Error:(19, 12) java: 递归构造器调用
*/
在这段代码中,我们可以发现,在第一个构造方法中,用this调用了第二个构造方法,在第二个构造方法中,又使用this调用了第一个构造方法,两个构造方法之间形成环,导致编译器报错:
3、默认初始化
public class Date {
public int year;
public int month;
public int day;
public Date(int year, int month, int day) {
// 成员变量在定义时,并没有给初始值, 为什么就可以使用呢?
System.out.println(this.year);
System.out.println(this.month);
System.out.println(this.day);
}
public static void main(String[] args) {
// 此处a没有初始化,编译时报错:
// Error:(24, 28) java: 可能尚未初始化变量a
// int a;
// System.out.println(a);
Date d = new Date(2021,6,9);
}
}
要搞清楚这个过程,就需要知道 new 关键字背后所发生的一些事情:
Date d = new Date(2021,6,9);
1. 检测对象对应的类是否加载了,如果没有加载则加载2. 为对象分配内存空间3. 处理并发安全问题比如:多个线程同时申请对象, JVM 要保证给对象分配的空间不冲突4. 初始化所分配的空间5. 设置对象头信息
6. 调用构造方法,给对象中各个成员赋值
4. 初始化所分配的空间
4、就地初始化
public class Date {
public int year = 1900;
public int month = 1;
public int day = 1;
public Date(){
}
public Date(int year, int month, int day) {
}
public static void main(String[] args) {
Date d1 = new Date(2021,6,9);
Date d2 = new Date();
}
}
在这段代码中,我们直接先对成员变量进行了初始化,这种初始化便被称为就地初始化,但是由于不可能所有的年份都是1900,所有的日期都是1月1日,因此就地初始化通常很少被使用到。
六、封装
1、封装的概念
2、访问限定符
public :可以理解为一个人的外貌特征,谁都可以看得到default: 什么都不写的时候,对于自己家族中 ( 同一个包中 ) 不是什么秘密,对于其他人来说就是隐私了private :只有自己知道,其他人都不知道protected:受保护的(在学完继承之后才能够了解)
注意:这些只是访问权限,不仅可以修饰方法,也可以用于修饰类
1、protected 主要是用在继承中,继承部分详细介绍2、default 权限指:什么都不写时的默认权限3、访问权限除了可以限定类中成员的可见性,也可以控制类的可见性
此时,我们创建一个电脑类,来进一步了解访问限定符:
public class Computer {
private String cpu; // cpu
private String memory; // 内存
public String screen; // 屏幕
String brand; // 品牌---->default属性
public Computer(String brand, String cpu, String memory, String screen) {
this.brand = brand;
this.cpu = cpu;
this.memory = memory;
this.screen = screen;
}
public void Boot(){
System.out.println("开机~~~");
}
public void PowerOff(){
System.out.println("关机~~~");
}
public void SurfInternet(){
System.out.println("上网~~~");
}
}
public class TestComputer {
public static void main(String[] args) {
Computer p = new Computer("HW", "i7", "8G", "13*14");
System.out.println(p.brand); // default属性:只能被本包中类访问
System.out.println(p.screen); // public属性: 可以任何其他类访问
// System.out.println(p.cpu); // private属性:只能在Computer类中访问,不能被其他类访问
}
}
3、封装扩展之包
(1)包的概念
(2)导入包中的类
public class Test {
public static void main(String[] args) {
java.util.Date date = new java.util.Date();
// 得到一个毫秒级别的时间戳
System.out.println(date.getTime());
}
}
import java.util.*;
public class Test {
public static void main(String[] args) {
Date date = new Date();
// 得到一个毫秒级别的时间戳
System.out.println(date.getTime());
}
}
那么这里的.*是什么意思呢?
其实在这里,.*相当于一个通配符:可以充当任何类,但不是导入util下的所有类,而是当你用到哪个类,便会帮你导哪个类
import java.util.*;
import java.sql.*;
public class Test {
public static void main(String[] args) {
// util 和 sql 中都存在一个 Date 这样的类, 此时就会出现歧义, 编译出错
Date date = new Date();
System.out.println(date.getTime());
}
}
// 编译出错
Error:(5, 9) java: 对Date的引用不明确
java.sql 中的类 java.sql.Date 和 java.util 中的类 java.util.Date 都匹配
在这种情况下需要使用完整的类名 :
import java.util.*;
import java.sql.*;
public class Test {
public static void main(String[] args) {
java.util.Date date = new java.util.Date();
System.out.println(date.getTime());
}
}
import static java.lang.Math.*;
public class Test {
public static void main(String[] args) {
double x = 30;
double y = 40;
// 静态导入的方式写起来更方便一些.
// double result = Math.sqrt(Math.pow(x, 2) + Math.pow(y, 2));
double result = sqrt(pow(x, 2) + pow(y, 2));
System.out.println(result);
}
(3)自定义包
1、在文件的最上方加上一个 package 语句指定该代码在哪个包中 .2、包名需要尽量指定成唯一的名字 , 通常用公司的域名的颠倒形式 ( 例如 com.baidu.www ).3、包名要和代码路径相匹配. 例如创建 com.bit.demo1 的包 , 那么会存在一个对应的路径 com/bit/demo1 来存储代码.4、如果一个类没有 package 语句 , 则该类被放到一个默认包中 .5、包名一定要小写!
操作步骤:
1. 在 IDEA 中先新建一个包: 右键 src -> 新建 -> 包
2. 在弹出的对话框中输入包名, 例如 com.baidu.www
3. 在包中创建类, 右键包名 -> 新建 -> 类, 然后输入类名即可.
4. 此时可以看到我们的磁盘上的目录结构已经被 IDEA 自动创建出来了
5. 同时我们也看到了, 在新创建的 Test.java 文件的最上方, 就出现了一个 package 语句
那么这个package便是声明Java文件在哪个包当中
(4)包的访问权限控制
访问权限与访问限定符和是否调用了这个包有关,我们可以根据之前已经展示过的这个图来判断包的访问权限:
(5)常见的包
1. java.lang: 系统常用基础类 (String 、 Object), 此包从 JDK1.1 后自动导入。2. java.lang.reflflect:java 反射编程包 ;3. java.net: 进行网络编程开发包。4. java.sql: 进行数据库开发的支持包。5. java.util: 是 java 提供的工具程序包。 ( 集合类等 ) 非常重要6. java.io:I/O 编程开发包。
七、static成员
1、再谈学生类
public class Student{
// ...
public static void main(String[] args) {
Student s1 = new Student("Li leilei", "男", 18, 3.8);
Student s2 = new Student("Han MeiMei", "女", 19, 4.0);
Student s3 = new Student("Jim", "男", 18, 2.6);
}
}
假设三个同学是同一个班的,那么他们上课肯定是在同一个教室,那既然在同一个教室,那能否给类中再加一个成员变量,来保存同学上课时的教室呢?答案是不行的。
2、static修饰成员变量
static修饰的成员变量,称为静态成员变量,静态成员变量最大的特性:不属于某个具体的对象,是所有对象所共享的。
1. 不属于某个具体的对象,是类的属性,所有对象共享的,不存储在某个对象的空间中2. 既可以通过对象访问,也可以通过类名访问,但一般更推荐使用类名访问3. 类变量存储在方法区当中4. 生命周期伴随类的一生 ( 即:随类的加载而创建,随类的卸载而销毁 )
我们可以来看一个访问静态成员变量的例子:
public class Student{
public String name;
public String gender;
public int age;
public double score;
public static String classRoom = "Bit306";
// ...
public static void main(String[] args) {
// 静态成员变量可以直接通过类名访问
System.out.println(Student.classRoom);
Student s1 = new Student("Li leilei", "男", 18, 3.8);
Student s2 = new Student("Han MeiMei", "女", 19, 4.0);
Student s3 = new Student("Jim", "男", 18, 2.6);
// 也可以通过对象访问:但是classRoom是三个对象共享的
System.out.println(s1.classRoom);
System.out.println(s2.classRoom);
System.out.println(s3.classRoom);
}
}
静态成员的访问,不建议通过对象的引用访问,建议通过类名.的方式访问
类的静态成员变量通过类名访问,也就是说,这个静态成员变量不属于对象!!!
最后,我们来思考几个问题:
1、引用可以指向引用吗?
答案是不可以!! 引用只能指向对象!!
这时候有人就会提问了:下面这种情况不就是引用指向引用吗?
但是实际上,这段代码的意思是:student2这个引用指向了student1这个引用指向的对象,也就是说,实际上还是引用指向了一个对象
2、一个引用可以指向多个对象吗?
例如:
答案也是不可以!!
在这段代码种,我们可以运行发现,最终的结果是student1中存放的是"zhangsan3"和18,由此我们可以知道:一个引用不能指向多个对象
3、Person p = null;代表不指向任何对象
classRoom不在对象当中,不属于对象,而又不指向任何对象,因此不会发生空指针异常
3、static修饰成员方法v
public class Student{
private String name;
private String gender;
private int age;
private double score;
private static String classRoom = "Bit306";
// ...
}
public class TestStudent {
public static void main(String[] args) {
System.out.println(Student.classRoom);
}
}
public class Student{
// ...
private static String classRoom = "306";
// ...
public static String getClassRoom(){
return classRoom;
}
}
public class TestStudent {
public static void main(String[] args) {
System.out.println(Student.getClassRoom());
}
}
静态方法的内部是不能直接调用非静态的方法的,因为静态方法不依赖对象,但是非静态方法依赖对象
1. 不属于某个具体的对象,是类方法2. 可以通过对象调用,也可以通过类名 . 静态方法名 (...) 方式调用,更推荐使用后者3. 不能在静态方法中访问任何非静态成员变量4. 静态方法中不能调用任何非静态方法,因为非静态方法有 this 参数,在静态方法中调用时候无法传递 this 引用5. 静态方法无法重写,不能用来实现多态
3. 不能在静态方法中访问任何非静态成员变量
public static String getClassRoom(){
System.out.println(this);
return classRoom;
}
// 编译失败:Error:(35, 28) java: 无法从静态上下文中引用非静态 变量 this
public static String getClassRoom(){
age += 1;
return classRoom;
}
// 编译失败:Error:(35, 9) java: 无法从静态上下文中引用非静态 变量 age
4. 静态方法中不能调用任何非静态方法,因为非静态方法有this参数,在静态方法中调用时候无法传递this引用
public static String getClassRoom(){
doClass();
return classRoom;
}
// 编译报错:Error:(35, 9) java: 无法从静态上下文中引用非静态 方法 doClass()
4、static成员变量初始化
(1) 就地初始化
public class Student{
private String name;
private String gender;
private int age;
private double score;
private static String classRoom = "Bit306";
// ...
}
(2)静态代码块初始化
八、代码块
1、代码块概念以及分类
2、普通代码块
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
这种用法较少见
3、构造代码块
public class Student{
//实例成员变量
private String name;
private String gender;
private int age;
private double score;
public Student() {
System.out.println("I am Student init()!");
}
//实例代码块
{
this.name = "lisi";
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) {
Student stu = new Student();
stu.show();
}
}
// 运行结果
I am instance init()!
I am Student init()!
name: lisi age: 12 sex: man
4、静态代码块
public class Student{
private String name;
private String gender;
private int age;
private double score;
private static String classRoom;
//实例代码块
{
this.name = "LISI";
this.age = 12;
this.gender = "man";
System.out.println("I am instance init()!");
}
// 静态代码块
static {
classRoom = "LISI306";
System.out.println("I am static init()!");
}
public Student(){
System.out.println("I am Student init()!");
}
public static void main(String[] args) {
Student s1 = new Student();
Student s2 = new Student();
}
}
1、静态代码块不管生成多少个对象,其只会执行一次2、静态成员变量是类的属性,因此是在JVM加载类时开辟空间并初始化的3、如果一个类中包含多个静态代码块,在编译代码时,编译器会按照定义的先后次序依次执行 ( 合并 )4、实例代码块只有在创建对象时才会执行
那么这时候,我们要注意一个问题:当静态代码块,构造块,和不调用参数的构造方法之间如果同时出现,那么哪个会先执行呢?我们来做一个小实验:
class Student{
public String name;
public int age;
public String sex;
Student(){
System.out.println("不带参数的构造方法");
}
{
this.name = "lisi";
this.age = 12;
this.sex = "man";
System.out.println("构造方法块");
}
static {
System.out.println("静态方法块");
}
}
public class Test {
public static void main(String[] args) {
Student student = new Student();
}
}
那么现在我们来看一下这段代码的运行结果是什么:
由此,我们可以知道这三个代码块之间的运行顺序:
十、对象的打印
class Person {
String name;
String gender;
int age;
public Person(String name, String gender, int age) {
this.name = name;
this.gender = gender;
this.age = age;
}
public static void main(String[] args) {
Person person = new Person("Jim","男", 18);
System.out.println(person);
}
}
// 打印结果:day20210829.Person@1b6d3586
这段代码的运行结果表示的是这个对象所存储的地址
那么如果想要默认打印对象中的属性该如何处理呢?
答案:重写toString方法即可。
public class Person {
String name;
String gender;
int age;
public Person(String name, String gender, int age) {
this.name = name;
this.gender = gender;
this.age = age;
}
@Override
public String toString() {
return "[" + name + "," + gender + "," + age + "]";
}
public static void main(String[] args) {
Person person = new Person("Jim","男", 18);
System.out.println(person);
}
}
// 输出结果:[Jim,男,18]