第三阶段 JAVA常见对象的学习

第一章 常见对象——Object类

引言:

在讲解Object类之前,我们不得不简单的提一下什么是API,先贴一组百度百科的解释:

API(Application Programming Interface,应用程序编程接口)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

简单的说:就是 Java 中有好多现成的类库,其中封装了许多函数,只提供函数名和参数,但隐藏了函数的具体实现,这些可见的部分作为与外界联系的桥梁,也就是我们所称的 API ,不过由于Java是开源的,所以这些隐藏的实现我们也是可以看到的。

(一) Object 类的概述

(1) Object是类层次结构的根类,所有的类都隐式的(不用写extends)继承自Object类。

(2) Java 所有的对象都拥有Object默认方法

(3) Object类的构造方法有一个,并且是无参构造

这就对应了前面学习中的一句话,子类构造方法默认访问父类的构造是无参构造

我们需要了解的方法又有哪些呢?

A: hashCode() B: getClass() C: finalize() D: clone

E: notify() F: notifyAll()

我们需要掌握的方法又有哪些呢?

A: toString() B: equals()

(1) hashCode

返回对象的哈希值(散列码),不是实际地址值,不过可以理解为地址值。

它实际上返回一个int型整数,用于确定该对象在哈希表中的索引位置

暂时了解即可,学习集合框架内容后将会专篇深入学习

//Student类

public class Student extends Object {
}
//StudentDemo类

public class StudentDemo {
public static void main(String[] args) {
Student s1 = new Student();
System.out.println(s1.hashCode()); Student s2 = new Student();
System.out.println(s2.hashCode()); Student s3 = s1;
System.out.println(s3.hashCode());
}
} //运行结果:
460141958
1163157884
460141958

(2) getClass

返回对象的字节码文件对象,在反射篇章详细解释, 暂做简单了解。

public class StudentDemo {
public static void main(String[] args) {
Student s = new Student();
Class c = s.getClass();
String str = c.getName();
System.out.println(str); //链式编程
String str2 = s.getClass().getName();
System.out.println(str2);
}
} //运行结果
cn.bwh_02_getClass.Student
cn.bwh_02_getClass.Student

(3) finalize()

在对象将被垃圾回收器清除前调用,但不确定时间,并且对象的finalize()方法只会被调用一次,调用后也不一定马上清除该对象。

(4) clone()

以实现对象的克隆,包括成员变量的数据复制,但是它和两个引用指向同一个对象是有区别的。

我们先来解释一下后半句话

如果我们想要复制一个变量,可以这样做 Eg:

int a = 20;
int b = a;

那么我们想要复制一个对象,是不是也可以这样做呢?

//Student

public class Student {
int age; public int getAge() {
return age;
} public void setAge(int age) {
this.age = age;
}
}
//StudentDemo

public class StudentDemo {
public static void main(String[] args) {
Student s1 = new Student();
s1.setAge(20);
Student s2 = s1;//将引用赋值
System.out.println("学生1年龄:" + s1.getAge());
System.out.println("学生2年龄:" + s2.getAge());
System.out.println("------------------------------"); s2.setAge(25);
System.out.println("学生1年龄:" + s1.getAge());
System.out.println("学生2年龄:" + s2.getAge());
}
} //运行结果
学生1年龄:20
学生2年龄:20
---------------------------
学生1年龄:25
学生2年龄:25

很明显,即使将对象s1赋值给对象s2,但是通过set传值的时候,两者仍然会同时变化,并没有起到克隆(独立)的作用,这是因为赋值时只是将存储在栈中,对对象的引用赋值,因此它们两个的引用指向同一个对象(堆中),所以无论如何赋值,只要堆中的对象属性发生了变化,通过引用显示属性的时候,均是相同的。

用法:
  1. 实现Cloneable接口
  2. 重写clone方法
分类:

浅拷贝: 仅拷贝对象,不拷贝成员变量,仅复制了变量的引用,拷贝前后变量使用同一块内存,内存销毁后,必须重新定义(两者同生共死)

深拷贝: 不仅拷贝对象,也拷贝成员变量(真正意义上的复制, 两者独立无关)

//浅拷贝

public class Person implements Cloneable{
private int age = 20; @Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
//深拷贝

public class Person implements Cloneable {
public int age = 20; @Override
protected Object clone() throws CloneNotSupportedException {
//拷贝对象
Person person = (Person) super.clone();
//拷贝成员变量
person.age = (int) age.clone();
//返回拷贝对象
return person;
}
}

我们来利用浅拷贝解决刚开始那个问题

//Person类,写出get、set方法、重写clone方法

//PersonDemo类

public class PersonDemo {
public static void main(String[] args) throws CloneNotSupportedException {
Person p1 = new Person();
p1.setAge(20);
Person p2 = (Person) p1.clone();
System.out.println("第一个人的年龄:"+ p1.getAge() );
System.out.println("第二个人的年龄:"+ p2.getAge() );
System.out.println("--------------------------"); p2.setAge(25);
System.out.println("第一个人的年龄:"+ p1.getAge() );
System.out.println("第二个人的年龄:"+ p2.getAge() );
}
} 运行结果:
第一个人的年龄:20
第二个人的年龄:20
--------------------------
第一个人的年龄:20
第二个人的年龄:25

(5) wait、notify和notifyAll

三者属于线程通信间的Api,此部分放在日后讲

(6) toString()——重要

public static toString(): 返回该对象的字符串表示

//Student类

public class Student {
private String name;
public int age; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public int getAge() {
return age;
} public void setAge(int age) {
this.age = age;
}
}
//StudentDemo类

package cn.bwh_04_toString;

public class StudentDemo {
public static void main(String[] args) {
Student s = new Student();
s.setName("admin");
s.setAge(20);
//直接输出s也会默认的调用toString方法
System.out.println(s.toString());
}
} //通过set方法赋值后,直接调用toString() 运行结果:
cn.bwh_04_toString.Student@1b6d3586

很明显,给我们返回这样的信息意义是不大的,所以我们建议对所有子类重写该方法

//在Student类中重写 toString()

	@Override
public String toString() {
return "Student[" + "name=" + name + ", " + "age=" + age + "]"; //运行结果:
Student[name=admin, age=20]

通过重写toString后,结果按照我们所定的规则以字符串的形式输出

(重写后会优先使用类中的toString方法)

为什么要用它呢?

主要目的还是为了简化输出

  1. 在类中重写toString()后,输出类对象就变得有了意义(输出s 和 s.toString()是一样的 ,不写也会默认调用),变成了我们实实在在的信息,而不是上面的cn.bwh_04_toString.Student@1b6d3586。

  2. 如果我们想要多次输出 类中的成员信息,就需要多次书写get方法(每用一次就得写)

    System.out.println("Student[" + "name=" + s.getName() + ", " + "age=" + s.getAge() + "]");

    而调用toString()就简单多了

补充:
//两者等价
toString();
getClass().getName()+ '@' + Integer.toHexString(hashCode()) //输出结果
cn.bwh_04_toString.Student@1b6d3586。

(7) equals()——重要

比较两个对象是否相同

默认情况下,比较的是地址值是否相同。

而比较地址值是没有意义的,所以,一般子类也会重写该方法。

在诸多子类,如String、Integer、Date 等均重写了equals()方法

改进思路:

我们可以将比较地址值转变为比较成员变量

  1. 因为name为String类型,而String类型为引用类型,所以不能够用==比较,应该用equal()

  2. String中默认重写过的equal()方法是用来比较字符串内容是否相同

  3. 我们要使用的是学生类的成员变量,所以父类 Object不能调用子类Student的特有功能

所以使用向下转型

    //重写v1.0
public boolean equals(Object o) {
Student s = (Student) o;
if (this.name.equals(s.name) && this.age == s.age) {
return true;
} else {
return false;
}
}
    //重写v2.0 (可作为最终版)
public boolean equals(Object o) {
if (this.name == o) {
return true;
}
//测试它左边的对象是否是它右边的类的实例,返回 boolean 的数据类型。
if (!(o instanceof Student)) {
return false;
}
Student s = (Student) o;
return this.name.equals(s.name) && this.age == s.age;
}
   //idea自动生成版
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Student student = (Student) o;
return age == student.age &&
Objects.equals(name, student.name);
}
equals() 和 == 的区别

== 的作用:   

基本类型:比较值是否相同   

引用类型:比较的就是堆内存地址是否相同

equals 的作用:

引用类型:默认情况下,比较的是地址值。

注:一般选择重写方法,比较对象的成员变量值是否相同 ,不过一般重写都是自动生成。

结尾:

如果内容中有什么不足,或者错误的地方,欢迎大家给我留言提出意见, 蟹蟹大家 !_

如果能帮到你的话,那就来关注我吧!(系列文章均会在公众号第一时间更新)

在这里的我们素不相识,却都在为了自己的梦而努力 ❤

一个坚持推送原创Java技术的公众号:理想二旬不止

Object类入门这一篇就够了!的更多相关文章

  1. React入门看这篇就够了

    摘要: 很多值得了解的细节. 原文:React入门看这篇就够了 作者:Random Fundebug经授权转载,版权归原作者所有. React 背景介绍 React 入门实例教程 React 起源于 ...

  2. [转]React入门看这篇就够了

    摘要: 很多值得了解的细节. 原文:React入门看这篇就够了 作者:Random Fundebug经授权转载,版权归原作者所有. React 背景介绍 React 入门实例教程 React 起源于 ...

  3. [转帖]Zookeeper入门看这篇就够了

    Zookeeper入门看这篇就够了 https://my.oschina.net/u/3796575/blog/1845035 Zookeeper是什么 官方文档上这么解释zookeeper,它是一个 ...

  4. .NET Core实战项目之CMS 第五章 入门篇-Dapper的快速入门看这篇就够了

    写在前面 上篇文章我们讲了如在在实际项目开发中使用Git来进行代码的版本控制,当然介绍的都是比较常用的功能.今天我再带着大家一起熟悉下一个ORM框架Dapper,实例代码的演示编写完成后我会通过Git ...

  5. Git入门仅这篇就够了

    版权声明:本文为博主原创文章,未经博主允许不得转载. 转载请表明出处:http://www.cnblogs.com/cavalier-/p/5978937.html 前言 大家好,我是Cavalier ...

  6. Struts2入门这一篇就够了

    前言 这是Strtus的开山篇,主要是引入struts框架...为什么要引入struts,引入struts的好处是什么,以及对Struts2一个简单的入门.... 为什么要引入struts? 既然Se ...

  7. Hibernate入门这一篇就够了

    前言 本博文主要讲解介绍Hibernate框架,ORM的概念和Hibernate入门,相信你们看了就会使用Hibernate了! 什么是Hibernate框架? Hibernate是一种ORM框架,全 ...

  8. 《ElasticSearch入门》一篇管够,持续更新

    一.顾名思义: Elastic:灵活的:Search:搜索引擎 二.官方简介: Elasticsearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTf ...

  9. c语言入门这一篇就够了-学习笔记(一万字)

    内容来自慕课网,个人学习笔记.加上了mtianyan标签标记知识点. C语言入门 -> Linux C语言编程基本原理与实践 -> Linux C语言指针与内存 -> Linux C ...

随机推荐

  1. jQuery.extend([deep], target, object1, [objectN])

    jQuery.extend([deep], target, object1, [objectN]) 概述 用一个或多个其他对象来扩展一个对象,返回被扩展的对象.直线电机 如果不指定target,则给j ...

  2. Lock的lock/unlock, condition的await/singal 和 Object的wait/notify 的区别

    在使用Lock之前,我们都使用Object 的wait和notify实现同步的.举例来说,一个producer和consumer,consumer发现没有东西了,等待,produer生成东西了,唤醒. ...

  3. Kafka 消息队列系列之分布式消息队列Kafka

    介绍 ApacheKafka®是一个分布式流媒体平台.这到底是什么意思呢?我们认为流媒体平台具有三个关键功能:它可以让你发布和订阅记录流.在这方面,它类似于消​​息队列或企业消息传递系统.它允许您以容 ...

  4. 数据结构实验之数组一:矩阵转置(SDUT 2130)

    Problem Description 数组--矩阵的转置 给定一个m*n的矩阵(m,n<=100),求该矩阵的转置矩阵并输出. Input 输入包含多组测试数据,每组测试数据格式如下: 第一行 ...

  5. Python3操作YAML文件

    数据及配置文件之争 数据及文件通常有三种类型: 配置文件型:如ini,conf,properties文件,适合存储简单变量和配置项,最多支持两层,不适合存储多层嵌套数据 表格矩阵型:如csv,exce ...

  6. js-常见算法操作

    字符串操作 判断回文字符串 1 2 3 4 5 6 7 8 9 10 11 12 13 14 function palindrome(str){  // \W匹配任何非单词字符.等价于“[^A-Za- ...

  7. 20175313 张黎仙《Java程序设计》第十一周学习总结

    目录 一.教材学习内容总结 二.教材学习中的问题和解决过程 三.代码托管 四.心得体会 五.学习进度条 六.参考资料 一.教材学习内容总结 第十三章内容 主要内容 URL类 InetAdress类 套 ...

  8. postgresql interval 字段拼接

    无拼接时: SELECT scan_time + '5 day' FROM tbl_temp_record SELECT scan_time + '-5 day' FROM tbl_temp_reco ...

  9. spring项目启动错误——java.lang.NoClassDefFoundError: org/springframework/context/ApplicationContext

    最近在搭spring项目框架的时候,遇到一个很伤的问题,翻了很多帖,都报告说什么少spring-context包啊之类的,但实际上spring的那些依赖我根本没漏,下面是我的pom: <depe ...

  10. Vue UI组件库

    1. iView UI组件库  iView官网:https://www.iviewui.com/ 2.Vux UI组件库   Vux官网:https://vux.li/ 3.Element UI组件库 ...