我们来看看this是否可以出现在static的方法当中,请看以下代码以及编译结果:
-
public
class
ThisInStaticMethod {
-
public static void main(String[] args) {
-
ThisInStaticMethod.method();
-
}
-
public static void method(){
-
System.
out.println(
this);
-
}
-
}
编译报错,如下图所示:
图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就是“谁”。测试一下,请看以下代码:
-
public
class
Customer {
-
private String name;
-
public Customer(){
-
}
-
public Customer(String _name){
-
name = _name;
-
}
-
public void setName(String _name){
-
name = _name;
-
}
-
public String getName(){
-
return name;
-
}
-
public void shopping(){
-
System.
out.println(
"shopping() --> " +
this);
-
}
-
}
-
public
class
CustomerTest {
-
public static void main(String[] args) {
-
Customer jack =
new Customer(
"jack");
-
System.
out.println(
"main() ---> " + jack);
-
jack.shopping();
-
System.
out.println(
"====================");
-
Customer rose =
new Customer(
"rose");
-
System.
out.println(
"main() ---> " + rose);
-
rose.shopping();
-
}
-
}
运行结果如下图所示:
图11-3:测试this
以上代码的输出结果具体是什么不重要,重要的是可以看出谁和谁是相等的。运行结果和代码结合起来分析一下this:
图11-4:this指向了当前对象
通过以上内容的学习得知,this可以使用在实例方法当中,它指向当前正在执行这个动作的对象。
大家是否还记得实例变量怎么访问?正规的访问方式是采用“引用.”去访问。请看下面的代码:
-
public
class
Customer {
-
private String name;
-
public Customer(){
-
}
-
public Customer(String _name){
-
name = _name;
-
}
-
public void setName(String _name){
-
name = _name;
-
}
-
public String getName(){
-
return name;
-
}
-
public void shopping(){
-
System.
out.println(name +
" is shopping!");
-
}
-
}
-
public
class CustomerTest {
-
public static void main(String[] args) {
-
Customer jack =
new Customer(
"jack");
-
jack.shopping();
-
Customer rose =
new Customer(
"rose");
-
rose.shopping();
-
}
-
}
运行结果如下图所示:
图11-5:测试结果
将以上部分代码片段取出来进行分析:
-
public
class
Customer {
-
private String name;
//实例变量
-
...
-
public void shopping(){
-
//jack调用shopping,当前对象是jack
-
//rose调用shopping,当前对象是rose
-
//name是实例变量,不用“引用”可以访问?(以上结果表示可以)
-
System.
out.println(name +
" is shopping!");
-
//正规的访问方式应该是“引用.name”,比如
-
//System.out.println(jack.name + " is shopping!");
-
//或者
-
//System.out.println(rose.name + " is shopping!");
-
//对不起,jack和rose在main方法当中,在这里不可见,不能用
-
//难道是这样???
-
System.
out.println(
this.name +
" is shopping!");
-
}
-
}
-
public
class CustomerTest {
-
public static void main(String[] args) {
-
Customer jack =
new Customer(
"jack");
-
jack.shopping();
-
Customer rose =
new Customer(
"rose");
-
rose.shopping();
-
}
-
}
把完整的代码拿过来:
-
public
class
Customer {
-
private String name;
-
public Customer(){
-
}
-
public Customer(String _name){
-
name = _name;
-
}
-
public void setName(String _name){
-
name = _name;
-
}
-
public String getName(){
-
return name;
-
}
-
public void shopping(){
-
System.
out.println(name +
" is shopping!");
-
System.
out.println(
this.name +
" is shopping!");
-
}
-
}
-
public
class
CustomerTest {
-
public static void main(String[] args) {
-
Customer jack =
new Customer(
"jack");
-
jack.shopping();
-
System.
out.println(
"=======================");
-
Customer rose =
new Customer(
"rose");
-
rose.shopping();
-
}
-
}
运行结果如下图所示:
通过以上的测试我们得知:System.out.println(name + " is shopping!")和System.out.println(this.name + " is shopping!")是等效的。也就是说在shopping()这个“实例方法”当中直接访问“实例变量”name就表示访问当前对象的name。换句话说在实例方法中可以直接访问当前对象的实例变量,而“this.”是可以省略的。“this.”什么时候不能省略呢?请看以下代码:
-
public
class
Customer {
-
private String name;
-
public Customer(){
-
}
-
public Customer(String _name){
-
name = _name;
-
}
-
public void setName(String _name){
-
name = _name;
-
}
-
public String getName(){
-
return name;
-
}
-
public void shopping(){
-
System.
out.println(name +
" is shopping!");
-
}
-
}
你有没有看到name=_name这样的代码很丑陋,怎样可以优雅一些呢?请看:
-
public
class
Customer {
-
private String name;
-
public Customer(){
-
}
-
public Customer(String name){
-
this.name = name;
//这里的“this.”不能省略
-
}
-
public void setName(String name){
-
this.name = name;
//这里的“this.”不能省略
-
}
-
public String getName(){
-
return name;
//这里的“this.”可以省略
-
}
-
public void shopping(){
-
//这里的“this.”可以省略
-
System.
out.println(name +
" is shopping!");
-
}
-
}
以上代码当中this.name = name,其中this.name表示实例变量name,等号右边的name是局部变量name,此时如果省略“this.”,则变成name = name,这两个name都是局部变量(java遵守就近原则),和实例变量name无关了,显然是不可以省略“this.”的。
最终的结论是,this不能出现在static的方法中,可以出现在实例方法中,代表当前对象,大部分情况下this都是可以省略的,只有当在实例方法中区分局部变量和实例变量的时候不能省略。
接下来我们再来扩展一下this的使用,请看代码:
-
public
class
Customer {
-
private String name;
-
public Customer(){
-
}
-
public Customer(String name){
-
this.name = name;
-
}
-
public void setName(String name){
-
this.name = name;
-
}
-
public String getName(){
-
return name;
-
}
-
//实例方法
-
public void shopping(){
-
System.
out.println(name +
" is shopping!");
-
System.
out.println(name +
" 选好商品了!");
-
//pay()支付方法是实例方法,实例方法需要使用“引用”调用
-
//那么这个“引用”是谁呢?
-
//当前对象在购物,肯定是当前对象在支付,所以引用是this
-
this.pay();
-
//同样“this.”可以省略
-
pay();
-
}
-
//实例方法
-
public void pay(){
-
System.
out.println(name +
"支付成功!");
-
}
-
}
-
public
class
CustomerTest {
-
public static void main(String[] args) {
-
Customer jack =
new Customer(
"jack");
-
jack.shopping();
-
System.
out.println(
"=======================");
-
Customer rose =
new Customer(
"rose");
-
rose.shopping();
-
}
-
}
运行结果如下图所示:
图11-7:测试结果
通过以上的测试,可以看出在一个实例方法当中可以直接去访问其它的实例方法,方法是对象的一种行为描述,实例方法中直接调用其它的实例方法,就表示“当前对象”完成了一系列行为动作。例如在实例方法shopping()中调用另一个实例方法pay(),这个过程就表示jack在选购商品,选好商品之后,完成支付环节,其中选购商品是一个动作,完成支付是另一个动作。接下来继续扩展,请看以下代码:
-
public
class
ThisTest {
-
int i =
10;
-
public static void main(String[] args) {
-
System.
out.println(i);
-
}
-
}
以上代码编译报错了,请看:
图11-8:编译错误提示信息
为什么会编译报错,在main方法中为什么无法直接访问变量i?我们来分析一下,首先i变量是实例变量,实例变量要想访问必须先创建对象,然后通过“引用”去访问,main方法是static的方法,也就是说main方法是通过“类名”去调用的,在main方法中没有“当前对象”的概念,也就是说main方法中不能使用this,所以编译报错了。那应该怎么修改呢?请看:
-
public
class
ThisTest {
-
int i =
10;
-
public static void main(String[] args) {
-
//这肯定是不行的,因为main方法带有static,不能用this
-
//System.out.println(this.i);
-
//可以自己创建一个对象
-
ThisTest tt =
new ThisTest();
-
//通过引用访问
-
System.
out.println(tt.i);
-
}
-
}
运行结果如下图所示:
图11-9:测试结果
通过以上的测试得知,在static的方法中不能直接访问实例变量,要访问实例变量必须先自己创建一个对象,通过“引用”可以去访问,不能通过this访问,因为在static方法中是不能存在this的。其实这种设计也是有道理的,因为static的方法在执行的时候是采用“类名”去调用,没有对象的参与,自然也不会存在当前对象,所以static的方法执行过程中不存在this。那么在static方法中能够直接访问实例方法吗?请看以下代码:
-
public
class
ThisTest {
-
public static void main(String[] args) {
-
doSome();
-
}
-
public void doSome(){
-
System.
out.println(
"do some...");
-
}
-
}
编译报错了,请看下图:
图11-10:编译报错提示信息
为什么在main方法中无法直接调用实例方法doSome()呢?很简单,因为实例方法必须先创建对象,通过引用去调用,在以上的main方法中并没有创建对象,更没有this。所以在main方法中无法直接访问实例方法。结论就是:在static的方法中不能直接访问实例方法。怎么修改呢?同样需要先创建对象,请看:
-
public
class
ThisTest {
-
public static void main(String[] args) {
-
ThisTest tt =
new ThisTest();
-
tt.doSome();
-
}
-
public void doSome(){
-
System.
out.println(
"do some...");
-
}
-
}
运行结果如下图所示:
图11-11:运行结果
综上所述,我们需要记住这样的一个结论:this不能使用在static的方法中,可以使用在实例方法中,代表当前对象,多数情况下this是可以省略不写的,但是在区分局部变量和实例变量的时候不能省略,在实例方法中可以直接访问当前对象实例变量以及实例方法,在static方法中无法直接访问实例变量和实例方法。
转载:https://blog.csdn.net/bjpowernode_com/article/details/109855101