作者:小牛呼噜噜 | https://xiaoniuhululu.com

计算机内功、JAVA底层、面试相关资料等更多精彩文章在公众号「小牛呼噜噜 」

Java对象究竟是什么?

对象:对象是类的一个实例,有状态和行为。

类:类是一个模板,它描述一类对象的行为和状态。

例如

人 是一个类

其状态有:姓名、性别、身高、体重等

其行为:吃饭、睡觉、聊天、运动等

    public class Person {
/**
* 状态 or 属性
*/
String name;//姓名
String sex;//性别
int height;//身高
int weight;//体重 /**
* 行为
*/
public void sleep(){
System.out.println(this.name+"--"+ "睡觉");
}
public void eat(){
System.out.println("吃饭");
}
public void Dance(){
System.out.println("跳舞");
}
}

对象就是指具体的哪个人,比如"小张" 就是对象,可以通过new 来创建出来

public static void main(String[] args) {
Person zhang = new Person();
zhang.name = "小张";
zhang.sex ="男";
zhang.height=180;
zhang.weight=150;
}

通过上面的例子,我们可以发现 面向对象提出一种计算机世界里解决复杂软件工程的方法论,拆解问题复杂度,从人类思维角度提出解决问题的步骤和方案。

因为面向过程让计算机有步骤地顺次做一件事情,是一种过程化的叙事思维,简单明了。但是随着软件项目越来越庞大的时候,发现用面向过程语言开发,软件维护、软件复用存在着巨大的困难。

创建对象的过程

一般来说,我们创建对象 可以通过new来 创建一个,比如从上面的例子中这一句:

 Person zhang = new Person();

虽然我们写的时候是简单的一句,但是JVM内部的实现过程却是复杂的:

  1. 将硬盘上指定位置的Person.class文件加载进内存
  2. 执行main方法时,在栈内存中开辟了main方法的空间(压栈-进栈),然后在main方法的栈区分配了一个变量zhang。
  3. 执行new,在堆内存中开辟一个 实体类的 空间,分配了一个内存首地址值
  4. 调用该实体类对应的构造函数,进行初始化(如果没有构造函数,Java会补上一个默认构造函数)。
  5. 将实体类的 首地址赋值给zhang,变量zhang就引用了该实体。(指向了该对象)

创建多个对象时,内存的变化

当我们new 多个对象时,属性会另外开辟堆空间存放,而方法只有一份,不会额外消耗内存

我们接着来看一个例子:

public static void main(String[] args) {
Person ming = new Person();
ming.name = "ming";
ming.sleep(); Person wang = new Person();
wang.name = "wang";
wang.sleep();
}

运行结果:

ming--睡觉

wang--睡觉

对象ming的 属性在堆内存,方法在方法区。当我们在通过Person类来 新增一个wang对象时,栈内存会有一个对象名称wang,来指向在堆内存中 新创建的另一个Person对象,属性存放在堆内存中。我们可以看出对象ming和对象wang 属性 2者互不影响,相互独立。

但是 对象ming和对象wang的方法区 是共用的。 那为何2者属性输出结果不一样呢?

其实 方法就像一套指令模板,谁都可以传入数据交给它执行,然后得到对应执行结果

但是 JVM是如何确保 ming.sleep(); 返回的结果是 小明在睡觉 而不是 小王在睡觉 或者其他情况?

Java的this其实就是解决这个问题的,接下来我慢慢道来。

无处不在的this和super关键字

this 表示当前对象的引用,可以理解为指向对象本身的一个"指针",但是JAVA中是没有指针这个概念的。

我们知道在C/C++中,指针是指向内存中的地址,该地址就是存储变量的值。该地址所存储的变量值是"公有"的,此处的"公有"是对于拥有该地址的变量而言。它们随时都可以访问该地址的内容,并且可对其进行修改,一经修改则所有指向该地址的变量值也将改变。

c++中也有结构体、对象的概念,但是为什么他们不像java一样有"封装"的概念?

因为在c、c++中指针很强大,可以通过指针直接访问操作内存中的数据。而java没有指针,这样封装就能极大地提升安全性

虽然java中没有指针的概念,但this("指针")无处不在.

从上面的例子 我们可以看出

        public void sleep(){
System.out.println(this.name+"--"+ "睡觉");
}

ming.sleep()和wang.sleep()语句调用的代码是方法区同一个内存,但是在JVM运行过程中,可以根据由哪个对象发起对sleep()的调用,方法中所用到的成员变量数据就使用哪个对象的数据。这个本质就像是方法传参一样,隐式传递this

this表示当前对象的引用:
this.属性 区别成员变量和局部变量
this.() 调用本类的某个方法
this() 表示调用本类构造方法,只能用在构造方法的第一行语句。
this关键字只能出现在非static修饰的代码中

我们来看一个例子:

public class Main {
public static void main(String[] args) {
Nanjing nanjing = new Nanjing(); Beijing beijing = new Beijing(); }
}
public class Country {
String name;
public Country(){ //构造器
System.out.println(this.getClass().getName());
}
} public class Beijing extends Country{
} public class Nanjing extends Country{
}

结果:

com.company.Nanjing

com.company.Beijing

子类Nanjing和Beijing 啥都没干,但是却通过父类Country的构造器,得到子类的名字。

当程序执行new Nanjing()语句去实例化子类时,它会去隐式调用父类的构造器,等同于:

public class Nanjing extends Country{
public Nanjing() {
super();//显式 调用父类的构造器
}
}

这一过程中,会去隐式传递this,不然各个子类的名称 不会显示

我们再来改造一下Nanjing类的代码:

public class Nanjing extends Country{
public Nanjing() {
System.out.println("nanjing 自定义构造器");
}
}

结果:

com.company.Nanjing

nanjing 自定义构造器

我们可以看出: 如果 子类Nanjing自定义构造器,会优先调用父类的构造器,再调用自己的构造器

我们接着来看下 super关键字

super 表示自己超(父)类对象的引用,可以理解为是指向自己超(父)类对象的一个指针,而这个超类指的是离自己最近的一个父类

super表示父类对象:
super.属性 表示父类对象中的成员变量
super.方法()表示父类对象中定义的方法
super() 表示调用父类构造方法
可以指定参数,比如super("Nanjin");
任何一个构造方法的第一行默认是super();
可以写上,如果未写,会隐式调用super();
super()只能在构造方法的第一行使用。
this()和super()都只能在构造的第一行出现,所以只能选择其一。
写了this()就不会隐式调用super()。
super 关键字在子类中显式调用父类中被覆盖的非静态成员方法和成员变量

我们来看一个super调用父类方法的例子:

class Father {

  void message() {
System.out.println("This is Father");
} } class Son extends Father {
void message() {
System.out.println("This is son");
} void display() {
message();
super.message();
} } class Main {
public static void main(String args[]) {
Son s = new Son();
s.display();
} }

结果:

This is son

This is father

可以看出 super和this功能差不多,主要区别:this 指向当前对象,super指向 离自己最近的一个父类,就不展开深入说了。

static关键字 为何如此特殊

Java中static`关键字主要用于内存管理, 可以用来修饰变量或者方法。

由于JAVA面向对象处处可见,在面向对象的思维下,方法与对象存在一种强耦合,简单点来说就是 方法在没有对象的情况下无法调用。

static关键字就是被设计来解决这个问题的。

我们来看一个例子:

public class Country {
String name = "china"; public void show() {
System.out.println(name);
} } public static void main(String[] args) {
Country c1 = new Country();
Country c2 = new Country();
c1.name = "china new";
c2.show(); }

结果:

china

如果用static修饰呢:

public class Country {
static String name = "china"; public void show() {
System.out.println(name);
} }

结果:

china new

我们可以看出:

如果给一个属性加上static,那么这个属性不再属于某一个对象了,而是是属于类的,是所有对象共享的,共用同一个static属性

可以通过类对象名.变量名 的方式访问,比如: Country.name

当程序进行类加载时,静态方法随着类加载而加载进JVM中,此时并没有对象实例化,优先于对象的创建。static属性在一个单独的内存区,而不是在new出的对象内存中

另外一般来说 静态方法不能访问实例变量,其实是由于Java不会在调用静态方法时传递this,没有this就没法处理差异化数据。

非static方法可以调用static方法,但static方法不能调用非static方法

尾语

笔者把Java中对象和类、this、super和static关键字都串起来,简单聊聊这些背后设计的原理,希望对大家有所帮助


很感谢你能看到最后,如果喜欢的话,欢迎关注点赞收藏转发,谢谢!更多精彩的文章

你真的了解JAVA中对象和类、this、super和static关键字吗的更多相关文章

  1. java中普通的顶级类是不能使用static关键字修饰的。只有内部类可以使用static修饰,也可以不使用staitc关键字修饰。

    java中普通的顶级类是不能使用static关键字修饰的.只有内部类可以使用static修饰,也可以不使用staitc关键字修饰. java中的类可以是static吗?答案是可以.在java中我们可以 ...

  2. Java中的实体类--Serializable接口、transient 关键字

    在java中,实体类是一个非常重要的概念,我们可以在实体类中封装对象.设置其属性和方法等.关于实体类,也经常涉及到适配器模式.装饰者模式等设计模式.那么在实际代码开发中,关于实体类的注意事项有哪些呢? ...

  3. Java常见对象Object类中的个别方法

    Java常见对象Object类 public int hashCode() : 返回该对象的哈希码值. 注意:哈希值是根据哈希算法计算出来的一个值,这个值和地址值有关,但是不是实际地址值.你可以理解成 ...

  4. Java中对象的生与灭- 核心篇

    前言 大家好啊,我是汤圆,今天给大家带来的是<Java中对象的生与灭- 核心篇>,希望对大家有帮助,谢谢 文章纯属原创,个人总结难免有差错,如果有,麻烦在评论区回复或后台私信,谢啦 简介 ...

  5. java中对象产生初始化过程

    以前面试的时候,很多公司的笔试题中有关new一个对象有关一系列初始化的过程的选择题目.请看下面的题目. class Parent { static { System.out.println(" ...

  6. Java基础(43):Java中的Object类与其方法(转)

    Object类 java.lang.Object java.lang包在使用的时候无需显示导入,编译时由编译器自动导入. Object类是类层次结构的根,Java中所有的类从根本上都继承自这个类. O ...

  7. Java中对象的深复制和浅复制详解

    1.浅复制与深复制概念 ⑴浅复制(浅克隆) 被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象. ⑵ ...

  8. Java中的Unsafe类111

    1.Unsafe类介绍 Unsafe类是在sun.misc包下,不属于Java标准.但是很多Java的基础类库,包括一些被广泛使用的高性能开发库都是基于Unsafe类开发的,比如Netty.Hadoo ...

  9. Java基础-对象与类

    面向对象程序设计概述 面向对象的程序设计(简称OOP)时当今主流的程序设计范型,已经取代了"结构化"过程化程序设计开发技术,Java是完全面向对象的. 类 类设计构造对象的模板或蓝 ...

随机推荐

  1. v-show与v-if的一次事故

    v-show等同于设置dom元素的display为none,dom元素没有消失而是被隐藏了 v-if是删除或添加dom元素,频繁地删除和添加dom元素会比较耗费性能

  2. 自己在ubuntu16.04 上用的软件和配置

    软件: 1.WPS2019: 这个不用多说了,真的是比之前的wps好太多了. 2.Chrom的画图插件: http://Draw.io,非常强,Draw.io 是一款在线图表编辑工具, 可以用来编辑工 ...

  3. 异步编程利器:CompletableFuture

    一.一个示例回顾Future 一些业务场景我们需要使用多线程异步执行任务,加快任务执行速度. JDK5新增了Future接口,用于描述一个异步计算的结果.虽然 Future 以及相关使用方法提供了异步 ...

  4. axios源码解析 - 请求拦截器

    axios请求拦截器,也就是在请求发送之前执行自定义的函数. axios源码版本 - ^0.27.2 (源码是精简版) 平时在业务中会这样去写请求拦截器,代码如下: // 创建一个新的实例 var s ...

  5. 什么叫做 SSO

    什么叫做 SSO 本文写于 2020 年 12 月 8 日 SSO 的全称叫做 Single Sign On,意味「单点登录」. 何为单点登录?就是你希望自己的两个网站,可以做到:一个网站登录了,另一 ...

  6. Swift初探01 变量与控制流

    Swift初探01 变量与控制流 输出"hello world"是几乎学习所有编程语言的第一课,这是程序员的情怀. 所以我们学习swift的第一步,就是输出一句"Hell ...

  7. 个人冲刺(五)——体温上报app(一阶段)

    任务:完成了体温录入.体温记录删除.体温修改以及历史记录查询操作 体温录入 public void insertDB(View view) { MyDBHelper mydbh=new MyDBHel ...

  8. CF1485E Move and Swap

    题意:Move and Swap 很好的题呢 n个节点的树,根为1,所有叶子的深度都是D,一开始根节点上有两个颜色分别微R,B的球,你执行下列操作D-1次: 1.R点跳到子树内 2.B点跳到下一层的任 ...

  9. Spring Ioc源码分析系列--Bean实例化过程(二)

    Spring Ioc源码分析系列--Bean实例化过程(二) 前言 上篇文章Spring Ioc源码分析系列--Bean实例化过程(一)简单分析了getBean()方法,还记得分析了什么吗?不记得了才 ...

  10. 钉钉登录二维码嵌套在vue页面中

    转自 https://www.csdn.net/tags/OtDacg3sMjQ2NTgtYmxvZwO0O0OO0O0O.html 钉钉登录二维码嵌套在vue页面中 2021-09-04 14:42 ...