首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >工作中递归返回多层级josn数据

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

作者头像
技术从心
发布2019-08-06 10:40:54
6780
发布2019-08-06 10:40:54
举报
文章被收录于专栏:技术从心技术从心

最近回顾之前的项目看到一段当时同事写的代码设计非常的巧妙,然后特意的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()]);

每天

进步一点点

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-06-25,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 技术从心 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 最近回顾之前的项目看到一段当时同事写的代码设计非常的巧妙,然后特意的debug分析了一段把几乎每一段可以写上注释的地方都写上了注释,感觉每一步都体现出当时码代码人的基本功,我记得当时这个还是同事半个下午写出来的,拿出一段分享一下。
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档