有这样一个需求,在excel导出报表的时候,产品经理想在首列加个序号,可能大家就会问了,直接取数据库的id不就可以吗?但是这样会有一个问题,很多时候我们从数据库捞数据出来还会对数据分组排序等一系列的业务操作,这样一来,这个id还会是我们想要的id吗?显然不是的,还有些id可能是分布式id,其长度就有10位之长,是产品经理想要的结果吗 ?显然不是。
这个时候Java8的stream就闪亮登场了:
List userList = new ArrayList<>(); userList.add(new User("张三","05-07")); userList.add(new User("李四","05-07")); userList.add(new User("张三","05-06")); userList.add(new User("李四","05-06")); System.out.println(JSONObject.toJSONString(userList)); //排序,把姓名相同的放在一起,再根据时间进行排序 userList = userList.stream().sorted( Comparator.comparing(User::getUserName) .thenComparing(User::getDate)).collect(Collectors.toList()); System.out.println(JSONObject.toJSONString(userList)); //给list每个元素添加序号,可以试下这里如果换成 Integer i = 1; 下面set的时候传 i++ 试下看会发生什么? Integer[] arr = {1}; userList = userList.stream().peek(e->e.setId(arr[0] ++)).collect(Collectors.toList()); System.out.println(JSONObject.toJSONString(userList));
打印结果:
[{"date":"05-07","userName":"张三"}, {"date":"05-07","userName":"李四"}, {"date":"05-06","userName":"张三"}, {"date":"05-06","userName":"李四"}][{"date":"05-06","userName":"张三"}, {"date":"05-07","userName":"张三"}, {"date":"05-06","userName":"李四"}, {"date":"05-07","userName":"李四"}][{"id":1,"date":"05-06","userName":"张三"}, {"id":2,"date":"05-07","userName":"张三"}, {"id":3,"date":"05-06","userName":"李四"}, {"id":4,"date":"05-07","userName":"李四"}]
扩展
感兴趣的同学可以试下将Integer数组换成 Integer i = 1; 下面set的时候传 i++ 试下会发生什么?
这个时候会发现里面都是1,并没有发生自增运算的1,2,3,4..结果;
原因总结
这又是为什么呢?
因为Lambda表达式最终会编译为匿名内部类,搞java的人都知道,如果在匿名内部类中调用外部变量的时候,外部变量须为final,在这里如果是Integer的引用类型,则不会被改变,如果是数组,则它的引用不会改变,但是值可以通过自增运算去改变的!