19.this关键字
19.1 this的本质就是“创建好的对象的地址”!由于在构造方法调用前,对象已经创建。
因此,在构造方法中也可以使用this代表“当前对象”。
19.2 this最常用的用法:
1、调用成员变量:如果成员变量和局部变量同名,this必须书写,用来区分两者;
如果没有同名的局部变量,this可以不写。
2、调用成员方法:这种情况下,this可以省略。
3、调用构造方法:使用this关键字调用重载的构造方法,避免相同的初始化代码,但
只能在构造方法中用,**并且必须位于构造方法的第一句。**注意构造器之间不要相互调用,可能会导致死循环
代码实例:
public class TestCat {
public static void main(String[] args) {
Cat cat1 = new Cat("红色",10);//创建Cat对象,在两参构造方法内调用了一参构造
Cat cat2 = new Cat();
System.out.println(cat1.name);//static修饰name属性,为所有Cat类型对象共用
System.out.println(cat2.name);
}
}
public class Cat {
public String color;
static String name = "迪迦";
public int age;
public Cat() {
}
public Cat(String color) {
this.color = color; //局部和成员变量同名时,对成员变量的操作要加this指明调用
}
public Cat(String color, int age) {
//this必须是第一条
this(color);
this.age = age;
System.out.println("创建Cat对象,在两参构造方法内调用了一参构造");
}
}
4、this不能用于static方法中,因为static方法在方法区,不是对象所在的这一特定内存中
如果方法构造中形参名与属性名相同时,需要使用this关键字区分属性与形参
关于this不能出现在静态环境中,因为类加载执行时,static环境就开始跟随执行,此时对象可能还未创建,不能调用this
19.3 补充
-
this是java的一个关键字,其表示该类创建的某个对象,this既可以出现在构造方法和实例方法中,
但不可以出现在类方法中。在前者中,this代表该构造方法所创建的对象,而对于后者实例方法来说,this表示正在调用该方法的当前对象。
-
实例方法:方法类型说明符前无static关键字,有关键字的为类方法,实例方法只能通过对象来调用,不能通过类名来调用。当实例方法中含有this关键字的时候即表示正在调用该方法的当前对象
-
构造方法:名字与所在的类名完全相同(包括首字母大小写),此外该方法无类型
-
super与this无法同时存在
this构造函数中,无论显示隐示,肯定会调用父类的构造方法,当super与this同时存在时父类构造方法就必定被调用了至少两次。
-
定义money变量使用double:
养成习惯,因为大多数的时候钱的运算也是有小数点的,方便输出,和数据精确。
20.static关键字
注意! 由 static 关键字修饰的(如:类变量[静态变量]、静态代码块)将在类被初始化创建实例对象之前被初始化,而且是按顺序从上到下依次被执行
20.1 static只能修饰成员,不能修饰局部
补充( 由于方法不存在嵌套,所以也没有局部方法这一说 )
20.2 静态的是属于类的,成员的是属于对象的
加载顺序 : 先类后成员
static修饰 | |
---|---|
成员变量 | 静态变量|类变量 |
方法 | 静态方法|类方法 |
代码块 | 静态块 (可以做程序执行的初始配置) |
类 | 静态内部类 |
- 静态变量在类第一次加载到内存之后,就进行初始化!
- 静态变量是属于所属类的,它存在于方法区中的静态区中,与对象所占据的堆内存无关,并且是独一份的,为所有当前的对象所共享的
20.3 static修饰变量及方法的使用
可通过 对象.静态变量 或 对象.静态方法(参数列表) 但不建议这种方式
标准通过 类.静态变量 或 类.静态方法(参数列表) 的方式来使用
20.4 调用静态内容的区别
20.4.1 若在当前类的静态方法中调用,可直接调用该类的静态变量(内容),但是调用非静态成员内容需要通过该类的实例对象来调用
20.4.2 若在成员方法中调用,两者都可以直接使用
补充 :
-
静态成员是可变的,其执行时机比较早,也可以理解为对类型对象的初始化(约束意义)
-
对于静态方法,在类加载的时候,静态方法也已经加载(不是执行)了,但我们必须要通过类名才能访问
-
父类和子类执行顺序&对象初始化顺序:
首先执行父类静态的内容,接着去执行子类的静态的内容,–>父类构造块+父类构造方法–>子类构造块+子类构造方法
21.块-block !(自成作用域)
注意 : 作用域范围内声明的变量不能同名!
代码说明 :
public static void main(String[] args) {
int i = 10;
{
int i = 0;//报错!
}
}
21.1.1常见类型
类型 | 说明 | 执行时机 |
---|---|---|
局部代码块/普通语句块 | 在方法或者语句中定义的{ } | 所在方法被调用时会从上到下执行 |
成员代码块/构造块 | 成员位置中定义的 { } | 跟随new,且先于构造器方法中的语句之前执行,特殊条件为this或super让步,然后跟随进去! |
静态代码块 | 被static修饰的成员代码块 | 类第一次加载完成后执行,且只执行一次(先于main) |
同步块 | synchronize(任意对象){ },能保证在同一时刻最多只有一个线程执行该段代码 |
21.2.2 关于构造代码块的实现细节
在编译期间,会将构造块挪到构造器内部,在构造器代码之前的位置,当遇到this或super时,其会做出让步,到this/super之后**,然后跟随进去**
所以在编译之后,是没有构造块存在的
代码实现 :
java文件 :
public class Test {
public static void main(String[] args) {
Dog dog = new Dog();
}
}
class Dog{
private int age;
static int a = 10; //第一个执行
{
System.out.println("第二次输出");
}
public Dog(){
this(a);
System.out.println("最后一次输出");
}
public Dog(int n){
System.out.println("第三次输出");
this.age = n;
}
}
Dog类编译后的字节码文件(class) :
class Dog {
private int age;
static int a = 10;
public Dog() {
this(a);
System.out.println("最后一次输出");
}
public Dog(int n) {
System.out.println("第二次输出");
System.out.println("第三次输出");
this.age = n;
}
}
由于其早于构造方法为属性初始化信息的特点,可以用来规范默认值(如,对非0或非null)
22. "初识"封装
22.1 了解Java Bean
JavaBean(是java类)是一种JAVA语言写成的可重用组件(Component)。为写成JavaBean,类必须是具体的和公共的,并且**具有无参数的构造器。JavaBean 通过提供符合一致性设计模式的公共方法将内部域暴露成员属性,(属性私有!)** ,set和get方法获取。众所周知,属性名称符合这种模式,其他Java 类可以通过自省机制(反射机制)发现和操作这些JavaBean 的属性。
22.2 概念性问题
私有属于封装,但封装不只是私有
类、方法、属性私有化…都是封装的体现