用java实现cat命令_024:Java流实现Shell:cat 1.log | grep a | sort | uniq -c | sort -rn

88008a84be5ef2fe01c38a0da117cea3.png

本文阅读时间大约13分钟(本文实践性很强,建议pc端阅读,最好亲自实践)。

参考答案

这个问题考察的是对Linux命令的熟悉程度,以及对Java中集合操作的综合运用,自从转到Java 8以后,我就一直使用流来处理集合了,下面的代码就是我用流来实现的参考答案:package org.java.learn.java8.stream;

import java.io.*;

import java.util.*;

import java.util.function.Function;

import java.util.stream.Collectors;

public class ShellExample {

public static void main(String[] args) throws IOException {

//cat命令,相当于是读取文件中的所有行,并输出

File file = new File("/Users/duqi/IdeaProjects/LearnJava/src/main/java/org/java/learn/java8/stream/t1.txt");

BufferedReader bufferedReader = new BufferedReader(new FileReader(file));

List lines = new ArrayList<>();

String str = null;

while ((str = bufferedReader.readLine()) != null) {

lines.add(str);

}

//grep a,相当于filter

lines = lines.stream().filter(s -> s.contains("a")).collect(Collectors.toList());

//sort 按照字典序从小到大排序

lines = lines.stream().sorted().collect(Collectors.toList());

//uniq -c,统计相同的元素的个数

Map integerMap =

lines.stream().sorted().collect(Collectors.groupingBy(Function.identity(), Collectors.counting()));

//sort -rn,排序后逆序输出

List res = integerMap.values().stream().sorted(Comparator.reverseOrder()).collect(Collectors.toList());

res.forEach(System.out::println);

}

}

知识点梳理

背景&基本概念

在以前,要操作一个集合,按照Java作为命令式语言的特点,开发者需要自己去关心集合的循环,每个循环里针对元素的操作(过滤、转换、合并)等等,这些代码写起来很繁琐,又容易出错。

流(stream)是Java API的新成员,它允许开发者以声明方式处理集合(类似于写SQL),开发者只需要直接指明自己要做什么操作,而不需要关心对集合的迭代。使用流写出来的代码可读性很好、表达能力很强,我目前在开发中,能使用流的地方一定会使用流,它帮助我减少了很多代码行数。

流也需要对集合做迭代,只是JDK的开发者将迭代放在了API背后,称为内部迭代,而集合的迭代则需要开发者自己维护,称为外部迭代。使用内部迭代的好处,一方面开发者的代码得以简化,另一方面,流可以在内部对迭代进行种种优化,同时不影响开发者的业务代码。

常见api

流的API分为两种,中间操作和终端操作,中间操作产生的结果还是一个流,终端操作产生的结果可能是一个集合或者是一个数字,总之不是一个流。

c8e0f272ca814afde6e35df7e1c9d168.png

常见的流的操作有:筛选(filter)、切片(limit)、映射(map、flatMap)、查找(find)、匹配(match)和规约(reduce);流不仅支持串行操作,还支持并行操作,使用并行流可以提高处理超大集合时候的性能。这里我整理了自己在工作中常用的流操作:

操作类型返回类型使用的类型/函数式接口函数描述符filter中间StreamPredicateT -> boolean

distinct中间Stream

skip中间Streamlong

limit中间Streamlong

map中间StreamFunctionT -> R

flatMap中间StreamFunctionT -> Stream

sorted中间StreamComparator(T, T) -> int

anyMatch终端booleanPredicateT -> boolean

noneMatch终端booleanPredicateT -> boolean

allMatch终端booleanPredicateT -> boolean

findAny终端Optional

findFirst终端Optional

forEach终端voidConsumerT -> void

collect终端RCollector

reduce终端OptionalBinaryOperator(T, T) -> T

count终端Optional

使用案例

假设有交易和交易员两个概念——分别是下面的Trader和Transaction,现在有个交易列表,里面记录了这些交易员在某些年份的交易。

首先,看交易员的定义package stream;

import lombok.AllArgsConstructor;

import lombok.Data;

import lombok.NoArgsConstructor;

/**

@Data

@AllArgsConstructor

@NoArgsConstructor

public class Trader {

private String name;

private String city;

}

然后,看交易的定义package stream;

import lombok.AllArgsConstructor;

import lombok.Builder;

import lombok.Data;

import lombok.NoArgsConstructor;

/**

@Data

@AllArgsConstructor

@NoArgsConstructor

public class Transaction {

private Trader trader;

private int year;

private int value;

}

上下文,有一个交易列表package stream;

import java.util.Arrays;

import java.util.List;

public class StreamExample {

public static void main(String[] args) {

Trader raoul = new Trader("Raoul", "Cambridge");

Trader mario = new Trader("Mario", "Milan");

Trader alan = new Trader("Alan", "Cambridge");

Trader brian = new Trader("Brian", "Cambridge");

List transactions = Arrays.asList(

new Transaction(brian, 2011, 300),

new Transaction(raoul, 2012, 1000),

new Transaction(raoul, 2011, 400),

new Transaction(mario, 2012, 710),

new Transaction(mario, 2012, 700),

new Transaction(alan, 2012, 950)

);

}

}

基于上述背景,这里将给出如下练习:找出2011年所有的交易并按照交易额排序(从低到高)List transactions2011 = transactions.stream()

.filter(transaction -> transaction.getYear() == 2011) //过滤出所有2011年的交易

.sorted(Comparator.comparing(Transaction::getValue)) //按照交易的金额排序

.collect(Collectors.toList()); //将所有的结果整理成列表交易员都在哪些不同的城市工作过List cities = transactions.stream()

.map(transaction -> transaction.getTrader().getCity())

.distinct()

.collect(Collectors.toList());查找所有来自Cambridge的交易员,并按照姓名排序List traders = transactions.stream()

.filter(transaction -> "Cambridge".equals(transaction.getTrader().getCity()))

.map(Transaction::getTrader)

.sorted(Comparator.comparing(Trader::getName))

.collect(Collectors.toList());将所有交易员的姓名按照字母顺序排序,并连接成一个字符串返回String nameStr = transactions.stream()

.map(transaction -> transaction.getTrader().getName())

.distinct()

.sorted()

.collect(Collectors.joining());有没有交易员是在Milan工作的?boolean milanBased = transactions.stream()

.anyMatch(transaction -> "Milan".equals(transaction.getTrader().getCity()));打印所有城市在剑桥的交易员的交易额transactions.stream()

.filter(transaction -> "Cambridge".equals(transaction.getTrader().getCity()))

.map(Transaction::getValue)

.forEach(System.out::println);所有交易中,最高的交易额是多少?Optional maxValue = transactions.stream()

.map(Transaction::getValue)

.reduce(Integer::max);将所有的交易按照年份分组,存放在一个Map中Map yearMap = transactions.stream()

.collect(Collectors.toMap(Transaction::getYear, transaction -> transaction));找到交易额最小的交易Optional minTransaction = transactions.stream()

.min(Comparator.comparing(Transaction::getValue));

参考资料

https://www.journaldev.com/2774/java-8-stream

《Java 8实战》

https://docs.oracle.com/javase/8/docs/api/java/util/stream/package-summary.html#package.description

你再主动一点点

56174c3814d6ecd4cf82688fcd0f4faf.png我们就有故事了

本号专注于后端技术、JVM问题排查和优化、Java面试题、个人成长和自我管理等主题,为读者提供一线开发者的工作和成长经验,期待你能在这里有所收获。

c5975c3d04c27a650da460aaae6b2717.png

扫码关注

有趣的灵魂在等你

ed6d99c5cc96e2bc36e42282a1f7dc06.png下方查看历史文章

0ec5bf16ea80b3f7c9b994f25381f8f7.png

1d93ace429b05d338350ec887f9224c1.png在看点一下

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值