工作中递归返回多层级josn数据

最近回顾之前的项目看到一段当时同事写的代码设计非常的巧妙,然后特意的debug分析了一段把几乎每一段可以写上注释的地方都写上了注释,感觉每一步都体现出当时码代码人的基本功,我记得当时这个还是同事半个下午写出来的,拿出一段分享一下。

@Override
    public List<APIPropVO> findByAPIId(Integer apiId) throws Exception {
        // 查出该API第一等级属性
        List<APIPropEO> onePropEOList = apiPropBiz.findOneLevel(apiId);
        //查询第一级的APIPropEO对象转换成APIPropVO调用getApiPropVO方法
        List<APIPropVO> onePropVOList = getApiPropVO(onePropEOList);
        //调用setSonList设置子集合
        setSonList(onePropVOList);
        return onePropVOList;
    }

    //getApiPropVO的方法将APIPropEO对象转换成APIPropVO
    private List<APIPropVO> getApiPropVO(List<APIPropEO> onePropEOList) {
        //定义APIPropVO集合
        List<APIPropVO> propVOList = new ArrayList<APIPropVO>();
        for (APIPropEO prop : onePropEOList) {
            //把每一个遍历的APIPropEO对象放到APIPropVO对象中
            APIPropVO propVO = new APIPropVO(prop);
            //将其添加对应的集合
            propVOList.add(propVO);
        }
        return propVOList;
    }

    //设置子集合方法遍历第一层集合
    private void setSonList(List<APIPropVO> onePropVOList) {
        //当前传入的List的集合大小
        int len = onePropVOList.size();
        //遍历
        for (int i = 0; i < len; i++) {
            //通过List的集合的get方法取到每一个APIPropVO对象
            APIPropVO propVO = onePropVOList.get(i);
            // 没有子属性集合则跳过
            if (propVO.getSonFlag() == 0) {
                continue;
            }
            //有子集id查询数据库APIPropEO的集合
            List<APIPropEO> sonList = apiPropBiz.findSonList(propVO.getId());
            //再将第二级APIPropEO对象转换成APIPropVO调用getAPIPropVO方法
            List<APIPropVO> sonVOList = getAPIPropVO(sonList);
            /**
             * Long[] l = (Long[])list.toArray();//这个语句会出现ClassCastException 
             * 处理方式如下面代码:
             * Long [] l = (Long []) list.toArray(new Long[list.size()]); 
             */
            //确定该数组的转换类型和数组的长度参数指定空数组,节省空间 
            APIPropVO[] sonVOArr = new APIPropVO[sonVOList.size()];
            //将数组转换为集合
            propVO.setSonArr(sonVOList.toArray(sonVOArr));
            //给当前子集i的key值设置propVO
            onePropVOList.set(i, propVO);
            //递归调用子集下面是否还有子集
            setSonList(sonVOList);
        }
    }

    //和getApiPropVO的方法一样将APIPropEO对象转换成APIPropVO
    private List<APIPropVO> getAPIPropVO(List<APIPropEO> sonList) {
        List<APIPropVO> sonVOList = new ArrayList<APIPropVO>();
        for (APIPropEO apiProp : sonList) {
            APIPropVO apiPropVO = new APIPropVO(apiProp);
            sonVOList.add(apiPropVO);
        }
        return sonVOList;
    }

集合转数组的toArray()和toArray(T[] a)方法 下面代码是jdk ArrayList中的源码

public <T> T[] toArray(T[] a) {   
        if (a.length < size)   
            a = (T[])java.lang.reflect.Array.   
                newInstance(a.getClass().getComponentType(), size);   
            System.arraycopy(elementData, 0, a, 0, size);   
        if (a.length > size)   
            a[size] = null;   
        return a;   
    }   



 public Object[] toArray()  
    {  
        Object aobj[] = new Object[size];  
        System.arraycopy(((Object) (elementData)), 0, ((Object) (aobj)), 0, size);  
        return aobj;  
    }  

    public Object[] toArray(Object aobj[])  
    {  
        if(aobj.length < size)  
            aobj = (Object[])(Object[])Array.newInstance(((Object) (aobj)).getClass().getComponentType(), size);  
        System.arraycopy(((Object) (elementData)), 0, ((Object) (aobj)), 0, size);  
        if(aobj.length > size)  
            aobj[size] = null;  
        return aobj;  
    }

1.该方法用了泛型,并且是用在方法的创建中(相当于定义泛型,T[]是在使用泛型T) 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法 2.该方法返回集合中所有元素的数组;返回数组的运行时类型与指定数组的运行时类型相同。 3.与 public Object[] toArray() 的比较

public Object[] toArray() { 
Object[] result = new Object[size]; 
System.arraycopy(elementData, 0, result, 0, size); 
return result; 
    }

从源码中可以看出它仅能返回 Object[]类型的,相当于toArray(new Object[0]) 注意:数组不能强制转换

不带参数的toArray方法,是构造的一个Object数组,然后进行数据拷贝,此时进行转型就会产生ClassCastException

String[] tt =(String[]) list.toArray(new String[0]); 这段代码是没问题的,但我们看到String[] tt =(String[]) list.toArray(new String[0]) 中的参数很奇怪,然而去掉这个参数new String[0]却在运行时报错。。。

该容器中的元素已经用泛型限制了,那里面的元素就应该被当作泛型类型的来看了,然而在目前的java中却不是的,当直接String[] tt =(String[]) list.toArray()时,运行报错。回想一下,应该是java中的强制类型转换只是针对单个对象的,想要偷懒将整个数组转换成另外一种类型的数组是不行的,,这和数组初始化时需要一个个来也是类似的。

带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayList的size一致的空数组,虽然方法本身还是以 Object数组的形式返回结果,不过由于构造数组使用的ComponentType跟需要转型的ComponentType一致,就不会产生转型异常。

解决方案. Solutions

因此在使用toArray的时候可以参考以下三种方式

1. Long[] l = new Long[<total size>];

     list.toArray(l);

  2. Long[] l = (Long[]) list.toArray(new Long[0]);

  3. Long[] a = new Long[<total size>];

      Long[] l = (Long[]) list.toArray(a);

1).参数指定空数组,节省空间 String[] y = x.toArray(new String[0]); 2).指定大数组参数浪费时间,采用反射机制 String[] y = x.toArray(new String[100]); //假设数组size大于100 3).姑且认为最好的 String[] y = x.toArray(new String[x.size()]);

以下代码会出现ClassCastException

List list = new ArrayList();   
list.add(new Long(1)); 
list.add(new Long(2));   
list.add(new Long(3)); 
list.add(new Long(4));   
Long[] l = (Long[])list.toArray();//这个语句会出现ClassCastException

处理方式如下面代码: Long [] l = (Long []) list.toArray(new Long[list.size()]);

每天

进步一点点

本文分享自微信公众号 - 技术从心(gh_d845efe513db)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-06-25

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏我是攻城师

Java基本类型的内存分配在栈还是堆

我们都知道在Java里面new出来的对象都是在堆上分配空间存储的,但是针对基本类型却有所区别,基本类型可以分配在栈上,也可以分配在堆上,这是为什么?

27810
来自专栏即时通讯技术

自已开发IM有那么难吗?手把手教你自撸一个Andriod版简易IM (有源码)

一直想写一篇关于im即时通讯分享的文章,无奈工作太忙,很难抽出时间。今天终于从公司离职了,打算好好休息几天再重新找工作,趁时间空闲,决定静下心来写一篇文章,毕竟...

22620
来自专栏码的一手好代码

spark RPC原理

Spark-1.6以后RPC默认使用Netty替代Akka,在Netty上加了一层封装,为实现对Spark的定制开发,所以了解Spark中RPC的原理还是有必要...

14420
来自专栏码的一手好代码

Scala使用

Scala是一门主要以Java虚拟机(JVM)为目标运行环境并将面向对象和函数式编程语言的最佳特性综合在一起的编程语言。你可以使用Scala编写出更加精简的程序...

11430
来自专栏码的一手好代码

Java中的按值传递

这个时候可能会有疑问了,为什么add方法可以修改List数组,但是append和addNum却没有修改传进来的值

8440
来自专栏测试邦

RestAssured接口系列|环境搭建

http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.h...

14040
来自专栏shysh95

Dubbo注册中心

关于源码和原理的分析,我们都需要找寻一个切入点,找到切入点的前提是你要知道注册中心的功能是什么,注册中心相信大家都不陌生,每一个通用的注册中心都需要提供两个基本...

23550
来自专栏信息安全小学生

Gradle 笔记

Task 是 Gradle 构建的最小执行单元。 clean 就是一种最常见的 Task。

10630
来自专栏Jerry的SAP技术分享

how do you usually upload picture in SCN A workaround for current SCN upload is

版权声明:署名,允许他人基于本文进行创作,且必须基于与原先许可协议相同的许可协议分发本文 (Creative Commons)

12040
来自专栏码的一手好代码

Java中的静态绑定与动态绑定

由上面我们可以得出结论,如果一个方法不可被继承或者继承后不可被覆盖,那么这个方法就采用的静态绑定。

9330

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励