这里有一个问题,connect可以指定超时时间,但是read无法指定超时时间。但是可以设置阻塞(block)时间。 正确的写法: ? 为了解决这些问题,一般采用缓存和异步/消息队列处理。 频繁使用计时器 错误代码: ? 这个代码有两个问题, 一个是没有告诉调用者, 系统调用出错了. 第二个是日志没有出错原因, 很难跟踪定位问题。 正确的写法: ? 重复包装RuntimeException 错误的写法: ?
本文为《Java Coding Problems》读书笔记。学习语言从解决问题开始,《Java Coding Problems》一书中包含了Java编程中常遇到的一些问题。 57Working With Date And Time 58-77Type Inference 78-98Arrays, Collections, And Data Structures 99-128Java I/O Paths, Files, Buffers, Scanning, And Formatting 129-148Java Reflection Classes, Interfaces, Constructors 书中以Java的方式解决诸多常见问题。后续文章中记录这些问题的解法,以此学习Java语言的常规技巧。 参考资料:https://learning.oreilly.com/library/view/java-coding-problems/9781789801415/。
原文:Java Coding Problems 协议:CC BY-NC-SA 4.0 贡献者:飞龙 本文来自【ApacheCN Java 译文集】,自豪地采用谷歌翻译。 本节介绍的问题和解决方案基于 Java 语言架构师 Brian Goetz 的定义: “Optional旨在为库方法返回类型提供一种有限的机制,在这种情况下,需要有一种明确的方式来表示无结果,并且使用null 问题 使用以下问题来测试你的Optional编程能力。 解决方案 以下各节介绍上述问题的解决方案。记住,通常没有一个正确的方法来解决一个特定的问题。另外,请记住,这里显示的解释仅包括解决问题所需的最有趣和最重要的细节。 本场景的候选对象是 Java 反射 APIMethod.invoke()(见第 7 章、“Java 反射类、接口、构造器、方法、字段”。
,而程序员思维经常被当做贬义词,因为多数情况下程序员思考问题像个计算机,并把这种思考模式带到了生活当中。 复杂到看不出问题还是简单到明显没有问题? 熟悉git吗?熟悉svn吗?他们的原理如何?最佳实践呢? 代码运行效率 统计过CPU/GPU/磁盘IO/网络IO/内存的消耗吗? 一次磁盘IO耗时多少? 耐性 中国社会由于种种问题,相对于西方发达国家来说社会整体比较浮躁、急于求成。无论一个人有多么的天才,总是需要一个积累的过程。 至少一门静态编程语言,一门动态编程语言,一门函数性语言 2. 会web编程、app编程 3. 会大数据相关的技术:存储、挖掘、分析 4. 实践 其实这只是变为优秀程序员的一个步骤而已,根据我的观察,多数人学习编程时死在了这个山头。
本章包括 21 个涉及 JEP286 或 Java 局部变量类型推断(LVTI)的问题,也称为var类型。这些问题经过精心设计,以揭示最佳实践和使用var时所涉及的常见错误。 问题 使用以下问题来测试您的类型推断编程能力。 结合 LVTI 和面向接口编程技术:编写一个程序,通过面向接口编程技术来举例说明var的用法。 结合 LVTI 和菱形运算符:编写一个程序,举例说明var和菱形运算符的用法。 9d08083bcc91.png)] 这个问题的解决方案包括使用显式 Java 字面值: // Prefer var intNumber = 10; // inferred as int var 使用var而不考虑可能的清晰度损失会产生这些问题。像这样的一些问题和代码将成为一个真正的痛苦。 83 LVTI 与面向接口编程技术相结合 Java 最佳实践鼓励我们将代码绑定到抽象。
UDP传输编程 ☆发送端 在发送端,要在数据包对象中明确目的地IP及端口。 package cn.hncu.url.udp; import java.io.IOException; import java.net.DatagramPacket; import java.net.DatagramSocket ; import java.net.DatagramPacket; import java.net.DatagramSocket; import java.net.SocketException; public ; import java.net.DatagramPacket; import java.net.DatagramSocket; import java.net.InetAddress; import ; break; } } ds.close(); } } 好了,到现在就可以实现2台联网的机子的互动了。
Executors 在Java 5之后,并发编程引入了一堆新的启动、调度和管理线程的API。 Executor框架便是Java 5中引入的,其内部使用了线程池机制,它在java.util.cocurrent 包下,通过该框架来控制线程的启动、执行和关闭,可以简化并发编程的操作。 因此,在Java 5之后,通过Executor来启动线程比使用Thread的start方法更好,除了更易管理,效率更好(用线程池实现,节约开销)外,还有关键的一点:有助于避免this逃逸问题——如果我们在构造器中启动一个线程 Java 开发者很有必要学习和理解他们,以便更高效的使用 Java 提供的不同类型的线程池。 2、无界队列。
本章包括 22 个涉及 Java 函数式编程的问题。这里,我们将重点讨论在流中遇到的涉及经典操作的几个问题(例如,filter和map),并讨论无限流、空安全流和缺省方法。 一旦您涵盖了本章和上一章,您就可以在生产应用上释放函数式编程了。下面的问题将为您准备各种各样的用例,包括角落用例或陷阱。 问题 使用以下问题来测试您的函数式编程能力。 apachecn/apachecn-java-zh/raw/master/docs/java-coding-prob/img/3417a264-ab2a-4d7e-a0c7-e1c999a487a7.png 在流和函数式编程(java8)之前,这样的任务是通过一堆繁琐、冗长且容易出错的意大利面代码应用于集合的。从 Java8 开始,我们有分组收集器。 在下一节中,我们来看看单级分组和多级分组。 从 JDK8 开始,java.util.Comparator类增加了几个新的比较器,包括用于链接比较器的thenComparing()口味。 此处的问题由应删除的选项表示。
原文:Java Coding Problems 协议:CC BY-NC-SA 4.0 贡献者:飞龙 本文来自【ApacheCN Java 译文集】,自豪地采用谷歌翻译。 本章包括涉及 Java 并发的 13 个问题,涉及 Fork/Join 框架、CompletableFuture、ReentrantLock、ReentrantReadWriteLock、StampedLock 读完本章,您将对并发性有相当的了解,这是每个 Java 开发人员都需要的。 问题 使用以下问题来测试您的并发编程能力。 thread:main [10:30:53] [INFO] Result: 10 216 CompletableFuture JDK8 通过用CompletableFuture增强Future,在异步编程领域迈出了重要的一步 我们可以通过同步来修复问题,或者通过原子变量来更好地解决问题。 原子变量类在java.util.concurrent.atomic中可用。
每天在写Java程序,其实里面有一些细节大家可能没怎么注意,这不,有人总结了一个我们编程中常见的问题。虽然一般没有什么大问题,但是最好别这样做。 另外这里提到的很多问题其实可以通过Findbugs( http://findbugs.sourceforge.net/ )来帮我们进行检查出来。 字符串连接误用 错误的写法: ? 正确的写法: ? 问题在第三行,append char比String性能要好,另外就是初始化StringBuffer没有指定size,导致中间append时可能重新调整内部数组大小。 另外一个问题不能一次就将一个xml文件用String保存,这样对内存会造成不必要的浪费,正确的做法用InputStream来边读取边处理。为了解决编码的问题, 最好使用XML解析器来处理。
错误的写法: String s = ""; for (Person p : persons) { s += ", " + p.getName(); } s = s.substring(2) 另外一个问题不能一次就将一个xml文件用String保存,这样对内存会造成不必要的浪费,正确的做法用InputStream来边读取边处理。为了解决编码的问题, 最好使用XML解析器来处理。 为了解决这些问题,一般采用缓存和异步/消息队列处理。 第二个是日志没有出错原因, 很难跟踪定位问题。 Exception e) { throw new MyRuntimeException("Could not do stuff because: "+ e.getMessage, e); } 作者:java
这里本意是希望用当前类来加载希望的对象, 但是这里的getClass()可能抛出异常, 特别在一些受管理的环境中, 比如应用服务器, web容器, Java WebStart环境中, 最好的做法是使用当前应用上下文的类加载器来加载 这里有两个错误, 一个是没有没有将毫秒归零, 不过最大的错误是没有指定TimeZone, 不过一般的桌面应用没有问题, 但是如果是服务器端应用则会有一些问题, 比如同一时刻在上海和伦敦就不一样, 因此需要指定的 关于时间的问题可以参考这篇文章: http://www.odi.ch/prog/design/datetime.php 这里主要的问题是Date对象并不包含Time Zone信息. 这个方法的本意是不支持传递超过2GB的文件. 最好的做法是对长度进行检查, 溢出时抛出异常。 正确的写法: ? 另一个溢出bug是cast的对象不对, 比如下面第一个println. 如果java编译器能针对这种情况给出警告. 或者在java语言规范中不支持浮点数类型的==操作就最好了。 正确的写法: ? 用浮点数来保存money 错误的写法: ? 这个也是一个老生常谈的错误.
在实际开发中,两者可以结合使用,以便更好地解决问题 函数式编程之所以突然兴起,是因为它具有以下优点: 易于并行处理: 由于函数式编程中的函数没有副作用,即对同样的输入始终产生相同的输出,因此可以很容易地将一个大问题分解成多个小问题 总之,在某些情况下使用函数式编程可能会比面向对象编程更加适合。但是,在其他情况下使用面向对象编程可能会更加适合。最佳选择取决于具体情况。 2. 函数式编程能代替JUC编程吗? 同时,在Java 8中引入了lambda表达式和Stream API等特性,使得函数式编程在Java中得到了更好的支持。 在实际开发中,我们通常会使用多线程来实现并发编程 4. 函数式编程如何解决线程安全问题? 函数式编程可以通过使用不可变数据和纯函数来解决线程安全问题。 总之,在函数式编程中使用不可变数据和纯函数可以有效地解决线程安全问题,并且使得程序更加容易被并行执行
本章包括 11 个涉及 Java 函数式编程的问题。我们将从一个问题开始,这个问题旨在提供从 0 到函数式接口的完整过程。 然后,我们将继续研究 GoF 中的一套设计模式,我们将用 Java 函数风格来解释这些模式。 在本章结束时,您应该熟悉函数式编程,并准备好继续处理一组问题,这些问题允许我们深入研究这个主题。 问题 使用以下问题来测试您的函数式编程能力。我强烈建议您在使用解决方案和下载示例程序之前,先尝试一下每个问题: “编写函数式接口”:编写一个程序,通过一组有意义的例子定义从 0 到函数式接口的路径。 /raw/master/docs/java-coding-prob/img/11024e3c-1a23-4e17-a15b-f2e9060caada.png)] 如果我们将每个选择条件/标准看作一种行为 175 实现级联生成器模式 我们已经在第 2 章、“对象、不变性和switch表达式”中讨论过这个模式,“通过构建器模式编写一个不可变类”部分。处理这个问题是明智的,就像快速提醒构建器模式一样。
mybatis 解决 SQL 注入问题 我们使用 mybatis 编写 SQL 语句时,难免会使用模糊查询的方法,mybatis 提供了两种方式#{}和${}。 能有效解决 SQL 注入问题 ${}表示使用拼接字符串,将接受到参数的内容不加任何修饰符拼接在 SQL 中,使用${}拼接 sql,将引起 SQL 注入问题。 2 如果将配置文件中的 SQL 语句改成#{}形式,可避免 SQL 注入。 替代了,很好地解决了 SQL 语句的问题,防止了 SQL 注入。查询结果将为空。
本章介绍的基本问题将非常有助于了解日期-时间 API 的整体情况,并将像拼图中需要拼凑起来的部分一样解决涉及日期和时间的复杂挑战。 问题 使用以下问题来测试您的日期和时间编程能力。 65 日期时间的加减 这个问题的解决方案依赖于专用于处理日期和时间的 Java API。让我们在下一节中看看它们。 使用Date 对于Date对象,解决方案可能依赖于Calendar实例。 从 JDK8 开始 新的 Java 日期时间 API 为解决这个问题提供了新的工具。 但是,假设问题要求您查找 2019 年 2 月 27 日之后的 21 天,也就是 2019 年 3 月 20 日。对于这个问题,没有预定义的策略,因此需要自定义策略。 这个问题的解决方案应该循环【2019 年 2 月 1 日,2019 年 2 月 21 日】间隔一天,并在屏幕上打印每个日期。基本上要解决两个主要问题: 一旦开始日期和结束日期相等,就停止循环。
Synchronized是Java并发编程中最常用的用于保证线程安全的方式,其使用相对也比较简单。 但是如果能够深入了解其原理,对监视器锁等底层知识有所了解,一方面可以帮助我们正确的使用Synchronized关键字,另一方面也能够帮助我们更好的理解并发编程机制,有助我们在不同的情况下选择更优的并发策略来完成任务 对平时遇到的各种并发问题,也能够从容的应对。 在JDK 1.6中对锁的实现引入了大量的优化,了解决在没有多线程竞争或基本没有竞争的场景下因使用传统锁机制带来的性能开销问题。 总之,在使用synchronized的时候,如果JIT经过逃逸分析之后发现并无线程安全问题的话,就会做锁消除. 2、轻量级锁 JVM的开发者发现在很多情况下,在Java程序运行时,同步块中的代码都是不存在竞争的,不同的线程交替的执行同步块中的代码。这种情况下,用重量级锁是没必要的。
说明:Long a = 2l; 写的是数字的 21,还是 Long 型的 2? 【推荐】不要使用一个常量类维护所有常量,要按常量功能进行归类,分开维护。 2) 应用内共享常量:放置在一方库中,通常是子模块中的 constant 目录下。 "yes"; 类 B 中:public static final String YES = "y"; A.YES.equals(B.YES),预期是 true,但实际返回为 false,导致线上问题 正例:public enum SeasonEnum { SPRING(1), SUMMER(2), AUTUMN(3), WINTER(4); private int seq; SeasonEnum
上文说到synchronized,JAVA并发编程synchronized全能王的原理,虽然被评为并发全能王,不过用起来也是格外注意,不能搞大力出奇迹那一套,容易出现性能问题。 内存屏障是一条CPU指令,用来控制在特定条件下的重排序和可见性问题。java编译器会根据内存屏障的规则禁止重排序。 4.volatile的缺点-原子性问题 比如两个线程对一个volatile修饰的count字段,进行2w次++,由于原子性问题,导致结果并不是20000. package lading.java.mutithread ; /** * volatile关键字存在原子性问题 * 对volatile修饰的count进行20000次并发+1,预期结果是20000,但由于原子性问题,与预期不符 */ public class 5.volatile怎么用更科学 像4的示例,volatile修饰的count并发++2w次,结果出现原子性问题。
import java.sql.*; import java.beans.Statement; import java.sql.Connection; import java.sql.DriverManager ; import java.sql.SQLException; public class Main{ public static void main(String[] args){ try{