Java 方法 Java的方法和this关键字怎样理解与应用
spring小杨 人气:0java类的方法,我特别喜欢《java编程思想》里面的描述,这本书说java类之间的相互通信是通过消息。比如顾客类的对象调用一个eat方法,可以看成是我们向顾客类发送了一条消息,命令他去吃饭。
步骤 1 、给顾客增加一个吃饭的方法
假设我们有一个顾客类:
class Customer { String name; // 客户姓名 String sex; //性别 String birthDate; //生日 String phoneNumber; //电话号码 int status; //客户状态(1:正常,2:不正常) void eat(){ System.out.println("eating..."); } }
void代表方法没有返回值。
步骤 2 、 没有加static的属性和方法,一定需要先new对象
这是一句口诀,new的意思是新建一个对象,Customer类只是一个声明式的东西,不是真实存在的对象。
只要你没有加static,就必须要在运行的时候创建了对象,才可以使用!
说得武断一点,只要你没有加static,你要运行eat方法,就必须要new一个对象。
至于为什么,现在先不管,就这么强记一下吧。
步骤 3 、 用new出来的对象去执行eat方法
class Demo { public static void main(String[] args){ Customer c = new Customer(); c.eat(); } }
运行,就可以看到eat方法被调用了。
步骤 4 、 怎么理解c.eat()
c是顾客对象的一个引用,或者叫指针。c就是一个标记,本身不会保存顾客对象的数据,但是我们操作c,就和操作顾客对象是一样的。
这行代码代表我们向这个顾客发送了一条吃饭消息,顾客很聪明的,一下子就懂了,于是他就用一个句点 . 加上eat(方法名),再打一个圆括号的方式,执行了这个方法。
这是一次成功的通讯。
步骤 5 、 消息接受器
一个类里面的方法,我们可以把它称为消息接收器,这样非常有利于理解。
步骤 6 、 如果有两个顾客?
问个问题,如果有两个顾客,一个叫亚瑟,一个叫安琪拉。那么他们调用的eat方法是相同的还是不同的。
如果这个出面试题,是很有区分度的。
步骤 7 、 答案
答案是相同的方法,没想到吧。
原因是类加载器把class文件加载到虚拟机,内存的划分会把类的所有方法都放到一个叫做方法区的地方。不管你new多少个对象,调用的都是同一个方法。因为如果你每new一个对象,都专门开辟一个内存空间来存放方法的话,未免也太浪费了。
现在就有一个新的问题,既然方法是同一份,我怎么知道是谁呢?
步骤 8 、其实有个this
事情的真相是这样的…
Demo.java被javac命令编译成Demo.class和Customer.class,然后运行的时候,首先是类加载器去加载Demo.class,发现还用到了Customer.class,于是顺便把Customer.class也加载了。
在这个例子中,JVM里面有两个地方需要我们关注,一个是堆,一个是方法区。
什么是堆?堆是一块存放对象的区域,你每次new一个对象,这个对象就是放在堆里面。比如有这样的代码:
class Demo { public static void main(String[] args){ Customer c1 = new Customer(); Customer c2 = new Customer(); c1.name = "亚瑟"; c2.name = "安其拉"; } }
我们new了两个对象,其中c1和c2是他们的引用,这两个对象是放在堆里面的。
而eat方法,是放在方法区,只有一份。
然后重头戏来了,当你用其中某一个对象去调用eat方法的时候,其实会传过去一个this。
c1.eat();
this会神不知鬼不觉地作为eat方法的参数,传入eat方法。
每一个对象都有一个this,this并不是你设置的某个属性,而是java给每一个类都设置的标记,你察觉不到它的存在,但是它一直都在。
步骤 9 、 在eat方法里面直接使用this
我们在eat方法里面直接使用this,就是这样的:
void eat(){ System.out.println(this.name + " eating..."); }
效果:
其实,它的底层是这样的,虽然我们不能这么写:
void eat(this){ System.out.println(this.name + " eating..."); }
我们在方法里面调用某个变量,这个变量要么就是参数传进来的,要么就是这个类的属性,这个你同意不?基于这个常理,这个this不是我们定义的属性,那么只能是参数传进来的。只不过,这个过程是JVM做的,我们察觉不到。
步骤 10 、 构造方法
每一个类都有构造方法,写法是这样的。
public Customer(){ }
这是空的构造方法,你不写也没事,就算不写也会有这么一个方法。构造方法的作用就是在你new对象的时候,直接调用。
记住一个口诀:
1.类名打括号 = 调用构造方法,左边一定会有个new。
2.方法名打括号 = 调用这个方法,默认传入一个this(没有static的情况)
3.方法名打括号 = 调用这个方法,不传入this(有static的情况)
现在再回过头来看new的语句,是不是对上了?
Customer c1 = new Customer();
步骤 11 、 总结:this的意义是什么?
因为方法是独一份,所有类的对象共享这些方法,JVM为了搞清楚到底是谁去调用某个方法,就在每一个类里面维护一个this。当类被实例化(就是new的时候),这个this就指向这个对象,然后在调用方法的时候,隐式地传入this,就可以在那个方法中获取到当前真实对象的数据了。
就有点像什么呢,有点像花名。
什么叫花名,比如小区的物业会设置一个花名,他们对外会宣称,管理1-10栋的物业管家的名字是小芳,这个小芳就是花名。这是一个虚拟的名字,业主也只知道物业管家是小芳,但是小芳可能今天是张三,明天是李四。张三李四都离职了,王五成为物业管家,对外的说法也是他叫管家小芳。
物业设置管家这个职位,并不知道究竟谁来做,于是先设置好,管家就叫做小芳。小芳就相当于this,只有等到张三来任职了,这个this就是张三。张三跑路,李四来了,这个this就是李四。
步骤 12 、 道理我都懂,那static又是什么?
我们已经不止一次提到static了,static到底是啥啊,别急,下节我们来分享,先把这一节的知识点消化一下吧。
步骤 13 、 本节完整源码
class Customer { String name; // 客户姓名 String sex; //性别 String birthDate; //生日 String phoneNumber; //电话号码 int status; //客户状态(1:正常,2:不正常) public Customer(){ } void eat(){ System.out.println(this.name + " eating..."); } } class Vip { } class Demo { public static void main(String[] args){ Customer c1 = new Customer(); Customer c2 = new Customer(); c1.name = "亚瑟"; c2.name = "安其拉"; c1.eat(); } }
视频讲解:
加载全部内容