草庐IT

全面理解java中的构造方法以及this关键字的用法(超详细)

是小鱼儿哈 2023-04-15 原文

Hello,各位铁汁们!我是小🐟儿哈!今天我又来更新我的Java基础学习博客了。

本篇主要内容概述:

1、🍚如何用构造方法初始化对象

2、🍚为啥要有this这个关键字

3、🍚this.属性名访问成员变量、成员方法

4、🍚this.方法名 || this.()的用法

目录

初识构造方法 

构造方法的使用 

初识this 

this.xx的用法

this()用于构造函数的调用


初识构造方法 

我们上篇讲了java中类的创建,那么让我们来实战演练一下:创建一个学生类,里面有学生的基本信息,包括姓名、性别、年龄、学号,你可能会写出这样的代码:

class Student {
    String name;
    String gender;
    int age;
    long studentID;
}
public class TestDemo2 {
    public static void main(String[] args) {
        Student stu1 = new Student();
        stu1.name = "张三";                  // 给stu1对象的各个成员变量赋值
        stu1.gender = "男";
        stu1.age = 19;
        stu1.studentID = 231245431;
        // 打印输出
        System.out.println("姓名:" + stu1.name + " 性别:" + stu1.gender + " 年龄:" + stu1.age + " 学号:" + stu1.studentID);
    }
}

 🍑但你在写的时候,是不是感觉给对象的成员变量一个一个的赋值太繁琐,能不能一下子就把值赋好呢?还真能,用Java中的构造方法就能做到这一点

那么快来看看什么是构造方法吧!

🍑构造方法是类的一种特殊方法,用来初始化类的一个新的对象,在创建对象(new 运算符)之后自动调用,Java中的每个类都有一个默认的构造方法,并且可以有一个以上的构造方法。

Java 构造方法有以下特点:

  • 方法名必须与类名相同
  • 可以有 0 个、1 个或多个参数
  • 没有任何返回值,包括 void
  • 默认返回类型就是对象类型本身
  • 只能与 new 运算符结合使用


🔔值得注意的是,如果为构造方法定义了返回值类型或使用 void 声明构造方法没有返回值,编译时不会出错,但 Java 会把这个所谓的构造方法当成普通方法来处理

🍑下面就是一个构造方法的使用例子

class Student {
    String name;   //学生类的属性
    int age;
    public void eat() {
        System.out.println(name + "在吃饭");   // 学生类的行为(方法)
    }
    public Student() {
        System.out.println("这是自定义的一个不带参数的构造方法");
    }
}
public class TestDemo3 {
    public static void main(String[] args) {
        Student student1 = new Student();
    }
}

看到输出结果,你是不是感到:咦?不对啊,我的main方法里明明就只是实例化了一个对象啊😮!我没输出啊,为啥还会输出构造方法中的内容啊

  Student student1 = new Student();

🍑原因就是:当类的对象被创建时,该构造方法将被自动调用

🍑其实我们在用new关键字实例化对象时 , 程序一定干了这两件事(但可能不只有这两步)

  1. 📝在堆区分配对象需要的内存
  2. 📝调用合适的构造方法

🍑看到这,你可能又有问题了,我们之前实例化对象的时候明明没定义构造方法呀!怎么能说实例化对象时程序一定调用构造方法呢?

😊是这样的,当一个类中没有提供任何构造方法,系统默认提供一个无参数的构造方法,就像这样

Student() {
}

🔔 但如果类中显式地定义了一个或多个构造方法,则系统不再提供默认构造方法。
 

在一个类中,与类名相同的方法就是构造方法。每个类可以具有多个构造方法,但要求它们各自包含不同的方法参数,比如这样:

Student() {
    System.out.println("这是自定义的一个没有参数的构造方法");
}
Student(String name) {
    this.name = name;
    System.out.println("这是自定义的带有一个参数的构造方法");
}

  • 🌻 该示例就定义了两个构造方法,分别是无参构造方法和有参构造方法。
  • 🌻如果在一个类中定义多个具有不同参数的同名方法,称作方法的重载。
  • 🌻这两个构造方法的名称都与类名相同,均为Student。在实例化该类时可以调用不同的构造方法进行初始化。


构造方法的使用 

 那么,就让我们来看一下,调用构造方法是怎样解决上面那个麻烦的问题的,嘻嘻😎

class Student {
    String name;
    String gender;
    int age;
    long studentID;
    // 该构造方法可由IDEA自动生成
    public Student(String name, String gender, int age, long studentID) {  
        this.name = name;
        this.gender = gender;
        this.age = age;
        this.studentID = studentID;
    }
}
public class TestDemo2 {
    public static void main(String[] args) {
        Student stu1 = new Student("张三", "男", 19, 231245431);
        System.out.println("姓名:" + stu1.name + " 性别:" + stu1.gender + " 年龄:" + stu1.age + " 学号:" + stu1.studentID);
    }
}

 🍑怎么样,是不是一下子就把值赋好,哈哈😉,并且类中的构造方法的定义还不用自己写,在IDEA有相应的快捷键可以自动生成,就像这样:

在IDEA里点击鼠标右键,弹出窗口-->在窗口中选择Generate-->点进去后选择Constructo-->按住ctrl键的同时点击鼠标,把那几个参数全选上,点击OK😎

嘻嘻,是不是IDEA帮你自动生成了🥰,方便吧


初识this 

 不过,构造方法中的this.name是什么意思啊,以前不都是 "对象名.成员变量" 吗,来接下来,我们慢慢说这个this到底是什么东东🤔

首先,咱先来看一段代码😉

class Date {
    int year;
    int month;
    int day;
    // 通过构造函数传参来赋值
    public Date(int year, int month, int day) {
        year = year;
        month = month;
        day = day;
    }

    public void printDate() {
        System.out.println(year + "/" + month + "/" + day);
    }
}

public class TestDemo2 {
    public static void main(String[] args) {
        Date date1 = new Date(2022, 4, 2);
        date1.printDate();
    }
}

这段代码有什么问题吗?细心的同学可能以经发现了 :在Date类中,传入构造方法的参数名称和Date的成员变量名字相同🤔,那么在Date类中的构造方法中就会出现一个问题:

public Date(int year, int month, int day) {
        // 那函数体中到底是谁给谁赋值?成员变量给成员变量?参数给参数?参数给成员变量?
        // 成员变量给参数?估计自己都搞不清楚了
        year = year;            // 当传入的参数相同是怎么办!
        month = month;
        day = day;
}

大家觉得在上面的main方法中会输出什么?是2022/4/2吗😉? 

让我们一起来看看吧?

😮输出竟然是0/0/0,好像我们根本没给year,month,day赋值似的? 

其实当构造方法的参数与类所定义的属性同名时,对于方法中的year = year,虚拟机不清楚在这里year到底是传过来的参数还是对象的成员变量,那么为了让虚拟机知道我们是把参数2022赋值给当前对象的成员变量year,我们就需要借助java中this这个关键字:

class Date {
    int year;
    int month;
    int day;
    // 添加了this关键字的构造函数
    public Date(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }

    public void printDate() {
        System.out.println(year + "/" + month + "/" + day);
    }
}

public class TestDemo2 {
    public static void main(String[] args) {
        Date date1 = new Date(2022, 4, 2);
        date1.printDate();
    }
}

 

 💡那为什么用this.name = name 赋值就好了呢?


this.xx的用法

🏀按照官方正规的解释,this关键字的意义被解释为“指向当前对象的引用”,比如在上述代码中当前所实例化的对象是date1,那么this其实就指向date1,是date1的一个引用(也可以理解为是date1的一个别名)那么this.xxx其实就代表了date1.xxx😎

🏀所以在执行”this.name=name;”这条语句的时候,虚拟机就会把参数year的值”2022”赋值给对象date1的year成员变量。也就是说在这条赋值语句中,”=”左边的”this.year”表示对象date1的year成员变量,而”=”右边的year表示传给我们构造方法的参数

🏀讲到这里,有的小伙伴可能会问:”this.year”为什么不能被解释为”当前对象date1自己的year参数”呢🤔?

🍑因为”参数”这个概念是就某个方法而言的,它相当于某个方法的”局部变量”,只是这个”局部变量”比起在方法中定义的真正的局部变量来讲有点特殊,🧢它能够接收从主调方法中传递过来的值因此,当我们说到”参数”这个概念的时候,都是相对于一个”方法”而不是一个”对象”而言的,所以也就不会有”某个对象的参数”这一说法。因此,”this.year”只能被虚拟机认定为当前对象date1自己的year成员变量,绝不会被当作参数。

不知道大家理解💡了吗

 🧢可能现在你还有点模糊,但没关系,我们现在只是刚接触到this这个关键字,先能模仿着用就行,以后可以慢慢理解😂

🧢刚才我们用this.成员变量名访问了我们当前对象的成员变量,那么我们能不能用this.成员方法名去访问我们的方法呢?当然可以呀this.xxx此时的作用不就相当于对象名.xxx吗😎,直接上代码😁

class Cat {
    public void jump() {
        System.out.println("这个猫在跳!!!");
    }
    public void run() {
        this.jump();        // 在run方法中用this.方法名调用当前对象的一个成员方法jump
        System.out.println("这个猫在跑!!!");
    }
}
public class TestDemo2 {
    public static void main(String[] args) {
        Cat cat = new Cat();
        cat.run();
    }
}

🍑 可以看出在我们成功的在run方法中用this.方法名调用了同一个类下的其他方法。

注意🔔:

虽然调用本类的普通方法前可以不使用this关键词。但是建议追加是this,这样的目的是可以区分方法的定义来源🤔


this()用于构造函数的调用

🍑咱们之前说了,同一个类中可以同时有多个构造函数,通过this()还可以在一个构造函数里调用另一个构造函数。

class Cat {
    String name;
    int age;
    Cat() {
        System.out.println("这是自定义的一个不带参数的构造函数");
    }

    public Cat(String name, int age) {
        this();            // 调用另一个不带参数的构造方法
        
        this.name = name;
        this.age = age;
        System.out.println(this.name + this.age + "岁了");
        System.out.println("这是自定义的一个带两个参数的构造函数");
    }
}
public class TestDemo2 {
    public static void main(String[] args) {
        Cat cat = new Cat("小花", 3);
    }
}

 从输出结果可以看出通过💡this(),我们成功在一个构造方法中调用另一个不带参数的构造方法

🔔但有几个问题需要我们注意一下 

  • this( ) 不能在普通方法中使用,只能写在构造方法中。
  • 在构造方法中使用时,必须是第一条语句。

另外,再补充一些小细节📝 

很多小伙伴可能不理解,为什么要通过这种方式来调用构造方法呢?我们在上面调用Cat类中一个不带参数的构造方法时,难道不能直接写一个🍑Cat();来调用吗?

🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟🐟

🔔这里必须做出解释:在Java语言中,一个类的构造方法与类名相同。但是,一个类当中也可以定义一个与类名相同的”普通方法”,💡换句话说就是:并不是只有构造方法与类名相同,”普通方法”也可以取和类相同的名称(只不过全世界的程序员都不会这么干)。

🔔那么,在这种情况下,编译器如何区分这个方法是”普通方法”还是”构造方法”呢?💡很简单,”普通方法”的名称前面必须定义返回值类型,而”构造方法”的名称前面则没有返回值类型的定义。这样,编译器就能够分得清哪个是”构造方法”,哪个是”和类同名的普通方法”。

🔔定义的时候分得清,但是在调用的时候,都是通过方法名来调用的💡,这时如何分得清代码中哪一句调用的是”构造方法”, 哪一句调用的是”和类同名的普通方法”呢?为了解决这个问题,Java语言规定,在本类中调用构造方法的时候,需要通过”🍑this(参数)”的方式来调用

今天有关java中构造方法和this的使用分享就到这里了😊,下一篇博客让我们搞清楚public、static到底是个什么东东 🤔

码字不易😅,各位铁汁们,都看到这里了,点个赞再走好吗😍

有关全面理解java中的构造方法以及this关键字的用法(超详细)的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby - 其他文件中的 Rake 任务 - 2

    我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

  3. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  4. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

  5. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  6. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  7. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  8. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  9. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  10. ruby - rspec 需要 .rspec 文件中的 spec_helper - 2

    我注意到像bundler这样的项目在每个specfile中执行requirespec_helper我还注意到rspec使用选项--require,它允许您在引导rspec时要求一个文件。您还可以将其添加到.rspec文件中,因此只要您运行不带参数的rspec就会添加它。使用上述方法有什么缺点可以解释为什么像bundler这样的项目选择在每个规范文件中都需要spec_helper吗? 最佳答案 我不在Bundler上工作,所以我不能直接谈论他们的做法。并非所有项目都checkin.rspec文件。原因是这个文件,通常按照当前的惯例,只

随机推荐