先说结论:在sorted方法中,o1是最后面的元素,o2是倒数第二个元素,以此类推,流是处理元素是从后面开始取值。
package com.br.itwzhangzx02.learn;
import org.junit.Test;
import java.util.ArrayList;
import java.util.List;
import com.br.itwzhangzx02.learn.POJO.User;
public class StreamTest {
/**
* 1.id全部满足是偶数
* 2.年龄大于10
* 3.用户名大写
* 4.用户名字母倒排序
* 5.只输出一个用户名
*
*
* */
@Test
public void testStream(){
List<User> list = new ArrayList<User>(){
{
add(new User(1l,"q",10, "清华大学"));
add(new User(2l,"f",12, "清华大学"));
add(new User(3l,"b",15, "清华大学"));
add(new User(4l,"a",12, "清华大学"));
add(new User(5l,"d",25, "北京大学"));
add(new User(6l,"c",16, "北京大学"));
add(new User(7l,"t",14, "北京大学"));
add(new User(8l,"g",14, "浙江大学"));
add(new User(9l,"j",17, "浙江大学"));
add(new User(10l,"l",10, "浙江大学"));
}
};
list.stream()
.filter(user -> user.getId()%2==0)
.filter(user -> user.getAge()>10)
.map(user -> user.getName().toUpperCase())
.sorted((o1,o2)->{return 1;})
.forEach(System.out::println);
//sorted 方法中,我们重写compare方法:如果return是1,则是按照原先的排序排。-1则是按照逆序排
//
}
/** a negative integer, zero, or a positive integer as the
* first argument is less than, equal to, or greater than the
* second.**/
}
package com.br.itwzhangzx02.learn.POJO;
public class User {
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getSchool() {
return school;
}
public void setSchool(String school) {
this.school = school;
}
private Long id; //主键id
private String name; //姓名
private Integer age; //年龄
private String school; //学校
public User(Long id, String name, Integer age, String school) {
this.id = id;
this.name = name;
this.age = age;
this.school = school;
}
}
一开始的疑惑:
* @param o1 the first object to be compared.
* @param o2 the second object to be compared.
* @return a negative integer, zero, or a positive integer as the
* first argument is less than, equal to, or greater than the
* second.
* @throws NullPointerException if an argument is null and this
* comparator does not permit null arguments
* @throws ClassCastException if the arguments' types prevent them from
* being compared by this comparator.
*/
int compare(T o1, T o2);
上面是我拿到的Comparator接口中关于,compare方法的注释:
按照规则:第一个参数>第二个参数时,返回的是一个正数,那么返回正数,代表什么?
结论:正数的话,就是保持俩个元素的位置不变,负数的话俩元素位置互换。
首先,我们验证返回正数时,元素位置不变:
返回为负数时,元素位置互换。比如:F,A--->A,F
sorted()方法,使用的是默认的自然顺序--升序排序。但是,stream中元素,需要实现comparable接口,重写了方法compareTo方法。查看string的这个方法:
public int compareTo(String anotherString) {
int len1 = value.length;
int len2 = anotherString.value.length;
int lim = Math.min(len1, len2);
char v1[] = value;
char v2[] = anotherString.value;
int k = 0;
while (k < lim) {
char c1 = v1[k];
char c2 = v2[k];
if (c1 != c2) {
return c1 - c2;
}
k++;
}
return len1 - len2;
}
比较规则很简单,就是从字符数组的char[0]开始,比较二者的值,然后返回 c1-c2.
比如我们的F,A比较的时候,F>A,返回值为正数,根据我们前面的结论,返回一个正数时,顺序是保持不变的才对,那么这里到底是怎么回事?
为了看问题方便,我只保留了F和A的数据,测试如下:
按照我们的结论:返回正数时,顺序是不变的,为啥现在顺序变了,而且刚好是逆序。
最后,仔细想是不是,流处理元素的时候顺序的问题。比如:o1代表的是我们的A那个元素,o2代表的是F那个元素。
怎么验证呢?
人为抛异常,先处理的那个元素,最先抛异常
正常我们的数据,在list中是F元素在前,A元素在后。然后我们的代码先,o1转Integer抛错,从异常看,说明o1是A元素。
所以,最后的结论是,o1是A,o2是F。也就是说我们处理流时,不是按我们的list的索引从0开始的处理,而是反向处理的。
另一种论证方式:直接打印出我们的o1和o2
结论:在sorted((o1,o2)->{return o1.compareTo(o2)})这个方法里,我们的参数是从后往前取的。o1是最后一个元素,o2是倒数第二个元素,依次类推。