很考验人的java内存加载面试题
源代码如下,求结果
public class MemoryAnalyse {
public static int k = 0;
public static MemoryAnalyse t1 = new MemoryAnalyse("t1");
public static MemoryAnalyse t2 = new MemoryAnalyse("t2");
public static int i = print("i");
public static int j = print("j");
public static int n = 99;
{
print("constructor code");
}
static {
print("static code");
}
public static int print(String s) {
System.out.println("i=" + i + " " + s + " k=" + k + " n=" + n
+ " j=" + j);
++i;
++k;
++n;
return i;
}
public MemoryAnalyse(String string) {
print(string);
}
public static void main(String[] args) throws ClassNotFoundException {
MemoryAnalyse d = new MemoryAnalyse("T");
}
}
然而结果是这个
i=0 constructor code k=0 n=0 j=0
i=1 t1 k=1 n=1 j=0
i=2 constructor code k=2 n=2 j=0
i=3 t2 k=3 n=3 j=0
i=4 i k=4 n=4 j=0
i=5 j k=5 n=5 j=0
i=6 static code k=6 n=99 j=6
i=7 constructor code k=7 n=100 j=6
i=8 T k=8 n=101 j=6
有没有很惊讶,结果竟然这么复杂.好,下面我们分析一下,在分析之前,先普及下不怎么用的基础知识
代码块和静态代码块何时运行问题:
代码块在创建对象时运行
静态代码块在类加载时运行
大家都知道static是属于类的并非对象,也就是说static修饰的东西都会在class加载到方法区时就存在在那里.所以方法区中类加载时内存过程如下
1.当类刚加载时会全部加载到方法区时,此时所有变量全部未实例化.

2.实例化参数t1

此时因为代码块在创建对象时执行,且在构造函数之前执行,所以先执行代码块
{
print("constructor code");
}
因为此时所有的变量都为默认值,所以执行后打印结果为
i=0 constructor code k=0 n=0 j=0
此时i,n,k的值都已经自加一,值为1
然后实例化调用构造函数
public MemoryAnalyse(String string) { //这里string为t1
print(string);
}
构造函数调用结果如下
i=1 t1 k=1 n=1 j=0
此时i,n,k的值都已经自加一,值为2
3.实例化参数t2

和第一步一样在构造函数之前执行代码块
{
print("constructor code");
}
i=2 constructor code k=2 n=2 j=0
此时i,n,k的值都已经自加一,值为3
然后实例化调用构造函数
public MemoryAnalyse(String string) { //这里string为t2
print(string);
}
构造函数调用结果如下
i=3 t2 k=3 n=3 j=0
此时i,n,k的值都已经自加一,值为4
4.初始化参数i

这里直接调用print("i")函数,得到结果为
i=4 i k=4 n=4 j=0
此时i,k,j值为5,注意i的值不是通过自加一变成5的,而是通过函数的返回值赋给i的
5.初始化参数j

这里和上一步一样,执行print("j"),然后把函数的返回值赋给j,打印结果为
i=5 j k=5 n=5 j=0
此时j的值已经为6
到这里类加载的内部参数变化就完成了,我们可以用加载类的方式调用一下
public static void main(String[] args) throws ClassNotFoundException {
//MemoryAnalyse d = new MemoryAnalyse("T");
Class.forName("MemoryAnalyse");
}
Class.forName(类名字符串)是手动加载类到方法区,得到结果为
i=0 constructor code k=0 n=0 j=0
i=1 t1 k=1 n=1 j=0
i=2 constructor code k=2 n=2 j=0
i=3 t2 k=3 n=3 j=0
i=4 i k=4 n=4 j=0
i=5 j k=5 n=5 j=0
i=6 static code k=6 n=99 j=6 //这行的出现是因为static代码段在类加载时执行.n=99是因为类加载了,n的初值为99把之前的值覆盖掉了
然后改为我们之前的demo
public static void main(String[] args) throws ClassNotFoundException {
MemoryAnalyse d = new MemoryAnalyse("T");
//Class.forName("MemoryAnalyse");
}
执行结果为
i=0 constructor code k=0 n=0 j=0 //t1代码块执行
i=1 t1 k=1 n=1 j=0 //t1构造函数执行
i=2 constructor code k=2 n=2 j=0 //t2代码块执行
i=3 t2 k=3 n=3 j=0 //t2构造函数执行
i=4 i k=4 n=4 j=0
i=5 j k=5 n=5 j=0
i=6 static code k=6 n=99 j=6 //静态代码块执行
i=7 constructor code k=7 n=100 j=6 //T代码块执行
i=8 T k=8 n=101 j=6 //T构造函数执行
最后两行的出现就很简单了,一个是代码块的,一个是构造函数的
这么一分析是不是简单了很多
很考验人的java内存加载面试题的更多相关文章
- 看起来很懵的java内存加载面试题
源代码如下,求结果 public class MemoryAnalyse { public static int k = 0; public static MemoryAnalyse t1 = new ...
- java内存加载机制
什么是java类加载? 类加载是指将.class类中的二进制数据存放到内存中,会在内存中的推中建立一个java.lang.String的引用对象来存放方法区的数据结构,而类中的数据会放到方法区中 类加 ...
- 【设计模式】C++单例模式的几种写法——Java自动加载内部类对象,C++怎么破?
单例模式是最简单的设计模式,就让我像玩简单的游戏一样写下去吧. v1: 简单模式 和这个版本有过一面之缘,但不敢苟同. class Singleton { private: Singleton() { ...
- java class加载机制及对象生成机制
java class加载机制及对象生成机制 当使用到某个类,但该类还未初始化,未加载到内存中时会经历类加载.链接.初始化三个步骤完成类的初始化.需要注意的是类的初始化和链接的顺序有可能是互换的. Cl ...
- 内存加载DLL
1.前言 目前很多敏感和重要的DLL(Dynamic-link library) 都没有提供静态版本供编译器进行静态连接(.lib文件),即使提供了静态版本也因为兼容性问题导致无法使用,而只提供DLL ...
- Java虚拟机加载类
Java虚拟机是如何加载Java类的 Java虚拟机加载Java类总共需要经过3步:加载-----链接-----初始化.Java语言的类型可以分为两大类:基本类型和引用类型.基本类型是有Java虚拟机 ...
- 原来热加载如此简单,手动写一个 Java 热加载吧
1. 什么是热加载 热加载是指可以在不重启服务的情况下让更改的代码生效,热加载可以显著的提升开发以及调试的效率,它是基于 Java 的类加载器实现的,但是由于热加载的不安全性,一般不会用于正式的生产环 ...
- java用户界面——加载图片 jpg GIF
java用户界面--加载图片 jpg GIF 代码如下: package day08; import java.awt.GridLayout; import javax.swing.Icon;impo ...
- java动态加载配置文件(申明:来源于网络)
java动态加载配置文件 地址:http://blog.csdn.net/longvs/article/details/9361449
随机推荐
- Java面向对象 Object类 内部类
Java面向对象 Object类 内部类 知识概要: 一:Object类 二:内部类 匿名内部类的写法 1.Object O ...
- Win10系统下安装Ubuntu16.04.3教程与设置
在Win10上刚刚装好Ubuntu16.04.3,装了不下于10次,期间出现很多问题,趁着还有记忆,写下这篇教程,里面还有Ubuntu系统的优化与Win10的一些设置. Part 1 制作Ubuntu ...
- Javascript 面向对象编程—继承和封装
前 言 Javascript是一种基于对象(object-based)的语言,你遇到的所有东西几乎都是对象.但是,它又不是一种真正的面向对象编程(OOP)语言,因为它的语法中没有class(类) ...
- Codeforces Round #441 (Div. 2, by Moscow Team Olympiad) D. Sorting the Coins
http://codeforces.com/contest/876/problem/D 题意: 最开始有一串全部由"O"组成的字符串,现在给出n个数字,指的是每次把位置n上的&qu ...
- SAP 动态设置 GUI STATUS 灰色不可用 或者隐藏(转)
http://blog.sina.com.cn/s/blog_66110f6201017rul.html 有时候需要根据用户的权限或者是操作动态设置gui状态上的某些按钮的可用和不可用. 1.先定 ...
- win10 uwp App-to-app communication 应用通信
这篇文章都是乱说的,如果觉得有不好的,可以发我邮箱 应用之间需要相互的发送信息,就是我们经常用的分享 有个人看到一个网页很好,于是就希望把这个网页发送到邮件,那么这样的话就是使用应用通信. 因为每个应 ...
- Oracle参数设置之set与reset的实际案例
Oracle参数设置之set与reset的实际案例 环境:Oracle 10.2.0.5 RAC 需求:节点1的aq_tm_processes要求恢复默认,节点2设置要求保持不变 1.构建测试环境 2 ...
- 容器与Docker简介(四)Docker容器,镜像与 Registries——微软微服务电子书翻译系列
当使用Docker时,开发人员创建一个应用程序或服务,并将其和其依赖关系打包到容器镜像中. 镜像是应用程序或服务及其配置和依赖的静态表示形式. 要运行应用程序或服务,应用程序的镜像将被实例化以创建一个 ...
- 学习一门新语言需要了解的基础-12 if和switch对比
本节内容 是否存在性能差异 使用场景 反汇编对比[付费阅读] 之前初步接触了汇编,然后利用汇编简单了解下函数调用的过程,包括怎么样保护堆栈帧现场和恢复现场.另外做了简单的函数调用参数复制,返回值的传递 ...
- 编译错误:expected an indented block
python 对缩进要求非常严格,当运行时出现以下几种情况时,就需要修改缩进: 1.expected an indented block 直接缩进4个space空格或者一个tab.