飞道的博客

Java零基础学习之路(十)Java this关键字的使用(在实例方法中)

281人阅读  评论(0)

我们来看看this是否可以出现在static的方法当中,请看以下代码以及编译结果:


  
  1. public class ThisInStaticMethod {
  2. public static void main(String[] args) {
  3. ThisInStaticMethod.method();
  4. }
  5. public static void method(){
  6. System. out.println( this);
  7. }
  8. }

编译报错,如下图所示:

图11-2:static的方法中不能使用this

通过以上的测试得知this不能出现在static的方法当中,这是为什么呢?首先static的方法,在调用的时候是不需要创建对象的,直接采用“类名”的方式调用,也就是说static方法执行的过程中是不需要“当前对象”参与的,所以static的方法中不能使用this,因为this代表的就是“当前对象”。

大家是否还记得在之前的“封装”过程中,曾编写属性相关的set和get方法,set和get方法在声明的时候不允许带static关键字,我们把这样的方法叫做实例方法,说到实例方法,大家肯定想到了实例变量,没错,实例变量和实例方法都是对象相关,必须有对象的存在,然后通过“引用”去访问。

为什么set和get方法设计为实例方法呢?那是因为set和get方法操作的是实例变量,“不同的对象”调用get方法最终得到的数据是不同的,例如zhangsan调用getName()方法得到的名字是zhangsan,lisi调用getName()方法得到的名字是lisi,显然get方法是一个对象级别的方法,不能直接采用“类名”调用,必须先创建对象,再通过“引用”去访问。

this可以出现在实例方法当中,因为实例方法在执行的时候一定是对象去触发的,实例方法一定是对象才能去调用的,而this恰巧又代表“当前对象”,所以“谁”去调用这个实例方法this就是“谁”。测试一下,请看以下代码:


  
  1. public class Customer {
  2. private String name;
  3. public Customer(){
  4. }
  5. public Customer(String _name){
  6. name = _name;
  7. }
  8. public void setName(String _name){
  9. name = _name;
  10. }
  11. public String getName(){
  12. return name;
  13. }
  14. public void shopping(){
  15. System. out.println( "shopping() --> " + this);
  16. }
  17. }

 


  
  1. public class CustomerTest {
  2. public static void main(String[] args) {
  3. Customer jack = new Customer( "jack");
  4. System. out.println( "main() ---> " + jack);
  5. jack.shopping();
  6. System. out.println( "====================");
  7. Customer rose = new Customer( "rose");
  8. System. out.println( "main() ---> " + rose);
  9. rose.shopping();
  10. }
  11. }

运行结果如下图所示:

图11-3:测试this

以上代码的输出结果具体是什么不重要,重要的是可以看出谁和谁是相等的。运行结果和代码结合起来分析一下this:

图11-4:this指向了当前对象

通过以上内容的学习得知,this可以使用在实例方法当中,它指向当前正在执行这个动作的对象。

大家是否还记得实例变量怎么访问?正规的访问方式是采用“引用.”去访问。请看下面的代码:


  
  1. public class Customer {
  2. private String name;
  3. public Customer(){
  4. }
  5. public Customer(String _name){
  6. name = _name;
  7. }
  8. public void setName(String _name){
  9. name = _name;
  10. }
  11. public String getName(){
  12. return name;
  13. }
  14. public void shopping(){
  15. System. out.println(name + " is shopping!");
  16. }
  17. }

 


  
  1. public class CustomerTest {
  2. public static void main(String[] args) {
  3. Customer jack = new Customer( "jack");
  4. jack.shopping();
  5. Customer rose = new Customer( "rose");
  6. rose.shopping();
  7. }
  8. }

运行结果如下图所示:

图11-5:测试结果

将以上部分代码片段取出来进行分析:


  
  1. public class Customer {
  2. private String name; //实例变量
  3. ...
  4. public void shopping(){
  5. //jack调用shopping,当前对象是jack
  6. //rose调用shopping,当前对象是rose
  7. //name是实例变量,不用“引用”可以访问?(以上结果表示可以)
  8. System. out.println(name + " is shopping!");
  9. //正规的访问方式应该是“引用.name”,比如
  10. //System.out.println(jack.name + " is shopping!");
  11. //或者
  12. //System.out.println(rose.name + " is shopping!");
  13. //对不起,jack和rose在main方法当中,在这里不可见,不能用
  14. //难道是这样???
  15. System. out.println( this.name + " is shopping!");
  16. }
  17. }

 


  
  1. public class CustomerTest {
  2. public static void main(String[] args) {
  3. Customer jack = new Customer( "jack");
  4. jack.shopping();
  5. Customer rose = new Customer( "rose");
  6. rose.shopping();
  7. }
  8. }

把完整的代码拿过来:


  
  1. public class Customer {
  2. private String name;
  3. public Customer(){
  4. }
  5. public Customer(String _name){
  6. name = _name;
  7. }
  8. public void setName(String _name){
  9. name = _name;
  10. }
  11. public String getName(){
  12. return name;
  13. }
  14. public void shopping(){
  15. System. out.println(name + " is shopping!");
  16. System. out.println( this.name + " is shopping!");
  17. }
  18. }

 


  
  1. public class CustomerTest {
  2. public static void main(String[] args) {
  3. Customer jack = new Customer( "jack");
  4. jack.shopping();
  5. System. out.println( "=======================");
  6. Customer rose = new Customer( "rose");
  7. rose.shopping();
  8. }
  9. }

运行结果如下图所示:

通过以上的测试我们得知:System.out.println(name + " is shopping!")和System.out.println(this.name + " is shopping!")是等效的。也就是说在shopping()这个“实例方法”当中直接访问“实例变量”name就表示访问当前对象的name。换句话说在实例方法中可以直接访问当前对象的实例变量,而“this.”是可以省略的。“this.”什么时候不能省略呢?请看以下代码:


  
  1. public class Customer {
  2. private String name;
  3. public Customer(){
  4. }
  5. public Customer(String _name){
  6. name = _name;
  7. }
  8. public void setName(String _name){
  9. name = _name;
  10. }
  11. public String getName(){
  12. return name;
  13. }
  14. public void shopping(){
  15. System. out.println(name + " is shopping!");
  16. }
  17. }

你有没有看到name=_name这样的代码很丑陋,怎样可以优雅一些呢?请看:


  
  1. public class Customer {
  2. private String name;
  3. public Customer(){
  4. }
  5. public Customer(String name){
  6. this.name = name; //这里的“this.”不能省略
  7. }
  8. public void setName(String name){
  9. this.name = name; //这里的“this.”不能省略
  10. }
  11. public String getName(){
  12. return name; //这里的“this.”可以省略
  13. }
  14. public void shopping(){
  15. //这里的“this.”可以省略
  16. System. out.println(name + " is shopping!");
  17. }
  18. }

以上代码当中this.name = name,其中this.name表示实例变量name,等号右边的name是局部变量name,此时如果省略“this.”,则变成name = name,这两个name都是局部变量(java遵守就近原则),和实例变量name无关了,显然是不可以省略“this.”的。

最终的结论是,this不能出现在static的方法中,可以出现在实例方法中,代表当前对象,大部分情况下this都是可以省略的,只有当在实例方法中区分局部变量和实例变量的时候不能省略。

接下来我们再来扩展一下this的使用,请看代码:


  
  1. public class Customer {
  2. private String name;
  3. public Customer(){
  4. }
  5. public Customer(String name){
  6. this.name = name;
  7. }
  8. public void setName(String name){
  9. this.name = name;
  10. }
  11. public String getName(){
  12. return name;
  13. }
  14. //实例方法
  15. public void shopping(){
  16. System. out.println(name + " is shopping!");
  17. System. out.println(name + " 选好商品了!");
  18. //pay()支付方法是实例方法,实例方法需要使用“引用”调用
  19. //那么这个“引用”是谁呢?
  20. //当前对象在购物,肯定是当前对象在支付,所以引用是this
  21. this.pay();
  22. //同样“this.”可以省略
  23. pay();
  24. }
  25. //实例方法
  26. public void pay(){
  27. System. out.println(name + "支付成功!");
  28. }
  29. }

 


  
  1. public class CustomerTest {
  2. public static void main(String[] args) {
  3. Customer jack = new Customer( "jack");
  4. jack.shopping();
  5. System. out.println( "=======================");
  6. Customer rose = new Customer( "rose");
  7. rose.shopping();
  8. }
  9. }

运行结果如下图所示:

 

图11-7:测试结果

通过以上的测试,可以看出在一个实例方法当中可以直接去访问其它的实例方法,方法是对象的一种行为描述,实例方法中直接调用其它的实例方法,就表示“当前对象”完成了一系列行为动作。例如在实例方法shopping()中调用另一个实例方法pay(),这个过程就表示jack在选购商品,选好商品之后,完成支付环节,其中选购商品是一个动作,完成支付是另一个动作。接下来继续扩展,请看以下代码:


  
  1. public class ThisTest {
  2. int i = 10;
  3. public static void main(String[] args) {
  4. System. out.println(i);
  5. }
  6. }

以上代码编译报错了,请看:

图11-8:编译错误提示信息

为什么会编译报错,在main方法中为什么无法直接访问变量i?我们来分析一下,首先i变量是实例变量,实例变量要想访问必须先创建对象,然后通过“引用”去访问,main方法是static的方法,也就是说main方法是通过“类名”去调用的,在main方法中没有“当前对象”的概念,也就是说main方法中不能使用this,所以编译报错了。那应该怎么修改呢?请看:


  
  1. public class ThisTest {
  2. int i = 10;
  3. public static void main(String[] args) {
  4. //这肯定是不行的,因为main方法带有static,不能用this
  5. //System.out.println(this.i);
  6. //可以自己创建一个对象
  7. ThisTest tt = new ThisTest();
  8. //通过引用访问
  9. System. out.println(tt.i);
  10. }
  11. }

运行结果如下图所示:

图11-9:测试结果

通过以上的测试得知,在static的方法中不能直接访问实例变量,要访问实例变量必须先自己创建一个对象,通过“引用”可以去访问,不能通过this访问,因为在static方法中是不能存在this的。其实这种设计也是有道理的,因为static的方法在执行的时候是采用“类名”去调用,没有对象的参与,自然也不会存在当前对象,所以static的方法执行过程中不存在this。那么在static方法中能够直接访问实例方法吗?请看以下代码:


  
  1. public class ThisTest {
  2. public static void main(String[] args) {
  3. doSome();
  4. }
  5. public void doSome(){
  6. System. out.println( "do some...");
  7. }
  8. }

编译报错了,请看下图:

图11-10:编译报错提示信息

为什么在main方法中无法直接调用实例方法doSome()呢?很简单,因为实例方法必须先创建对象,通过引用去调用,在以上的main方法中并没有创建对象,更没有this。所以在main方法中无法直接访问实例方法。结论就是:在static的方法中不能直接访问实例方法。怎么修改呢?同样需要先创建对象,请看:


  
  1. public class ThisTest {
  2. public static void main(String[] args) {
  3. ThisTest tt = new ThisTest();
  4. tt.doSome();
  5. }
  6. public void doSome(){
  7. System. out.println( "do some...");
  8. }
  9. }

运行结果如下图所示:

图11-11:运行结果

综上所述,我们需要记住这样的一个结论:this不能使用在static的方法中,可以使用在实例方法中,代表当前对象,多数情况下this是可以省略不写的,但是在区分局部变量和实例变量的时候不能省略,在实例方法中可以直接访问当前对象实例变量以及实例方法,在static方法中无法直接访问实例变量和实例方法。


转载:https://blog.csdn.net/bjpowernode_com/article/details/109855101
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场