java中的this引用及对象构造初始化

目录
  • 1. this 引用
    • 1.1 为什么要有this引用
    • 1.2 什么是this引用
    • 1.3 this引用的特性
    • 1.4 this引用练习题
  • 2. 对象的构造及初始化
    • 2.1 如何初始化对象
    • 2.2 构造方法
      • 2.2.1 概念
      • 2.2.2 特性
    • 2.3 默认初始化
    • 2.4 就地初始化

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引用 内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java面向对象程序设计:类的定义,静态变量,成员变量,构造函数,封装与私有,this概念与用法详解

    本文实例讲述了Java面向对象程序设计:类的定义,静态变量,成员变量,构造函数,封装与私有,this概念与用法.分享给大家供大家参考,具体如下: 本文内容: 什么是面对对象 类的定义与对象实例化 成员变量 成员变量的有效范围 成员变量的赋值 静态变量.方法与实例变量.方法 构造函数 封装与私有 this关键字 注:函数和方法是一样东西.[由于我之前学习的时候有些书籍不是同样的说法,书看多了,我就习惯了不同情况下用不同的说法] 首发时间:2018-03-22 什么是面对对象: 面向对象强调行为处于

  • 浅谈java面向对象(类,封装,this,构造方法)

    无论面向对象还是面向过程, 这俩都是解决问题的思路而已, 只是角度不同. 面向过程: 强调解决问题的每一个步骤都亲力亲为,每一个细节都自己手动实现. 面向对象: 使用特定功能对象去解决特定的问题, 每一个细节不需要关注,只需要创建对应的对象即可. 面向对象是基于面向过程的 类和对象及他们的关系 类: 具有相同特征和行为(功能)的事物的统称 , 是一个抽象概念 对象: 这类事物中某个确定的个体 类和对象的关系 一个类可以创建多个对象 , 类是对象的抽象, 对象是类的实例. 描述一个事物---->

  • java this引用逃逸详解

    1.什么是This逃逸? 在构造器构造还未彻底完成前(即实例初始化阶段还未完成),将自身this引用向外抛出并被其他线程复制(访问)了该引用,可能会问到该还未被初始化的变量,甚至可能会造成更大严重的问题. 废话不多说,看一下代码 /** * 模拟this逃逸 * @author Lijian * */ public class ThisEscape { //final常量会保证在构造器内完成初始化(但是仅限于未发生this逃逸的情况下,具体可以看多线程对final保证可见性的实现) final

  • Java this关键字的引用详解

    目录 为什么要有this引用? 什么是this引用 this引用的特性 为什么要有this引用? 我们先看一段代码示例 public class Date { 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

  • 详解Java基础篇--面向对象1(构造方法,static、this关键字)

    面向对象,面向过程的区别.拿下五子棋来说: 面向过程分析: 开始游戏 黑棋先走 绘制画面 判断输赢 轮到白棋 绘制画面 判断输赢 返回步骤2 输出结果 面向对象分析: 黑白双方,双方行为是一模一样的 棋盘系统,负责绘制画面 规则系统,判断犯规.输赢 传统的面向过程编程是思考问题的解决步骤,这种思维方式适用于问题规模较小时.可是当问题规模大,要求程序有更好的可扩展性,能更快速地查错时面向对象设计思想就能体现出其优势.面向对象更接近人类地自然思维方式,将现实世界中的事物抽象为对象和对象的方法. 面向

  • java中的this引用及对象构造初始化

    目录 1. this 引用 1.1 为什么要有this引用 1.2 什么是this引用 1.3 this引用的特性 1.4 this引用练习题 2. 对象的构造及初始化 2.1 如何初始化对象 2.2 构造方法 2.2.1 概念 2.2.2 特性 2.3 默认初始化 2.4 就地初始化 1. this 引用 1.1 为什么要有this引用 先来写一个日期类的例子: public class classCode { public int year; public int month; public

  • 深入理解Java中的弱引用

    不久之前,我面试了一些求职Java高级开发工程师的应聘者.我常常会面试他们说,"你能给我介绍一些Java中得弱引用吗?",如果面试者这样说,"嗯,是不是垃圾回收有关的?",我就会基本满意了,我并不期待回答是一篇诘究本末的论文描述. 然而事与愿违,我很吃惊的发现,在将近20多个有着平均5年开发经验和高学历背景的应聘者中,居然只有两个人知道弱引用的存在,但是在这两个人之中只有一个人真正了解这方面的知识.在面试过程中,我还尝试提示一些东西,来看看有没有人突然说一声&quo

  • 10分钟带你理解Java中的弱引用

    前言 本文尝试从What.Why.How这三个角度来探索Java中的弱引用,帮助大家理解Java中弱引用的定义.基本使用场景和使用方法. 一. What--什么是弱引用? Java中的弱引用具体指的是java.lang.ref.WeakReference<T>类,我们首先来看一下官方文档对它做的说明: 弱引用对象的存在不会阻止它所指向的对象被垃圾回收器回收.弱引用最常见的用途是实现规范映射(canonicalizing mappings,比如哈希表). 假设垃圾收集器在某个时间点决定一个对象是

  • Java 中的弱引用是什么

    Java里一个对象obj被创建时,被放在堆里.当GC运行的时候,发现没有任何引用指向obj,那么就会回收obj对象的堆内存空间. 换句话说,一个对象被回收, 必须满足两个条件: (1)没有任何引用指向它 (2)GC被运行. 在实际开发中,我们可以通过把所有指向某个对象的referece置空来保证这个对象在下次GC运行的时候被回收,类似下面: Object c = new Car(); c=null; 但是,这样做是一件很繁琐并且违背GC自动回收原则的事.对于简单的情况, 手动置空是不需要程序员来

  • Java中的强引用,软引用,弱引用,虚引用的作用介绍

    目录 1.强引用(StrongReference) 2.软引用(SoftReference) 3.弱引用(WeakReference) 4.虚引用(PhantomReference) 5.强引用>软引用>弱引用>虚引用 1.强引用( Strong Reference ) 最普遍的引用:Object obj=new Object()抛出OutOfMemoryError终止程序也不会回收具有强引用的对象通过将对象设置为null来弱化引用,使其被回收 2.软引用( Soft Reference

  • Java中Json字符串直接转换为对象的方法(包括多层List集合)

    使用到的类:net.sf.json.JSONObject 使用JSON时,除了要导入JSON网站上面下载的json-lib-2.2-jdk15.jar包之外,还必须有其它几个依赖包:commons-beanutils.jar,commons-httpclient.jar,commons-lang.jar,ezmorph.jar,morph-1.0.1.jar 下面是例子代码: // JSON转换 JSONObject jsonObj = JSONObject.fromObject(jsonStr

  • java中的静态代码块、构造代码块、构造方法详解

    运行下面这段代码,观察其结果: package com.test; public class HelloB extends HelloA { public HelloB() { } { System.out.println("I'm B class"); } static { System.out.println("static B"); } public static void main(String[] args) { new HelloB(); } } cla

  • 关于Java中String创建的字符串对象内存分配测试问题

    一.创建String对象的两种常用方法比较 举例说明 String str1 = "Java天下第一"; //方法1 String str2 = new String("Java天下第一"); //方法2 System.out.println(str1 == str2); //比较地址,false System.out.println(str1 == str2.intern()); //true 区别 2.1 方法1中,首先会去JVM的常量池里查找是否有存储&quo

  • 在Java中如何比较两个对象浅析

    Common Lang 中的 Builder 包内提供了一个 DiffBuilder 类,可以比较两个对象,并返回不同的部分. 首先在要比较对象的类中实现 Diffable 接口,然后实现 DiffResult diff(T obj)  方法. 在DiffResult diff(T obj)  方法中,新建一个 DiffBuilder 对象,把需要比较的类属性一一放入 DiffBuilder 中. DiffBuilder 的构造函数有三个入参,lhs 是当前对象,rhs 是要比较的对象,styl

  • Java中使用JCOM操作Office对象

    通过使用COM技术,我们用微软Office应用程序能够建立很多应用程序扩展,但是Java开发人员却无法享受它带来的便利--除非他们拥有方便的Java访问COM的途径(Java-to-COM桥).使用JCom的时候,你可以在Java中控制几乎所有的COM对象,而且它还带有一些用于Excel的强大的辅助类. 在你每次编写用HTML表格样式或Java表格对象显示数据的应用程序的时候,通常都需要带有"导出到Excel"功能.那么头疼的问题就出现了.怎么样实现这种功能呢?在HTML中显示的可以在

随机推荐