- 浏览: 210354 次
- 性别:
- 来自: 苏州
文章分类
- 全部博客 (163)
- java综合 (24)
- javaweb (6)
- javascript (14)
- web前端 (3)
- Struts1 (0)
- Struts2 (2)
- hibenate (4)
- spring (12)
- 软件工程 (2)
- 网站建设 (2)
- 数据库 (9)
- FusionCharts (5)
- EXTJS (4)
- Spring MVC (2)
- html (1)
- Linux (19)
- 其他 (2)
- SSI (2)
- velocity (3)
- 设计模式 (1)
- 发布方面 (2)
- 算法 (2)
- 生活感悟 (8)
- flex (6)
- 搜索方面 (7)
- 架构 (1)
- IOS开发 (5)
- xml (1)
最新评论
-
ddccjjwwjj:
谢谢你总算找到一个正确的答案!!!
ExtJs 中 combo 获得displayField 和 valueField -
yulanlian:
uld not open ServletContext res ...
web.xml中配置Spring -
jinnianshilongnian:
id能进行SAX解析ID验证,name不行。name在使用如s ...
Spring配置中<bean>的id和name属性区别 -
白色蜻蜓:
不错顶下
s2sh框架搭建需要的jar包
大家在去参加面试的时候,经常会遇到这样的考题:给你两个类的代码,它们之间是继承的关系,每个类里只有构造器方法和一些变量,构造器里可能还有一段代码对变量值进行了某种运算,另外还有一些将变量值输出到控制台的代码,然后让我们判断输出的结果。这实际上是在考查我们对于继承情况下类的初始化顺序的了解。我们大家都知道,对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序以此是(静态变量、静态初始化块)>(变量、初始化块)>构造器。我们也可以通过下面的测试代码来验证这一点: Java代码
public class InitialOrderTest {
// 静态变量
public static String staticField = "静态变量";
// 变量
public String field = "变量";
// 静态初始化块
static {
System.out.println(staticField);
System.out.println("静态初始化块");
}
// 初始化块
{
System.out.println(field);
System.out.println("初始化块");
}
// 构造器
public InitialOrderTest() {
System.out.println("构造器");
}
public static void main(String[] args) {
new InitialOrderTest();
}
}
运行以上代码,我们会得到如下的输出结果:
静态变量
静态初始化块
变量
初始化块
构造器
这与上文中说的完全符合。那么对于继承情况下又会怎样呢?我们仍然以一段测试代码来获取最终结果: Java代码
class Parent {
// 静态变量
public static String p_StaticField = "父类--静态变量";
// 变量
public String p_Field = "父类--变量";
// 静态初始化块
static {
System.out.println(p_StaticField);
System.out.println("父类--静态初始化块");
}
// 初始化块
{
System.out.println(p_Field);
System.out.println("父类--初始化块");
}
// 构造器
public Parent() {
System.out.println("父类--构造器");
}
}
public class SubClass extends Parent {
// 静态变量
public static String s_StaticField = "子类--静态变量";
// 变量
public String s_Field = "子类--变量";
// 静态初始化块
static {
System.out.println(s_StaticField);
System.out.println("子类--静态初始化块");
}
// 初始化块
{
System.out.println(s_Field);
System.out.println("子类--初始化块");
}
// 构造器
public SubClass() {
System.out.println("子类--构造器");
}
// 程序入口
public static void main(String[] args) {
new SubClass();
}
}
运行一下上面的代码,结果马上呈现在我们的眼前:
父类--静态变量
父类--静态初始化块
子类--静态变量
子类--静态初始化块
父类--变量
父类--初始化块
父类--构造器
子类--变量
子类--初始化块
子类--构造器
现在,结果已经不言自明了。大家可能会注意到一点,那就是,并不是父类完全初始化完毕后才进行子类的初始化,实际上子类的静态变量和静态初始化块的初始化是在父类的变量、初始化块和构造器初始化之前就完成了。那么对于静态变量和静态初始化块之间、变量和初始化块之间的先后顺序又是怎样呢?是否静态变量总是先于静态初始化块,变量总是先于初始化块就被初始化了呢?实际上这取决于它们在类中出现的先后顺序。我们以静态变量和静态初始化块为例来进行说明。同样,我们还是写一个类来进行测试: Java代码
public class TestOrder {
// 静态变量
public static TestA a = new TestA();
// 静态初始化块
static {
System.out.println("静态初始化块");
}
// 静态变量
public static TestB b = new TestB();
public static void main(String[] args) {
new TestOrder();
}
}
class TestA {
public TestA() {
System.out.println("Test--A");
}
}
class TestB {
public TestB() {
System.out.println("Test--B");
}
}
运行上面的代码,会得到如下的结果:
Test--A
静态初始化块
Test--B
大家可以随意改变变量a、变量b以及静态初始化块的前后位置,就会发现输出结果随着它们在类中出现的前后顺序而改变,这就说明静态变量和静态初始化块是依照他们在类中的定义顺序进行初始化的。同样,变量和初始化块也遵循这个规律。了解了继承情况下类的初始化顺序之后,如何判断最终输出结果就迎刃而解了。
一个简单的堆栈溢出例子,在构造方法里面new当前类实例。
// 构造器
public InitialOrderTest() {
System.out.println("构造器");
new InitialOrderTest().doit();
}
无限制的调用,堆栈溢出!
出处:http://blog.sina.com.cn/s/blog_4ca269070100dioo.html
public class InitialOrderTest {
// 静态变量
public static String staticField = "静态变量";
// 变量
public String field = "变量";
// 静态初始化块
static {
System.out.println(staticField);
System.out.println("静态初始化块");
}
// 初始化块
{
System.out.println(field);
System.out.println("初始化块");
}
// 构造器
public InitialOrderTest() {
System.out.println("构造器");
}
public static void main(String[] args) {
new InitialOrderTest();
}
}
运行以上代码,我们会得到如下的输出结果:
静态变量
静态初始化块
变量
初始化块
构造器
这与上文中说的完全符合。那么对于继承情况下又会怎样呢?我们仍然以一段测试代码来获取最终结果: Java代码
class Parent {
// 静态变量
public static String p_StaticField = "父类--静态变量";
// 变量
public String p_Field = "父类--变量";
// 静态初始化块
static {
System.out.println(p_StaticField);
System.out.println("父类--静态初始化块");
}
// 初始化块
{
System.out.println(p_Field);
System.out.println("父类--初始化块");
}
// 构造器
public Parent() {
System.out.println("父类--构造器");
}
}
public class SubClass extends Parent {
// 静态变量
public static String s_StaticField = "子类--静态变量";
// 变量
public String s_Field = "子类--变量";
// 静态初始化块
static {
System.out.println(s_StaticField);
System.out.println("子类--静态初始化块");
}
// 初始化块
{
System.out.println(s_Field);
System.out.println("子类--初始化块");
}
// 构造器
public SubClass() {
System.out.println("子类--构造器");
}
// 程序入口
public static void main(String[] args) {
new SubClass();
}
}
运行一下上面的代码,结果马上呈现在我们的眼前:
父类--静态变量
父类--静态初始化块
子类--静态变量
子类--静态初始化块
父类--变量
父类--初始化块
父类--构造器
子类--变量
子类--初始化块
子类--构造器
现在,结果已经不言自明了。大家可能会注意到一点,那就是,并不是父类完全初始化完毕后才进行子类的初始化,实际上子类的静态变量和静态初始化块的初始化是在父类的变量、初始化块和构造器初始化之前就完成了。那么对于静态变量和静态初始化块之间、变量和初始化块之间的先后顺序又是怎样呢?是否静态变量总是先于静态初始化块,变量总是先于初始化块就被初始化了呢?实际上这取决于它们在类中出现的先后顺序。我们以静态变量和静态初始化块为例来进行说明。同样,我们还是写一个类来进行测试: Java代码
public class TestOrder {
// 静态变量
public static TestA a = new TestA();
// 静态初始化块
static {
System.out.println("静态初始化块");
}
// 静态变量
public static TestB b = new TestB();
public static void main(String[] args) {
new TestOrder();
}
}
class TestA {
public TestA() {
System.out.println("Test--A");
}
}
class TestB {
public TestB() {
System.out.println("Test--B");
}
}
运行上面的代码,会得到如下的结果:
Test--A
静态初始化块
Test--B
大家可以随意改变变量a、变量b以及静态初始化块的前后位置,就会发现输出结果随着它们在类中出现的前后顺序而改变,这就说明静态变量和静态初始化块是依照他们在类中的定义顺序进行初始化的。同样,变量和初始化块也遵循这个规律。了解了继承情况下类的初始化顺序之后,如何判断最终输出结果就迎刃而解了。
一个简单的堆栈溢出例子,在构造方法里面new当前类实例。
// 构造器
public InitialOrderTest() {
System.out.println("构造器");
new InitialOrderTest().doit();
}
无限制的调用,堆栈溢出!
出处:http://blog.sina.com.cn/s/blog_4ca269070100dioo.html
发表评论
-
Java RandomAccessFile用法
2012-08-11 19:00 1424http://blog.csdn.net/akon_vm/ar ... -
Java 线程池的原理与实现
2012-06-08 14:58 752http://hi.baidu.com/obullxl/blo ... -
Java 线程池学习
2012-06-08 14:57 743http://www.cnblogs.com/jersey/a ... -
关联--依赖、组合--聚合 联系与区别
2012-06-06 11:51 992http://wenku.baidu.com/view/978 ... -
Java程序员面试中的多线程问题
2012-06-05 09:51 908http://sd.csdn.net/a/20120528/2 ... -
Java+多线程同步问题的探究
2012-05-16 21:05 8151、http://tech.ddvip.com/2010-05 ... -
java动态代理(JDK和cglib)
2012-05-06 18:00 772http://www.cnblogs.com/jqyp/arc ... -
Java语法总结 - 内部类
2012-05-03 23:08 748从Java1.1开始引入了内部类以来,它就引起了人们的激烈争论 ... -
深入分析 Java 中的中文编码问题
2012-03-15 09:02 486http://tech.ddvip.com/2012-03/1 ... -
Java框架数据库连接池比较(c3p0,dbcp和proxool)
2012-03-13 15:28 9321现在常用的开源数据连接池主要有c3p0,dbcp和proxoo ... -
java深入探索之1:类加载和实例化过程探析
2011-12-20 14:36 715http://www.cnblogs.com/xiaobai4 ... -
完整java开发中JDBC连接数据库代码和步骤
2011-12-19 19:54 848JDBC连接数据库 • ... -
JDBC中Statement接口提供的execute、executeQuery和executeUpdate之间的区别
2011-12-18 18:34 1026Statement 接口提供了三种 ... -
Java ThreadLocal介绍
2011-10-27 11:27 665概述 我们知道Spring通 ... -
怎样取得当前日期(包括时间)并把它存到数据库中?
2011-06-29 21:18 1199java.util.Date date = new ... -
JAVA中file路径问题
2011-06-22 11:40 1096前言 Java的路径问 ... -
使用 dom4j 解析 XML
2011-06-22 11:03 643http://www.ibm.com/developerwor ... -
Dom4j下载及使用Dom4j读写XML简介
2011-06-22 11:03 671http://www.blogjava.net/jungles ... -
JAVA文件拷贝
2011-06-20 09:37 801/** 02 * 将inputFile复制 ... -
Java中对Map(HashMap,TreeMap,Hashtable等)的排序
2011-05-25 14:52 947首先简单说一下他们之间的区别: HashMap: 最常用的 ...
相关推荐
### Tomcat启动时类加载顺序详解 #### 一、引言 Apache Tomcat是一款开源的Servlet容器,主要用于部署Java Web应用程序。它支持最新的Servlet、JSP等规范,并且以其轻量级、简单易用的特点而受到开发者的青睐。在...
java面试题静态加载顺序构造方法 继承与static 面试题目如下:请写出程序执行完成之后的结果。 package extend; public class X { Y y=new Y(); static{ System.out.println("tttt"); } X(){ System....
类实例化顺序是Java编程中一个关键的概念,它涉及到类加载、初始化以及对象创建等多个步骤。在面试或笔试中,理解这些顺序对于解答相关问题至关重要。以下是对类实例化顺序的详细讲解,以及阿里巴巴笔试题的解析。 ...
在Spring框架中,加载顺序是理解应用程序启动过程的关键部分,涉及到bean的实例化、初始化以及依赖注入等多个环节。本文将详细探讨Spring加载顺序,并结合`@PostConstruct`、`构造方法`以及`@Autowired`等关键注解...
了解Java中继承关系的加载顺序对于理解和优化程序性能至关重要。以下是对这个主题的详细阐述: 1. 类加载机制 Java的类加载过程分为三个主要阶段:加载、验证、准备、解析和初始化。当一个类被首次引用时(例如...
- 如果Bean配置了AOP代理,如`@AspectJ`切面或基于接口的代理,那么在Bean实例化后,Spring会创建代理对象,代理对象负责拦截方法调用并执行相关的切面逻辑。 6. **Bean的销毁** - 对于单例Bean,Spring容器在...
在Spring框架中,Bean的加载顺序是理解Spring IoC(Inversion of Control,控制反转)容器工作原理的关键部分。这个过程涉及到Bean定义的解析、实例化、初始化等多个步骤。让我们详细探讨Spring Bean加载顺序的各个...
1. 应用启动,Servlet容器查找并实例化实现ServletContainerInitializer的类(SpringServletContainerInitializer)。 2. SSCI发现并使用实现WebApplicationInitializer的类(比如我们的AACDSI子类)。 3. ...
误区一:认为在@Configuration类中@Bean方法的位置决定加载顺序。实际上,Bean的加载顺序并不由其在代码中的位置决定,而是由Spring的依赖解析机制和额外的排序规则决定。 误区二:误以为@Order注解可以控制所有Bean...
可以通过调整类加载顺序、设置类加载器优先级、使用命名空间隔离等手段来解决。 10. **类加载器调试** 为了帮助开发者调试类加载问题,WebLogic提供了详细的日志和跟踪选项,可以通过修改`weblogic.log.File`和`...
static代码块的加载时间是在类加载时执行的,而不是在实例化对象时执行的。 继承类中构造器的隐式调用 在上面的例子中,SubClass的构造器中会隐式的调用父类的构造器。这是因为在Java中,子类的构造器会隐式的调用...
此外,静态初始化块只在类加载时执行一次,而初始化块(也称为实例初始化块)会在每次创建类的新实例时执行。这意味着静态成员和静态初始化块对于所有类的实例都是共享的,而实例成员和实例初始化块则是每个对象独有...
Java中的类加载顺序是编程面试中常见的问题,它涉及到Java虚拟机(JVM)的工作机制。下面我们将详细探讨这个问题,并基于提供的代码实例进行解析。 首先,我们了解类加载的基本顺序: 1. **静态变量和静态初始化块...
Java类的初始化和实例化详解 Java类的初始化和实例化是Java编程语言中两个非常重要的概念,它们都是Java类生命周期的重要组成部分。下面我们将详细介绍Java类的初始化和实例化的过程,并解释其中的细节。 一、Java...
静态初始化块用于初始化类级别的静态变量,它们只在类加载时执行一次。 2. **成员变量初始化**:接下来,初始化父类的非静态成员变量,这是通过调用父类的构造器实现的。Java规定每个子类实例在构造之前必须先调用...
展类加载器的加载顺序来理解。系统类加载器在加载类时,会先尝试让扩展类加载器加载,如果扩展类加载器无法加载,则再由系统类加载器自己尝试加载。这样的设计是为了保证核心类库的唯一性和安全性,避免用户自定义的...
在Java中,类的加载顺序遵循“父类优先”原则,即子类在初始化前,其父类必须先完成初始化。同时,如果一个类没有显式地定义初始化块,JVM会自动提供一个空的方法。 至于文件`任务安排表.xls`,虽然与Java类的运行...
需要注意的是,静态初始化块只在类被加载时执行一次,而实例初始化块(构造函数)则会在每次创建类的实例时执行。静态变量的初始化是在类加载时完成的,而实例变量的初始化则在对象创建时进行。 在实际开发中,对...