【笔记】从List中取出指定字段等于指定值的对象

例如我们有一个Student集合,每个对象有“id”,“name”,“age”三个字段,如下:
@Data
@AllArgsConstructor
public class Student {

    private Integer id;

    private String name;

    private Integer age;
}
其中id字段是1000~1099,如下:
Student(id=1000, name=学生_0000, age=18)
Student(id=1001, name=学生_0001, age=22)
Student(id=1002, name=学生_0002, age=27)
……
……
需求:我们需要根据id=xxxx来取出Student对象。

方案一

根据id字段,把List转换为Map。
//注意:该方法要求id字段的值是唯一的 否则需要使用下面的方法转Map
Map<Integer, Student> studentMap = studentList.stream().collect(Collectors.toMap(Student::getId, Function.identity()));

//根据id字段分组,且分组之后的对象集合取第一个对象
Map<Integer, Student> studentMap = studentList.stream().collect(Collectors.groupingBy(Student::getId, Collectors.collectingAndThen(Collectors.toList(), students -> students.get(0))));
需要取哪个对象,直接从map中取出即可。

方案二

使用stream的filter进行过滤。
Student student = studentList.stream().filter(s -> Objects.equals(s.getId(), id)).findFirst().orElse(null);
在filter中过滤出符合条件的对象,不匹配返回null。
实际使用中,数据量都不会特别大,个人认为方案二更清晰一些。
如果数据量较大,且需要从集合中多次取出,方案一的效率更高。因为方案一只聚合一次,而方案二需要反复创建stream并过滤集合。

性能测试

创建100个对象的测试集合,循环取值1000000次。
public class Test {

    public static void main(String[] args) {
        //创建集合
        List<Student> studentList = new ArrayList<>();
        for (int i = 0; i < 100; i++) {
            studentList.add(new Student(1000 + i, "学生_" + String.format("%04d", i), randomInt(18, 30)));
        }
        //分组测试
        long mapTs = System.currentTimeMillis();
        Map<Integer, Student> studentMap = studentList.stream().collect(Collectors.toMap(Student::getId, Function.identity()));
        for (int i = 0; i < 1000000; i++) {
            int id = randomInt(1000, 1100);
            Student student = studentMap.get(id);
        }
        System.out.println("分组测试: " + (System.currentTimeMillis() - mapTs) + "ms");
        //过滤测试
        long filterTs = System.currentTimeMillis();
        for (int i = 0; i < 1000000; i++) {
            int id = randomInt(1000, 1100);
            Student student = studentList.stream().filter(s -> Objects.equals(s.getId(), id)).findFirst().orElse(null);
        }
        System.out.println("过滤测试: " + (System.currentTimeMillis() - filterTs) + "ms");
    }
}
测试五次的结果:
分组测试: 34ms
过滤测试: 500ms
-----
分组测试: 34ms
过滤测试: 503ms
-----
分组测试: 32ms
过滤测试: 511ms
-----
分组测试: 33ms
过滤测试: 511ms
-----
分组测试: 31ms
过滤测试: 491ms
  • 15
    点赞
  • 51
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
如果您想使用链表来实现笔记管理,可以按照以下步骤在指定位置插入笔记: 1. 首先,需要定义一个笔记的结构体,包含笔记的内容、时间、作者等信息。 2. 然后,需要定义一个链表节点的结构体,包含笔记结构体的指针和下一个节点的指针。 3. 接着,需要定义一个头指针,指向链表的第一个节点。 4. 如果要在指定位置插入笔记,需要先找到插入位置的前一个节点。可以使用循环遍历链表,直到找到需要插入位置的前一个节点。 5. 在找到插入位置的前一个节点之后,可以创建一个新的节点,将新节点的指针指向插入位置的后一个节点,将插入位置的前一个节点的指针指向新节点。 6. 最后,将新节点的笔记信息赋并存储到新节点的笔记结构体。 下面是一个示例代码,用来实现在链表指定位置插入笔记: ```c++ #include <cstdio> #include <cstdlib> struct Note { char content[100]; char time[20]; }; struct Node { Note *note; Node *next; }; int main() { Node *head = (Node*)malloc(sizeof(Node)); head->next = NULL; // 初始化链表为空 // 假设已经定义了需要插入的笔记 Note *new_note = (Note*)malloc(sizeof(Note)); strcpy(new_note->content, "This is a new note."); strcpy(new_note->time, "2021-09-15 10:30"); // 在第二个节点后面插入笔记 Node *p = head; int pos = 2; // 插入位置 int i = 0; while (p != NULL && i < pos - 1) { p = p->next; i++; } if (p == NULL || i > pos - 1) { printf("插入位置不合法\n"); return -1; } Node *new_node = (Node*)malloc(sizeof(Node)); new_node->note = new_note; new_node->next = p->next; p->next = new_node; // 打印链表笔记 p = head->next; while (p != NULL) { printf("%s %s\n", p->note->time, p->note->content); p = p->next; } return 0; } ``` 注意,在实际应用,需要考虑链表的删除、修改等操作,以及内存的管理问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值