ej3-0开端

2024-04-02 18:08
文章标签 开端 ej3

本文主要是介绍ej3-0开端,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

开始

编码多年,总有一些最佳实践,Java也是,比如设计模式,比如Effective Java 3 (ej3) 。 设计模式先后看过《大话设计模式》,《HeadFirst 设计模式》。而EffectiveJava3我打算阅读英语原版的,翻译过来,提高一下自己的英文阅读能力,同时也思考一下大师总结的编程最佳实践,应用到日常编码工作中。

    开端从ef3作者的[宣讲PPT](https://www.yuque.com/office/yuque/0/2019/pptx/186661/1574762227798-103405a7-8cec-4953-b03c-26e8eff4150c.pptx)开始。

建议先快速看一遍ppt,不过全部是英文的,我把它翻译过来,作为我开始ej3的开始。

版本变化

使用 lambdas优先匿名类


    使用匿名内部类的代码可读性和简洁性不断提高;

file

类型推断:

Collections.sort(stringList,(s1,s2)->Integer.compare(s1.length(),s2.length()));

    等价于:
`Collections.sort(stringList,(Comparator<String>)(String s1,String s2)->Integer.compare(s1.length(),s2.length()));`
    类型推断很魔幻:
1. 没有人知道规则,但是它是对的;
2. 忽略类型除非它可以让程序更清晰;
3. 编译器如果需要帮助它会告诉你;
    类型推断的警告:
1. 推断依赖于泛型信息; 比如 words被声明为List<String>,代码无法编译,如果words是一个原生的List;
2. 第二版说不要使用原生类型:惩罚是不必要的运行期bug和令人讨厌的代码;
3. 第三版本说的更严重:惩罚包括无法合适的使用lambdas;
4. 你必须理解泛型之后才能使用lambdas;

使用labmbdas的警告:

  1. 它缺少名字和文档:它应该是自解释的,它应该不超过一定行数,最好是一行;
  2. 如果它必须很长或者复杂:抽取到方法中然后使用方法引用,枚举则或者使用指定实例的类体;
  3. 匿名类任然有使用场景:lambdas需要函数式接口,并且无法访问自己,在lambdas中this指向了一个封闭的实例;

file

    把一个抽象方法的实现替换成了一个函数式接口;

方法引用优先lambdas

file

    说明:
1. 参数越多,代码块约长;但是参数可以提供文档信息;
2. 如果使用lambdas,需要小心的选择参数名字;
public class ExeTest {public static void main(String[] args) {new Thread(()->action()).start();System.out.println("====");new Thread(ExeTest::action).start();Arrays.asList("aaa","bbb","CCC").stream().map(x->x).collect(Collectors.toList());Arrays.asList("aaa","bbb","CCC").stream().map(Function.identity()).collect(Collectors.toList());}private static void action() {System.out.println("do action ");}
}

file

    写在最后:
1. 所有可以使用方法引用的地方,你都可以使用lambda替换;
2. 方法引用通常更简洁;
3. 有的时候,lambda更清晰
4. 基于你自己的判断力,你总是可以改变主意;

优先使用标准函数式接口

    jdk9中一共43个标准的函数式接口;

file

    使用标准的函数式接口的好处:
1. api更好学习降低了概念领域的学习;
2. 提供了互操作性便利:很多的标准函数式接口提供了有用的默认方法,比如Predicate提供了combine和negate;
3. 例子中方法可以使用BiPredicate来替代;
    为何如此关注比较器 Comparator?
1. 每次在api使用的时候名字提供了文档:使用频率高;
2. 在合法的实例列表中强烈需要:需要通用的比较协议,通过实现这个接口,你遵守了承诺;
3. 很多有用的默认方法可以组合和转换实例,6种形式的thenComparing和reversed;
    基于目的写函数式接口的标准:
1. 接口应该共享一个或多个比较器comparator的特征:经常使用,有个很好描述的名字,很强的关联,可以从默认方法中收益;
2. 如果你写了一个函数式接口,记住,它是一个接口:所有的接口都需要特别关注;

辩证的使用streams

    什么是流?
1. 一系列从集合,数组,输入设备等过来的数据对象,为了大量的数据处理;
2. 按照管道处理数据:一个数据源,0个或多个中间操作,一个终止操作;
3. 支持大部分函数式数据处理;
4. 支持无痛并行:简单的替换流为parallelStream:你可能看到性能提升;
    标准实现,不使用流
public static void test1(String[] args) throws FileNotFoundException {File dictionary = new File(args[0]);int minGroupSize = Integer.parseInt(args[1]);Map<String, Set<String>> groups = new HashMap<>();try (Scanner s = new Scanner(dictionary)) {// Item 9       while (s.hasNext()) {String word = s.next();groups.computeIfAbsent(alphabetize(word), (unused) -> new TreeSet<>()).add(word);}}for (Set<String> group : groups.values()) {if (group.size() >= minGroupSize) {System.out.println(group.size()   ": "   group);}}
}

    使用流,连字符串的处理都使用流
public static void test2(String[] args) throws IOException {Path dictionary = Paths.get(args[0]);int minGroupSize = Integer.parseInt(args[1]);try (Stream<String> words = Files.lines(dictionary)) {words.collect(groupingBy(word -> word.chars().sorted().collect(StringBuilder::new,(sb, c) -> sb.append((char) c),StringBuilder::append).toString())).values().stream().filter(group -> group.size() >= minGroupSize).map(group -> group.size()   ": "   group).forEach(System.out::println);}}

        使用流

---

    
    public static void test3(String[] args) throws IOException {Path dictionary = Paths.get(args[0]);int minGroupSize = Integer.parseInt(args[1]);try (Stream<String> words = Files.lines(dictionary)) {words.collect(groupingBy(Test2::alphabetize)).values().stream().filter(group -> group.size() >= minGroupSize).forEach(g -> System.out.println(g.size()   ": "   g));}}

    为何不用streas去实现alphabetize方法?
1. streams没有提供直接的支持char;
2. 实现结果:不清晰,很难写正确,很难阅读,可能更慢;
    看一下这段代码输出结果:"Hello world".chars().forEach(System.out::print);输出结果是:7210110810811132119111114108100
    因为chars得到的是一个IntStream;所以输出了整数;修正:"Hello world".chars().forEach(i-> System.out.print((char)i));
**禁止使用streams处理char;**
    一个难题: 笛卡尔产品
private static List<Card> newDeck() {List<Card> result = new ArrayList<>();for (Suit suit : Suit.values()) {for (Rank rank : Rank.values()) {result.add(new Card(suit, rank));}}return result;}private static List<Card> newDeck2() {return Stream.of(Suit.values()).flatMap(suit -> Stream.of(Rank.values()).map(rank -> new Card(suit, rank))).collect(toList());}


        写在最后:
1. 流在很多事情上非常出色,但是它不是灵丹妙药;
2. 你第一次学习流的时候,你可能会项把所有的循环转换成流式循环,别这么做,它可以让你的代码更短,但是不太简洁;
3. 一边练习一边评价,合适的使用,流可以提高简介和清晰,但是很多的程序应该结合iteration和流;
4. 并不是一开始就很清晰,如果你不懂,开始猜测然后开始探究源码,如果你感觉不对,尝试另外的方法

辩证的使用streams的并行化

    并行化不一定更快。

file

    为何例子中的并行程序跑的如此慢?
1. 流的库中没有明确的注意如何去并行执行,这个探索很痛苦的失败了;
2. 在最好的例子中,parallel没用:流的源头是stream.iteratoe或者中间操作被使用;
3. 反例:limit操作的默认策略计算超额的元素,计算素数一个的时间是另外一个的两倍时间;
4. 准则:不要无差别的并行化;
    parallelize适合的场景:
1. Arrays,ArrayList,HashMap,HashSet,ConcurrentHashMap,int,long的区间;
2. 这些数据源的公共特征:可以预见的分离,好的位置引用;
3. 终止操作也会影响:必须快速并且容易并行化:减法比如min,max,count,sum是适合的,collectors不适合;
4. 中间操作也会影响:mapping和filter非常适合,limit不适合;
    parallel()仅仅只是优化:
1. 优化需要判断;
2. 不使用parallel除非你能证明它维护正确;
3. 不适用parallel除非你确幸它能跑的更快;
4. 测量性能在使用前后;

大神的小结

  1. java比以前庞大和复杂了;
  2. 现在一个多模式的语言;
  3. 不要仅仅关注如何使用特征,还要关注使用哪些;
  4. lambdas和streams是一个巨大的成功,但是你必须批判的使用他们;
  5. java的力量越大责任越大;

非神小白的小结

ppt首先对比了3个版本的effectiveJava的区别,然后挑选了新增的章节中的5个条目进行了演示,确实让人耳目一新。Java8,9最亮的点就是lambdas和streams .要好好利用和理解。

原创不易,转载请注明出处,欢迎多沟通交流

这篇关于ej3-0开端的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/870694

相关文章

腾讯云开端口

轻量服务器 由于开发者计划,这些腾讯云 阿里云什么的小vps,是非常之便宜,甚至到了白送的地步(小阿),但是作为一个web安全学习者必要的vps操作还是要会的 开启端口 腾讯云的轻量服务器是没有安全组的,我们有的只是防火墙 来到服务器进入“查看详情” 进入防火墙添加规则 选择全部IPv4,自己加入自己想要的端口 即可进行监听端口弹shell、curl外带等等操作 Apache服务(

诺基亚Lumia930五月开端预定 约售3720元

诺基亚Lumia 930的发布不只遭到很多网友的遍及追捧,并且好像也成了运营商招引眼球的法宝。虽然该机要到六月份才会面市,但沃达丰德国公司仍是早早便宣告将于五月份开端预售诺基亚Lumia 930, 并且将会供给黑色、白色以及绿色和橙色等颜色版别可选。   依据国外网站Wmpoweruser发表的音讯称,沃达丰德国公司将于五月份便开端诺基亚Lumia 930的预定,

GPT-4o的崛起:人工智能新纪元的开端

如何评价GPT-4o? GPT-4o的崛起:人工智能新纪元的开端 随着人工智能技术的飞速发展,GPT-4o的发布再次引起了广泛关注。作为OpenAI推出的最新一代语言模型,GPT-4o不仅在技术能力上取得了重大突破,还在应用场景和用户体验上带来了显著提升。本文将对GPT-4o进行详细评价,分析其与前代版本的对比,探讨其技术能力,并分享个人的整体感受。 一、版本间的对比分析 自从GPT-3问

1-Django开端--学生管理系统

目录 项目结构 前端页面: add_data.html class_data.html index.html apps.py models.py views.py settings,py urls.py ...实现简略的身架... 项目结构 前端页面: add_data.html --添加数据. {% extends 'index/inde

linux网络编程启动!(开端)

网络设计模式 :就两种模型 b/s 模型 : 浏览器—>服务器 优点是:跨平台。开发成本低 缺点是:网络通信的时候必须要使用http/https协议 http协议 是个应用层协议 不能在磁盘缓存或者从磁盘加载大量数据 http 与https 多了一层加密 c/s模型 client→server 客户端→卓面应用程序 qq 微信 优点: 协议可以自己随意选择,因为app和

逻辑回归模型与GBDT+LR——特征工程模型化的开端

随着信息技术和互联网的发展, 我们已经步入了一个信息过载的时代,这个时代,无论是信息消费者还是信息生产者都遇到了很大的挑战: 信息消费者:如何从大量的信息中找到自己感兴趣的信息?信息生产者:如何让自己生产的信息脱颖而出, 受到广大用户的关注? 为了解决这个矛盾, 推荐系统应时而生, 并飞速前进,在用户和信息之间架起了一道桥梁,一方面帮助用户发现对自己有价值的信息, 一方面让信息能够展现在对它感

深度强化学习开端——DQN算法求解车杆游戏

深度强化学习开端——DQN算法求解车杆游戏 DQN,即深度Q网络(Deep Q-Network),是一种结合了深度学习和强化学习的算法,其主要用于解决序列决策问题,并且在许多复杂的决策任务中展现出了显著的效果。DQN算法的发明历史可以追溯到2013年,当时DeepMind团队首次提出了一种名为DQN(Deep Q-Network)的新型强化学习算法。这一算法标志着深度学习和强化学习成功结合的开始

云上的开端

博客开了有一段时间了,最初的计划是在这上面保存一些自己翻译的国外技术文章,供大家参考的。可是最近一直在做威客,有点小忙,再加上工作,就忙得有点晕头转向了,呵呵。但是某人曰过,“时间就是海绵,只有挤才能有。” 所以小弟我决心开始挤海绵,完成年初的一年计划,在这里能体现出来的就是要翻译一本小书,呵呵。本来是计划看本PYTHON的书籍的,但是无奈小弟PYTHON水平尚浅,不敢托大,遂决定先拿

[图] Android One 手机试玩 一个前景光明的开端

Android One项目是谷歌今年I/O大会上最引人关注的地方之一,在上个月,首批Android One智能手机也已经在印度正式发售。这些售价超低的智能手机品质如何?相信这是不少人所关心的问题。日前,科技网站Android Central就对其中的一款手机,Micromax Canvas A1进行了试玩体验。 什么是Android One? 虽 然Android One看上去似乎和Nexu

第1天 工作计划和开端

转自:图灵社区原文作者:刘祺原文地址:http://www.ituring.com.cn/article/215741本次转载已经过原作者授权,二次转载请自行联系原作者      合集简介市面上已经有很多关于编译原理的书了。无论是普通的程序爱好者、职业的计算机工程师还是***,自制编程语言一直被认为是最有趣的工作之一。即使自制编程语言的中途会遇到各种各样棘手的问题,但是这样一个工作绝对要比按照