静态块 静态方法的加载顺序

问题描述:

在工作中使用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。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
http://blog.csdn.net/methods2011/article/details/8584463 Java中的继承与静态static等的执行先后顺序的面试题 java面试题静态加载顺序构造方法 继承与static 面试题目如下:请写出程序执行完成之后的结果。 package extend; public class X { Y y=new Y(); static{ System.out.println("tttt"); } X(){ System.out.println("X"); } public static void main(String[] args) { new Z(); } } class Y{ Y(){ System.out.println("Y"); } } class Z extends X{ Y y=new Y(); static{ System.out.println("tt"); } Z(){ System.out.println("Z"); } } 先不告诉最后结果,我们先来分析下。一步一步推出结果。 1.首先分析一段程序的执行后的结果,我们得先找到程序的入口,然后才能着手分析。 也就是main()方法。 2.我们发现main()方法在X类中,要执行main()方法,还得先将X类加载到内存中。 3.X类加载完成后,会做什么事情呢?别急,先来看看static的作用,不知道吧。告诉你:static就是在类被第一次加载的时候执行,以后就不再执行。 4.知道了static的作用,那么X类被加载,那么就会先执行X类的静态属性和静态语句static),执行先后顺序看谁在前面就先执行谁。只在此时执行,以后都不会。 5.所以一个输出结果为tttt,没问题了吧。 6.X类的static语句执行完了,就该执行main()方法啦。 7.new Z();此方法被执行。 8.既然new Z();那么Z类就要被加载。因为Z类继承X类。所以必须先加载X类才行。因为X类已经被加载。所以此时不用再加载X类了。Z类加载好了就要执行Z类的static语句 9.那么就会打印出tt了吧。 10.都加在完后就要实例化对象了。 11.实例化Z之前,还得先实例化X对吧。因为子类的构造方法都会调用父类的构造方法。 12.那就先实例化X类吧。 13.执行X方法前还得先初始化对不。也就是获取所有属性。那么X类的属性Y就会获取。 14.即X类的Y y=new Y();要被执行。也就是会打印Y。 15.接着执行System.out.println("X"); 16.然后就是执行Z的构造方法 17.同样先获取Z的属性Y y=new Y();打印Y。 18.再执行System.out.println("Z"); 整个过程就是这样了。现在知道结果了吧: tttt tt Y X Y Z http://snow4909.diandian.com/post/2013-02-17/40049419937 记住,面试有可能会标出一些语句的,让你选择这些语句的执行顺序。不过只要知道原理,就没什么难的了。 执行先后顺序: 1、类加载到内存时static 加载进内润 2、调用类的构造方法时先调用父类的构造方法,在调用子类的构造方法 3、类初始化时,先初始化类的属性成员,在执行构造方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值