1、不要把Redis当成数据库来使用
需求:把数组转成list集合去处理。
方法:Arrays.asList 或者 JAVA8的stream流式处理
以下代码是否有问题?
int[] arr = {1, 2, 3};
List list = Arrays.asList(arr);
System.out.println(list);
System.out.println(list.size());
System.out.println(list.get(0).getClass());
日志:
[[I@4232c52b]
1
class [I
为什么打印出来的,不是我们想要的?
可以看到入参是泛型T类型可变参数,把 int数组 整体作为对象成为了T 传入了。
如何解决?
Integer[] arr2 = {1, 2, 3};
List list2 = Arrays.asList(arr2);
System.out.println(list2);
System.out.println(list2.size());
System.out.println(list2.get(0).getClass());
日志:
[1, 2, 3]
3
class java.lang.Integer
总结:不能直接使用Arrays.asList来转换基本类型数组。
现在我们已经转成list集合了,我们接下来,进行后续业务操作。
Integer[] arr2 = {1, 2, 3};
List list2 = Arrays.asList(arr2);
list2.add(4);
System.out.println(list2);
这时候,一运行,又报错了!
java.lang.UnsupportedOperationException
at java.util.AbstractList.add(AbstractList.java:148)
at java.util.AbstractList.add(AbstractList.java:108)
at ListTest.test2(ListTest.java:30)
ListTest.java:30 是代码中的 list2.add(4);
原因:
发现:Arrays.asList返回的List并不是java.util.ArrayList,而是Arrays内部类ArrayList,它没有覆写父类的add方法,所以调用会报错。
解决:
重新new一个ArrayList初始化Arrays.asList返回的List即可。
Integer[] arr2 = {1, 2, 3};
List list2 = Arrays.asList(arr2);
ArrayList list3 = new ArrayList(list2);
list3.add(4);
System.out.println(list3);
总结:Arrays.asList返回的List不支持增删操作。
Integer[] arr2 = {1, 2, 3};
List list2 = Arrays.asList(arr2);
arr2[1] = 6;
System.out.println(list2);
日志:
[1, 6, 3]
发现:对原始数组的修改影响到了我们获得的那个List。
解决:
重新new一个ArrayList初始化Arrays.asList返回的List即可。
Integer[] arr2 = {1, 2, 3};
ArrayList list3 = new ArrayList(Arrays.asList(arr2));
arr2[1] = 6;
System.out.println(list3);
日志:
[1, 2, 3]
总结:对原始数组的修改影响到了我们获得的那个List。
1、目的
Lambda表达式的初衷是进一步简化匿名类的语法。
2、简单实例
//匿名类
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("hello1");
}
}).start();
//lambda表达式
new Thread(() -> System.out.println("hello1")).start();
在idea中快捷键:Alt + Enter 会看到提示,自动转成lambda表达式
可以对集合进行投影,转换,过滤,排序等操作。
创建四个实体类,用于下方Demo的使用。
创建流程一般有五种方式。
ream方法把List或数组转换成流
/**
* list ==> 流
*/
List<String> list = Arrays.asList("a1", "a2", "a3");
list.stream().forEach(System.out::println);
/**
* 数组 ==> 流
*/
Arrays.stream(new int[]{1, 2, 3}).forEach(System.out::println);
b.通过stream.of方法直接传入多个元素构成一个流程
String[] arr = {"a", "b", "c"};
Stream.of(arr).forEach(System.out::println);
Stream.of("a", "b", "c").forEach(System.out::println);
Stream.of(1, 2, "a").forEach(System.out::println);
c.通过Stream.iterate方法使用迭代的方式构造一个无限流,然后使用limit限制流元素个数
Stream.iterate(2, item -> item * 2).limit(2).forEach(System.out::println);
Stream.iterate(BigInteger.ZERO, n -> n.add(BigInteger.TEN)).limit(3).forEach(System.out::println);
d.通过Stream.generate方法从外部传入一个提供元素的Supplier来构造无限流,
然后使用limit限制流元素个数。
Stream.generate(() -> "test").limit(3).forEach(System.out::println);
Stream.generate(Math::random).limit(3).forEach(System.out::println);
e.通过IntStream或DoubleStream构造基本类型的流
IntStream.range(1, 5).forEach(System.out::println);
DoubleStream.of(1.1, 2.2, 3.3).forEach(System.out::println);
可以实现过滤操作,类似SQL中的where,可以连续叠加filter方法进行多次过滤。
orders.stream()
.filter(order -> order.getTotalPrice() > 40)
.filter(order -> order.getCustomerId() == 1)
.forEach(System.out::println);
可以做转换(就是投影),类似于SQL中的select。
通过map,可以把对象转换成其他对象。
a、小写转大写
List<String> a = Arrays.asList("a", "b", "c", "d");
List<String> b = a.stream().map(String::toUpperCase).collect(Collectors.toList());
System.out.println(a);
System.out.println(b);
b、对象列表转字符串列表
List<String> collect =
orders.stream().map(x -> x.getCustomerName())
.collect(Collectors.toList());
System.out.println(collect);
c、对象列表转其他对象列表
List<User> users = customers.stream().map(temp -> {
User user = new User();
user.setCustomerId(temp.getId());
user.setName(temp.getName());
return user;
}).collect(Collectors.toList());
相当于 map + flat,通过map把每一个元素替换成一个流,然后展开这个流。
可以用于排序,类似于SQL中的 order by。
orders.stream()
.filter(order -> order.getTotalPrice() > 40)
.sorted(Comparator.comparing(Order::getTotalPrice).reversed())
.limit(2)
.forEach(System.out::println);
作用是去重,类似于SQL中的distinct。
List<String> list =
Arrays.asList("AA", "BB", "CC", "AA", "BB");
System.out.println(list);//[AA, BB, CC, AA, BB]
List<String> collect = list.stream().distinct().collect(Collectors.toList());
System.out.println(collect);//[AA, BB, CC]
用于分页,类似于MySQL中的limt。
skip实现跳过一定的项,limit用于限制项总数。
customers.stream()
.sorted(Comparator.comparing(Customer::getId).reversed())
.map(customer -> customer.getId() + "@" + customer.getName())
.limit(2).forEach(System.out::println);
//10@姓名10
//9@姓名9
customers.stream()
.sorted(Comparator.comparing(Customer::getId).reversed())
.map(customer -> customer.getId() + "@" + customer.getName())
.skip(2).limit(2).forEach(System.out::println);
//8@姓名8
//7@姓名7
收集操作,对流程进行终止操作。(终止意思是,后面无法再串联其他操作)
其他终止操作还有 forEach,toArray,min,max,cout,anyMatch等等。
#所有下单的用户,使用toSet去重后实现字符串的拼接
String collect = orders.stream()
.map(order -> order.getCustomerName())
.collect(Collectors.toSet())
.stream().collect(Collectors.joining(",", "[", "]"));
System.out.println(collect); //[李四,张三]
#使用toCollection收集器指定集合类型
LinkedList<Order> collect = orders.stream()
.limit(2)
.collect(Collectors.toCollection(LinkedList::new));
System.out.println(collect);
Collectors类的一些常用静态方法:
【注意】Lambda表达式有个很大的缺点,就是不方便调试。
【解决】
1、使用2020.01版本的IDEA编辑器
https://www.jetbrains.com/help/idea/analyze-java-stream-operations.html
2、使用peek方法
在Java8之前处理日期时间时,使用Date,Calender 和 SimpleDateFormat,使用不方便。
而且无法申明时区。
背景:
去年做的realme的pk大赛,服务器使用的是utc时间(世界标准时间),而Pk赛是印度的项目,业务逻辑需要使用印度时间。
解决:
写了一个UTC转印度时间的工具类,把new出来的时间往后加五个半小时,就是印度时间。
https://shimo.im/docs/yJggDyvktT8DJdhc/ 《【pk大赛】服务器UTC时间解决方案》
1、对于Date类
都一样。因为,Date 中保存的是 UTC 时间,UTC 是以原子钟为基础的统一时间,不以
太阳参照计时,并无时区划分。
2、时区问题
如果不理会时区问题,会导致什么现象呢?
官方文档:
https://alibaba.github.io/arthas/
开源地址:
https://github.com/alibaba/arthas/blob/master/README_CN.md
1、什么是Arthas
是Alibaba开源的Java诊断工具。
2、可以解决什么问题(摘抄自官网)
3、如何安装