【如何让代码变“高级”(二)】-这样操作值得一波666(Java Stream)(这么有趣)

“致"高级"工程师(BUG工程师)
一颗折腾的心
原创不易,点个赞💗,支持支持

开发中的代码

在开发中的代码是不是很常见这样的代码:

这样的?

for循环取元素取值

List<User> szUserList = new ArrayList<>();
 for (User user : userList) {
   if (user.getAddress().equals("shenzhen")) {
    szUserList.add(user);
   }
 }

或者这样的?

for循环去重

 for (int i = 0; i < list.size() - 1; i++) {
            for (int j = list.size() - 1; j > i; j--) {
                if (list.get(j).equals(list.get(i))) {
                    list.remove(j);
                }
            }
        }
Set set = new HashSet();
        List<String> newList = new ArrayList();
        for (Iterator iter = list.iterator(); iter.hasNext();) {
            Object element = iter.next();
            if (set.add(element))
                newList.add(element);
        }
        list.clear();
        list.addAll(newList);
      
  }

对于普通的CV族来说,这样就差不多了,功能实现了,又可以收拾包袱准备下班了.完美!!!

但对于我们"高级"CV族来,这不够,这远远的不够,我们需要保持一颗折腾的心💗,这样的代码彰显不出我们这段位的价值(青铜😀王者).

所以我们需要一种方式去提升我们代码的维度,让我们的代码变得更"高级", 使代码更加简洁并且更加语义化 .

由于"高级"CV"族保持这样的心态:

平凡的人,平凡的生活,平凡的工作

该有一颗不平凡的心💗

找出一种方式可以很好的升级这种代码问题:

Java 8 新提供给开发者的一组操作集合的 API----Stream 流

我们如何看待Stream流

那Stream流是如何来提升代码维度?

首先我们来看看Stream流处理for循环取元素取值:

List<User> szUserList=userList.stream()                              
                             .filter(user-> user.getAddress().equals("shenzhen"))                                         
                             .collect(Collectors.toList());

其实, stream流会把需要处理的元素集合看作一种流, 流在管道中传输, 并且可以在管道的节点上进行处理, 比如筛选、排序、聚合等。Stream 流可以极大的提高开发效率,也可以使用它写出更加简洁明了的代码。

那么Stream流可以分几种:

  • 顺序流 : 按照顺序对集合中的元素进行处理
  • 并行流 : 使用多线程同时对集合中多个元素进行处理

在使用并行流的时候就要注意线程安全的问题

元素流在管道中经过中间操作(intermediate operation)的处理,最后由终端操作 (terminal operation) 得到前面处理的结果。

  • 中间操作(intermediate operation): 中间操作会产生另一个流 ,( 流是一种惰性操作,所有对源数据的计算只在终止操作被初始化的时候才会执行), 而且中间操作还分无状态操作和有状态操作两种 .

    • 无状态操作 : 在处理流中的元素时,会对当前的元素进行单独处理。 (例如:过滤操作).
      • 有状态操作 : 某个元素的处理可能依赖于其他元素.( 例如:查找最小值,最大值,和排序 ).
  • 终止操作 (terminal operation):消费 Stream 流,并且会产生一个结果 . 如果一个 Stream 流被消费过了,那它就不能被重用的。

Stream流一般的执行过程可概括为:

  1. 源(Stream)
  2. 零个或多个中间操作(intermediate operation)
  3. 终止操作 (到这一步才会执行整个stream pipeline计算) (terminal operation)

源的创建方式

  • 使用Collection下的 stream() 和 parallelStream() 方法
  • 使用Stream中的静态方法:of()
List< String> createStream = new ArrayList< String>();
// 顺序流
Stream< String> stream = createStream.stream();
// 并行流
Stream< String> parallelStream = createStream.parallelStream();
// of()方法创建
Stream< String> stringStream = Stream.of(
    createStream.toArray(new String[createStream.size()]));

Intermediate操作

中间操作包括map (mapToInt, flatMap 等)、 filter、 distinct、 sorted、 peek、 limit、 skip、 parallel、 sequential、 unordered等.

常用操作解析:

  • filter : 筛选符合条件的元素后重新生成一个新的流。
  • map : 接收一个函数作为参数,该函数会被应用到每个元素上,并将其映射成一个新的元素。
  • flatMap: 接收一个函数作为参数,将流中的每个值都换成另一个流,然后把所有流连接成一个流。
  • distinct: 去重操作,将 Stream 流中的元素去重后,返回一个新的流。
  • sorted: 产生一个自然顺序排序或者指定排序条件的新流。
  • skip:跳过n元素,配合limit(n)可实现分页
  • peek: 生成一个包含原Stream的所有元素的新Stream,同时会提供一个消费函数(Consumer实例),新Stream每个元素被消费的时候都会执行给定的消费函数(一般用于重赋值那些);
  • limit: 对一个Stream进行截断操作,获取其前N个元素,如果原Stream中包含的元素个数小于N,那就获取其所有的元素;

Terminal操作

terminal操作

终止操作包括:forEach、 forEachOrdered、 toArray、 reduce、 collect、 min、 max、 count、 anyMatch、 allMatch、 noneMatch、 findFirst、 findAny、 iterator等

常用操作解析:

  • forEach: 遍历了流中的元素。(终端操作)
  • collect: 接收一个Collector实例,将流中元素收集成另外一个数据结构
  • max:获得流中最大值,比较器可以由自己定义。(终端操作)
  • min: 获得流中最小值,比较器可以由自己定义。(终端操作)
  • anyMatch : 判断 Stream 流中是否有任何符合要求的元素,如果有则返回 ture,没有返回 false。(终端操作)

该如何操作呢?

Stream操作都可以按照一般步骤进行.

比如上面的去重操作:

distinct操作:

 list=list.stream()
          .distinct()
          .collect(Collectors.toList()); 

NOTE: distinct()使用 hashCode()eqauls() 方法来获取不同的元素。因此,需要去重的类必须实现 hashCode()equals() 方法

结合filter,distinct,peek,skip,limit,collect例子:

List arrList = userList.stream().filter(user -> user.getName().equals("ccww"))//过滤
                .distinct()//去重
                .peek(user -> user.setAddress("shenzhen"))//重新赋值
                .skip(2)//跳读
                .limit(2)//读取2个元素
                .collect(Collectors.toList());

map的例子:

 List arrList1=userList.stream()
                .map(user->{
                     //todo 处理函数
                    user.setAddress(cityService.getCity());
                }).collect(Collectors.toList());
    }

现在我们主要了解了 Java 8 Stream 流的基础知识及使用,涵盖 Stream 流的分类、接口、相关 API 操作使用, 在实际开发中,一定还会有更多的应用,更多Stream详细内容,会在接下来好好文章中..
往期文章:

各位看官还可以吗?喜欢的话,动动手指点个赞💗,点个关注呗!!谢谢支持!

也欢迎关注公众号【Ccww笔记】,原创技术文章第一时间推出

image
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容