问题描述:
在工作中使用org.apache.commons.beanutils.BeanUtils.copyProperty(Object bean,String name,Object value)方法将数据库查询结果集的某个字段转成对象的属性值,由于属性字段刚好是java.sql.Timestamp字段,导致value为null时,直接报异常"No value specified for 'java.sql.Timestamp'"。在网上找了些资料,注册一下转换类。为了使用方便,我直接重新命名了一个类
class BeanUtilsEx extends BeanUtils { static{ ConvertUtils.register(new SqlTimestampConverter(null), java.sql.Timestamp.class); } }
但是实际调用的时候还是报异常,期初怀疑是new SqlTimestampConverter(null),不能处理null值,干脆自己写死一个Converter
class MyConverter implements org.apache.commons.beanutils.Converter{ @Override public Object convert(Class arg0, Object arg1) { return new Timestamp(System.currentTimeMillis()); } }
上面的注册类则改成:
class BeanUtilsEx extends BeanUtils { static{ ConvertUtils.register(new MyConverter(), java.sql.Timestamp.class); } }
可是在执行以下代码时还是报错:
public static void main(String[] args){ Test obj = new Test(); try { BeanUtilsEx.copyProperty(obj, "time", null); } catch (Exception e) { e.printStackTrace(); } }
于是想看看我的MyConverter有没有注册上:
public static void main(String[] args) throws Exception { try { BeanUtilsEx.copyProperty(obj, "time", null); } catch (Exception e) { e.printStackTrace(); } Converter converter = ConvertUtils.lookup(java.sql.Timestamp.class); if(converter instanceof MyConverter){ System.out.println("Yes"); } }
结果,没有打印Yes,怀疑BeanUtilsEx的静态代码块没有执行,于是打断点调试,果然确实没有执行。于是认为调用静态方法时不会触发静态代码块的调用,到网上求证了一下,我嚓,网上跟我想的正好相反,“调用静态方法时会先执行静态块(只执行一次)”,那我这里又是什么原因呢?我不太相信网上说的,于是在BeanUtilsEx中加个静态方法试试:
class BeanUtilsEx extends BeanUtils { static{ ConvertUtils.register(new MyConverter(), java.sql.Timestamp.class); } public static void print(){ System.out.println("static method"); } }
然后在main方法中调用:
public static void main(String[] args) throws Exception { Test obj = new Test(); try { BeanUtilsEx.print(); BeanUtilsEx.copyProperty(obj, "time", null); } catch (Exception e) { e.printStackTrace(); } Converter converter = ConvertUtils.lookup(java.sql.Timestamp.class); if(converter instanceof MyConverter){ System.out.println("Yes"); } }
我嚓,静态块还真执行了,而且确实比print()方法要早,而且这回也不报错了,也打印了“Yes”,说明我的转换器已经成功注册了,可是这又是为什么?执行print()的时候的顺序是:静态块->print()方法;执行copyProperty的时候则是:copyProperty->静态块不执行。print()方法和copyProperty()的差别在哪呢?难道是因为copyProperty()是父类的静态方法,所以执行copyProperty()的时候不会加载子类的static块,再写个子类验证一下:
class BeanUtilsEx extends BeanUtils { static{ ConvertUtils.register(new MyConverter(), java.sql.Timestamp.class); System.out.println("father static block"); } public static void print(){ System.out.println("father static method"); } } class SubBeanUtilsEx extends BeanUtilsEx{ static { System.out.println("sub static block"); } public static void subPrint(){ System.out.println("sub static method"); } }
然后首先执行SubBeanUtilsEx .print()方法,打印结果如下:
father static block father static method
可以看出,在执行父类的静态方法时,先执行父类的static块,而子类的static块则没有执行;
接着执行SubBeanUtilsEx.subPrint()方法,打印结果如下:
father static block sub static block sub static method
可以看出,在执行子类自身的静态方法时,首先执行父类static块,然后执行子类static块,最后才是子类的static method。
相关推荐
java面试题静态加载顺序构造方法 继承与static 面试题目如下:请写出程序执行完成之后的结果。 package extend; public class X { Y y=new Y(); static{ System.out.println("tttt"); } X(){ System....
本文将深入探讨Java中的静态块与静态方法,解析它们的区别、联系以及执行顺序,帮助读者更全面地理解`static`关键字在Java中的运用。 ### 静态方法 静态方法是通过`static`关键字声明的,它属于类而不属于类的任何...
这是因为静态块是在类加载时执行的,而类的加载顺序是按照继承层次自上而下进行的,所以父类的静态块总是先于子类的静态块执行。 2. **构造函数的执行**:接下来,父类的构造函数会被调用,然后才是子类的构造函数...
Java 代码块与静态代码块加载顺序详解 Java 代码块与静态代码块加载顺序是 Java 语言中一个重要的概念,理解这两个概念对于编写高效、可靠的 Java 代码至关重要。本文将详细介绍 Java 代码块与静态代码块加载顺序的...
一个类的静态块可以有多个,其执行顺序由它在代码中的顺序决定。 在静态块中,我们通常进行一些需要进行异常捕获的静态变量的初始化。 例如: ```java public class StaticExample { static { System.out....
statilocal java 静态变量和静态块加载顺序,静态变量和同名静态变量块加载的顺序。
1. **静态块的执行顺序**:首先,从父类到子类,按照定义的顺序依次执行静态块。这意味着父类的静态块先于子类的静态块执行。 2. **实例化时的执行顺序**: - 当创建子类对象时,首先执行父类无参构造器。如果父类...
静态块按照类的加载顺序执行,先父类后子类;而实例初始化块则在创建对象时按遇到的顺序执行,先父类后子类。在第二个示例中,输出顺序为`Father static!`、`Child static!`、`Father block!`、`Father Constructor!...
本文将深度剖析几个常见的Java面试题目,包括方法重写、静态代码块和构造方法的执行顺序。 首先,我们来讨论**方法重写(Override)**。在面向对象编程中,子类可以继承父类的属性和方法,有时需要根据自己的需求对...
当类第一次被加载时,静态块按照它们在代码中的顺序依次执行。 下面是一个简单的静态块示例,展示了如何使用静态块初始化静态变量: ```java public class StaticBlockExample { static int count; static { ...
在Java编程语言中,"静态"(static)关键字是一个非常重要的概念,它具有多种用途,包括定义静态变量、静态方法、静态初始化块以及静态类。理解这些概念对于编写高效的Java代码至关重要。 首先,我们来解释静态变量...
3. **执行顺序**:如果有多个静态块,它们会按照在类中出现的顺序依次执行。 4. **作用域**:静态块中的代码不依赖于类的实例,因此它不能访问非静态成员,除非通过类名引用。 ### 静态块的用途 1. **静态变量初始...
这两个静态代码块是在类加载时自动执行的,它们的执行顺序是按照它们在类中的声明顺序来的。 非静态代码块 非静态代码块是 Java 中的一种特殊的代码块,它是非静态的,意思是说它是属于对象的,而不是属于类的。非...
在多级继承关系中,静态代码块按照类的加载顺序执行,先加载的类先执行其静态代码块。 例如,在给定的代码中: ```java public class A { static { System.out.print(1); } public A() { System.out.print(2)...
特别是当涉及到静态域(静态变量)、非静态域(实例变量)、静态块、非静态块以及构造函数时,明确它们的初始化顺序有助于避免潜在的编程错误。 根据题目提供的内容,我们将重点讨论这些概念以及它们之间的相互关系...
静态初始化块是一段在类加载时执行的代码,用来初始化静态变量。这种块可以有多个,它们的执行顺序取决于在源代码中的位置。静态初始化块只执行一次,通常用于设置复杂的静态变量初始值。 5. 静态类 在Java中,静态...
2. **资源加载**:如果程序依赖于某些外部资源(如配置文件、数据库连接等),可以利用静态代码块在程序启动时完成这些资源的加载工作。 3. **一次性的设置**:如果有一些操作只需要在程序启动时执行一次,则可以...
了解Java中继承关系的加载顺序对于理解和优化程序性能至关重要。以下是对这个主题的详细阐述: 1. 类加载机制 Java的类加载过程分为三个主要阶段:加载、验证、准备、解析和初始化。当一个类被首次引用时(例如...
在继承关系中,静态代码块的执行顺序是:父类静态代码块 -> 子类静态代码块 -> 父类非静态代码块 -> 父类构造方法 -> 子类非静态代码块 -> 子类构造方法。 静态代码块和静态方法的区别在于:静态代码块是自动执行的...
先静态代码块,然后构造代码块,再是构造方法,然后方法里面的代码块按顺序执行。 子类继承父类的时候,先执行父类的静态代码块,然后子类的静态代码块,然后父类的构造代码块和构造方法,再是子类的构造代码块和...