1. this 引用

1.1 为什么要有this引用

先来写一个日期类的例子:

public class classCode {
    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
        classCode Date1 = new classCode();
        classCode Date2 = new classCode();
        classCode Date3 = new classCode();

        // 对d1,d2,d3的日期设置
        Date1.setDay(2022, 8, 9);
        Date2.setDay(2023, 8, 9);
        Date3.setDay(2024, 8, 9);

        //打印日期的内容
        Date1.printDate();
        Date2.printDate();
        Date3.printDate();
    }
}

以上代码定义了一个日期类,然后main方法中创建了三个对象,并通过classCode类中的成员方法对对象进行设置和打印,代码整体逻辑非常简单,没有任何问题。

有两个需要注意的地方:

1.形参名不小心与成员变量名相同:

public void setDay(int year, int month, int day){ 
    year = year;
    month = month;
    day = day; 
}

那函数体中到底是谁给谁赋值?成员变量给成员变量?参数给参数?参数给成员变量?成员变量参数?

2.三个对象都在调用setDate和printDate函数,但是这两个函数中没有任何有关对象的说明,setDate和printDate函数如何知道打印的是哪个对象的数据呢?

可以看到如果形参名和成员变量名的话,赋值以后变量的值为0,说明并没有赋值成功。

那应该怎么做呢?往下面看。

1.2 什么是this引用

this引用指向当前对象(成员方法运行时调用该成员方法的对象),在成员方法中所有成员变量的操作,都是通过该引用去访问只不过所有的操作对用户是透明的,即用户不需要来传递,编译器自动完成。

改进之后的代码:

public void setDay(int year, int month, int day){
    this.year = year;
    this.month = month;
    this.day = day;
}

可以看到添加 this 引用后,赋值成功。
this 是默认添加的。即使是不加 this,也会有一个默认添加的 this。但是不添加的话,如果形参名和成员变量名相同就会带来问题。

在下面代码中,可以看到三个对象都在调用 setDateprintDate 函数,而且也没有说明,那该怎么知道打印的是哪个对象的数据?

public static void main(String[] args) {
     // 构造三个日期类型的对象 Date1 Date2 Date3
     Date Date1 = new Date();
     Date Date2 = new Date();
     Date Date3 = new Date();

     // 对Date1,Date2,Date3的日期设置
     Date1.setDay(2022, 8, 9);
     Date2.setDay(2023, 8, 9);
     Date3.setDay(2024, 8, 9);

     //打印日期的内容
     Date1.printDate();
     Date2.printDate();
     Date3.printDate();
}

可以通过下面两点两点来判断打印哪个对象:

  • 前面的对象是哪个对象,打印哪个对象的数据
  • 隐藏的参数。
public void setDay(Date this, int y, int m, int d){
    this.year = y;
    this.month = m;
    this.day = d;
}
public void printDate(Date this){
    System.out.println(this.year  "-"  this.month  "-"  this.day);
}

this 的三种使用方式:

  • this.成员变量
  • this.访问成员方法
  • this();访问构造方法

1.3 this引用的特性

  • this的类型:对应类类型引用,即哪个对象调用就是哪个对象的引用类。
  • this只能在"成员方法"中使用。
  • 在"成员方法"中,this只能引用当前对象,不能再引用其他对象。
  • this是“成员方法”第一个隐藏的参数,编译器会自动传递,在成员方法执行时,编译器会负责将调用成员方法。

对象的引用传递给该成员方法,this负责来接收。

就算成员变量名与形参名不相同也建议把 this 写上,这相当于一种保护措施,而且也是一种好的编程规范。

1.4 this引用练习题

写一个学术类,有姓名、年龄等属性,然后通过一个方法来设置这些属性的值,其次通过写两个方法,在一个方法当中使用this调用另一个方法。

public class Student {
    public String name;
    public int age;

    public void setStudent(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public void printStudent() {
        System.out.println(this.name   "->"   this.age);
    }

    public static void main(String[] args) {
        Student student = new Student();
        student.setStudent("zhangsan", 19);
        student.printStudent();
    }
}

2. 对象的构造及初始化

2.1 如何初始化对象

通过前面知识点的学习知道,在Java方法内部定义一个局部变量时,必须要初始化,否则会编译失败。

public static void main(String[] args) {
    int a;
    System.out.println(a);
}// Error:(26, 28) java: 可能尚未初始化变量a.

如果是一个对象即使是没赋值也不会报错,因为这是一个引用变量。

 public static void main(String[] args) {
        // 构造一个日期类型的对象 
        Date date = new Date();
        date.setDay(2022, 8, 9);
        //打印日期的内容
        date.printDate();
    }//代码可以正常通过编译

通过上述例子发现两个问题:

  • 每次对象创建好后调用setDate方法设置具体日期,比较麻烦,那对象该如何初始化?
  • 局部变量必须要初始化才能使用,为什么字段声明之后没有给值依然可以使用?

这就引入了构造方法。接着往下看。

2.2 构造方法

2.2.1 概念

构造方法(也称为构造器)是一个特殊的成员方法,名字必须与类名相同,在创建对象时,由编译器自动调用,并且在整个对象的生命周期内只调用一次。

public class Student {
    public String name;
    public int age;
    public Student(){//这是一个构造方法
        System.out.println("不带参数的构造方法");
    }
    public Student(String name, int age) {//这是一个构造方法
        System.out.println("带参数的构造方法");
        this.name = name;
        this.age = age;
    }
    public void setStudent(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public void printStudent() {
        System.out.println(this.name   "->"   this.age);
    }
    public static void main(String[] args) {
        Student student = new Student();//这一行是构造方法的调用
    }
}

Student student = new Student();
new在实例化对象,而实例化对象一定会调用构造方法。

注意:当我们没有提供构造方法时,编译器会自动提供一个不带参数的构造方法。

2.2.2 特性

  • 名字必须与类名相同。
  • 没有返回值类型,设置为void也不行。
  • 创建对象时由编译器自动调用,并且在对象的生命周期内只调用一次。
  • 构造方法可以重载(用户根据自己的需求提供不同参数的构造方法。
 public Student(){//不带参数的构造方法
        System.out.println("不带参数的构造方法");
    }
    public Student(String name, int age) {//带两个参数的构造方法
        System.out.println("带参数的构造方法");
        this.name = name;
        this.age = age;

    }

上述两个构造方法:名字相同,参数列表不同,因此构成了方法重载。

如果用户没有显式定义,编译器会生成一份默认的构造方法,生成的默认构造方法一定是无参的。

public class Work {
    public int one = 1;
    public int three = 3;
    public int two = 2;

    public void printWork() {
        System.out.println(one   "-"   two   "-"   three);
    }

    public static void main(String[] args) {
        Work work = new Work();
        work.printWork();
    }
}

上述Work类中,没有定义任何构造方法,编译器会默认生成一个不带参数的构造方法。

那如何调用带参数的构造方法呢?

public class Work {
    public int one = 1;
    public int three = 3;
    public int two = 2;

    public Work(int one, int two, int three) {
        System.out.println("带参数的构造方法");
        this.one = one;
        this.two = two;
        this.three = three;
    }

    public void printWork() {
        System.out.println(one   "-"   two   "-"   three);
    }

    public static void main(String[] args) {
        Work work = new Work(3, 6, 9);
        work.printWork();
    }
}

注意:一旦用户定义,编译器则不再生成。

构造方法中,可以通过this调用其他构造方法来简化代码。

 public Work() {
        this(10, 20, 30);//调用本类当中其他的构造方法
        System.out.println("不带参数的的构造方法");
    }

    public Work(int one, int two, int three) {
        System.out.println("带参数的构造方法");
        this.one = one;
        this.two = two;
        this.three = three;
    }

注意:

  • this调用必须在构造方法里面,
  • 要在在第一行,
  • 不能写成循环调用。

绝大多数情况下使用public来修饰,特殊场景下会被private修饰(后序讲单例模式时会遇到)

2.3 默认初始化

为什么使用成员变量不需要初始化呢?

在程序层面只是简单的一条语句,在JVM(以后讲)层面需要做好多事情,下面简单介绍下:

  • 检测对象对应的类是否加载了,如果没有加载则加载
  • 为对象分配内存空间
  • 处理并发安全问题

比如:多个线程同时申请对象,JVM要保证给对象分配的空间不冲突初始化所分配的空间
即:对象空间被申请好之后,对象中包含的成员已经设置好了初始值

比如:

设置对象头信息(关于对象内存模型后面会介绍)调用构造方法,给对象中各个成员赋值

2.4 就地初始化

定义成员变量的时候就已经赋值好了。

public class HardWork {
    public int a = 10;//就地初始化
    public int b = 20;//就地初始化
    public String c = "zhangsan";//就地初始化

    public void setWork(int a,  int b, String c) {
        this.a = a;
        this.b = b;
        this.c = c;
    }

    public void printWork() {
        System.out.println(a   "-"   b   "-"   c);
    }
    public static void main(String[] args) {
        HardWork work = new HardWork();
        work.printWork();
        System.out.println();
    }
}

注意:代码编译完成后,编译器会将所有给成员初始化的这些语句添加到各个构造函数中。

到此这篇关于java中的this引用及对象构造初始化的文章就介绍到这了,更多相关java this引用 内容请搜索Devmax以前的文章或继续浏览下面的相关文章希望大家以后多多支持Devmax!

java中的this引用及对象构造初始化的更多相关文章

  1. three.js模拟实现太阳系行星体系功能

    这篇文章主要介绍了three.js模拟实现太阳系行星体系功能,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下

  2. HTML5页面无缝闪开的问题及解决方案

    这篇文章主要介绍了HTML5页面无缝闪开方案,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  3. ios – 为什么,将nil作为参数从Objc C发送到swift类初始化器,用新对象替换nil参数

    除非属性本身被声明为nonnull:

  4. ios – 在Swift中对MKCircle进行子类化

    我想通过添加另一个String属性来继承MKCircle,我们称之为“代码”.这个属性不是可选的和常量的,所以我必须从初始化器设置它,对吧?有没有办法定义一个单一的便利初始化器,在这种情况下需要3个参数?本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. ios – AVAudioPlayer不再使用Swift 2.0/Xcode 7 beta

    对于我的iPhone应用程序中的vartestAudio声明,我在这里收到错误“调用可以抛出,但错误不能从属性初始化程序中抛出”当我转到Xcode7测试版时,就发生了这种情况.如何在Swift2.0中使用此音频剪辑?

  6. ios – 斯威夫特.在初始化所有存储的属性之前在方法调用中使用’self’

    解决方法在初始化所有非可选实例变量之前,您无法在self上调用方法.有几种方法可以解决这个问题.>将属性更改为选项或隐式解包选项(不建议)>使buildCircle()方法静态或只是一个在文件中运行并为所有圆圈调用addSubview()在所有属性初始化并且您调用之后super.init()等等.你必须避免在自己之前打电话给自己class已初始化.

  7. ios – Objective-C警告未找到超类“-init”的指定的初始化程序的方法覆盖

    我在一个应用程序中清理警告,我收到了两次这个警告对于这行代码和这一行我相当新的Objective-C和谷歌这个警告,只是不明白的解决方案我的问题是如何摆脱这些警告?

  8. ios – UICollectionView不能使用UISearchController?

    在WWDC2014年的“AInsideInsidePresentationControllers”中,演示者展示了如何在UITableView中设置UISearchController.他们通过设置searchController的searchBar框架,然后将其设置为tableView的tableHeaderView来实现.不幸的是,UICollectionView没有相当于tableHeade

  9. ios7 – 如何使用默认的IOS映像

    嗨,我是IOS开发的新手.我知道如何在IOS应用程序中使用图像.但是我不知道如何使用默认图像,如开发者站点中提到的共享或书签图标.我想用它们我必须下载这些图像集或那些可用在xcode?

  10. ios – 在词典上引用成员’subscript’

    我正在尝试为类创建一个可用的初始化程序.我的类将使用来自网络请求的输入进行初始化.网络不可靠,我想创建一个初始化器,检查所有属性上的存在,否则它将失败.我试图在这里使用守卫,所以请随时指出方法中的任何明显的错误:守卫self.jobId行无法编译,错误:对成员’下标’的模糊引用关于如何纠正这个错误的任何想法?

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部