Java程序员常犯的五个错误_java

下面针对每一个错误用文字说明结合代码详解的方式展示给大家,具体内容如下:

1. Null 的过度使用

避免过度使用 null 值是一个最佳实践。例如,更好的做法是让方法返回空的 array 或者 collection 而不是 null 值,因为这样可以防止程序抛出 NullPointerException。下面代码片段会从另一个方法获得一个集合:

List<String> accountIds = person.getAccountIds();
for (String accountId : accountIds) {
 processAccount(accountId);
}

当一个 person 没有 account 的时候,getAccountIds() 将返回 null 值,程序就会抛出 NullPointerException 异常。因此需要加入空检查来解决这个问题。如果将返回的 null 值替换成一个空的 list,那么 NullPointerException 也不会出现。而且,因为我们不再需要对变量 accountId 做空检查,代码将变得更加简洁。

当你想避免 null 值的时候,不同场景可能采取不同做法。其中一个方法就是使用 Optional 类型,它既可以是一个空对象,也可以是一些值的封装。
 

Optional<String> optionalString = Optional.ofNullable(nullableString);
if(optionalString.isPresent()) {
 System.out.println(optionalString.get());
}

事实上,Java8 提供了一个更简洁的方法:
 

Optional<String> optionalString = Optional.ofNullable(nullableString);
optionalString.ifPresent(System.out::println);

Java 是从 Java8 版本开始支持 Optional 类型,但是它在函数式编程世界早已广为人知。在此之前,它已经在 Google Guava 中针对 Java 的早期版本被使用。

2. 忽视异常

我们经常对异常置之不理。然而,针对初学者和有经验的 Java 程序员,最佳实践仍是处理它们。异常抛出通常是带有目的性的,因此在大多数情况下需要记录引起异常的事件。别小看这件事,如果必要的话,你可以重新抛出它,在一个对话框中将错误信息展示给用户或者将错误信息记录在日志中。至少,为了让其它开发者知晓前因后果,你应该解释为什么没有处理这个异常。
 

selfie = person.shootASelfie();
try {
 selfie.show();
} catch (NullPointerException e) {
 // Maybe, invisible man. Who cares, anyway?
}

强调某个异常不重要的一个简便途径就是将此信息作为异常的变量名,像这样:
 

复制代码 代码如下:

try { selfie.delete(); } catch (NullPointerException unimportant) {  }

 
3. 并发修改异常

这种异常发生在集合对象被修改,同时又没有使用 iterator 对象提供的方法去更新集合中的内容。例如,这里有一个 hats 列表,并想删除其中所有含 ear flaps 的值:
 

List<IHat> hats = new ArrayList<>();
hats.add(new Ushanka()); // that one has ear flaps
hats.add(new Fedora());
hats.add(new Sombrero());
for (IHat hat : hats) {
 if (hat.hasEarFlaps()) {
 hats.remove(hat);
 }
}

如果运行此代码,ConcurrentModificationException 将会被抛出,因为代码在遍历这个集合的同时对其进行修改。当多个进程作用于同一列表,在其中一个进程遍历列表时,另一个进程试图修改列表内容,同样的异常也可能会出现。

在多线程中并发修改集合内容是非常常见的,因此需要使用并发编程中常用的方法进行处理,例如同步锁、对于并发修改采用特殊的集合等等。Java 在单线程和多线程情况下解决这个问题有微小的差别。

收集对象并在另一个循环中删除它们

直接的解决方案是将带有 ear flaps 的 hats 放进一个 list,之后用另一个循环删除它。不过这需要一个额外的集合来存放将要被删除的 hats。
 

List<IHat> hatsToRemove = new LinkedList<>();
for (IHat hat : hats) {
 if (hat.hasEarFlaps()) {
 hatsToRemove.add(hat);
 }
}
for (IHat hat : hatsToRemove) {
 hats.remove(hat);
}

使用 Iterator.remove 方法

这个方法更简单,同时并不需要创建额外的集合:
 

Iterator<IHat> hatIterator = hats.iterator();
while (hatIterator.hasNext()) {
 IHat hat = hatIterator.next();
 if (hat.hasEarFlaps()) {
 hatIterator.remove();
 }
}

使用 ListIterator 的方法

当需要修改的集合实现了 List 接口时,list iterator 是非常合适的选择。实现 ListIterator 接口的 iterator 不仅支持删除操作,还支持 add 和 set 操作。ListIterator 接口实现了 Iterator 接口,因此这个例子看起来和 Iterator 的 remove 方法很像。唯一的区别是 hat iterator 的类型和我们获得 iterator 的方式——使用 listIterator() 方法。下面的片段展示了如何使用  ListIterator.remove 和 ListIterator.add 方法将带有 ear flaps 的 hat 替换成带有sombreros 的。

IHat sombrero = new Sombrero();
ListIterator<IHat> hatIterator = hats.listIterator();
while (hatIterator.hasNext()) {
 IHat hat = hatIterator.next();
 if (hat.hasEarFlaps()) {
 hatIterator.remove();
 hatIterator.add(sombrero);
 }
}

使用 ListIterator,调用 remove 和 add 方法可替换为只调用一个 set 方法:
 

IHat sombrero = new Sombrero();
ListIterator<IHat> hatIterator = hats.listIterator();
while (hatIterator.hasNext()) {
 IHat hat = hatIterator.next();
 if (hat.hasEarFlaps()) {
 hatIterator.set(sombrero); // set instead of remove and add
 }
}

使用Java 8中的 stream 方法

在 Java8 中,开发人员可以将一个 collection 转换为 stream,并且根据一些条件过滤 stream。这个例子讲述了 stream api 是如何过滤 hats 和避免 ConcurrentModificationException 。hats = hats.stream().filter((hat -> !hat.hasEarFlaps()))

复制代码 代码如下:

 .collect(Collectors.toCollection(ArrayList::new));

Collectors.toCollection 方法将会创建一个新的 ArrayList,它负责存放被过滤掉的 hats 值。如果过滤条件过滤掉了大量条目,这里将会产生一个很大的 ArrayList。因此,需要谨慎使用。

使用 Java 8 中的 List.removeIf 方法

可以使用 Java 8 中另一个更简洁明了的方法—— removeIf 方法:
 

复制代码 代码如下:

 hats.removeIf(IHat::hasEarFlaps);

在底层,它使用 Iterator.remove 来完成这个操作。

使用特殊的集合

如果在一开始就决定使用 CopyOnWriteArrayList 而不是 ArrayList ,那就不会出现问题。因为  CopyOnWriteArrayList 提供了修改的方法(例如 set,add,remove),它不会去改变原始集合数组,而是创建了一个新的修改版本。这就允许遍历原来版本集合的同时进行修改,从而不会抛出  ConcurrentModificationException 异常。这种集合的缺点也非常明显——针对每次修改都产生一个新的集合。

还有其他适用于不同场景的集合,比如 CopyOnWriteSet 和 ConcurrentHashMap 。

关于另一个可能可能在并发修改集合时产生的错误是,从一个 collection 创建了一个 stream,在遍历 stream 的时候,同时修改后端的 collection。针对 stream 的一般准则是,在查询 stream 的时候,避免修改后端的 collection。接下来的例子将展示如何正确地处理 stream:
 

List<IHat> filteredHats = hats.stream().peek(hat -> {
 if (hat.hasEarFlaps()) {
 hats.remove(hat);
 }
}).collect(Collectors.toCollection(ArrayList::new));

peek 方法收集所有的元素,并对每一个元素执行既定动作。在这里,动作即为尝试从一个基础列表中删除数据,这显然是错误的。为避免这样的操作,可以尝试一些上面讲解的方法。

4. 违约

有时候,为了更好地协作,由标准库或者第三方提供的代码必须遵守共同的依赖准则。例如,必须遵守 hashCode 和 equals 的共同约定,从而保证 Java 集合框架中的一系列集合类和其它使用 hashCode 和 equals 方法的类能够正常工作。不遵守约定并不会产生 exception 或者破坏代码编译之类的错误;它很阴险,因为它随时可能在毫无危险提示的情况下更改应用程序行为。

错误代码可能潜入生产环境,从而造成一大堆不良影响。这包括较差的 UI 体验、错误的数据报告、较差的应用性能、数据丢失或者更多。庆幸的是,这些灾难性的错误不会经常发生。在之前已经提及了 hashCode 和equals 约定,它出现的场景可能是:集合依赖于将对象进行哈希或者比较,就像 HashMap 和 HashSet。简单来说,这个约定有两个准则:

如果两个对象相等,那么 hash code 必须相等。
如果两个对象有相同的 hash code,那么它们可能相等也可能不相等。
破坏约定的第一条准则,当你试图从一个 hashmap 中检索数据的时候将会导致错误。第二个准则意味着拥有相同 hash code 的对象不一定相等。

下面看一下破坏第一条准则的后果:

public static class Boat {
 private String name;
 Boat(String name) {
 this.name = name;
 }
 @Override
 public boolean equals(Object o) {
 if (this == o) return true;
 if (o == null || getClass() != o.getClass()) return false;
 Boat boat = (Boat) o;
 return !(name != null ? !name.equals(boat.name) : boat.name != null);
 }
 @Override
 public int hashCode() {
 return (int) (Math.random() * 5000);
 }
}

正如你所见,Boat 类重写了 equals 和 hashCode 方法。然而,它破坏了约定,因为 hashCode 针对每次调用的相同对象返回了随机值。下面的代码很可能在 hashset 中找不到一个名为 Enterprise 的boat,尽管事实上我们提前加入了这种类型的 boat:
 

public static void main(String[] args) {
 Set<Boat> boats = new HashSet<>();
 boats.add(new Boat("Enterprise"));

 System.out.printf("We have a boat named 'Enterprise' : %b\n", boats.contains(new Boat("Enterprise")));
}

另一个约定的例子是 finalize 方法。这里是官方 Java 文档关于它功能描述的引用:

finalize 的常规约定是:当 JavaTM 虚拟机确定任何线程都无法再通过任何方式访问指定对象时,这个方法会被调用,此后这个对象只能在某个其他(准备终止的)对象或类终结时被作为某个行为的结果。 finalize 方法有多个功能,其中包括再次使此对象对其他线程可用;不过 finalize 的主要目的是在不可撤消地丢弃对象之前执行清除操作。例如,表示输入/输出连接对象的 finalize 方法可执行显式 I/O 事务,以便在永久丢弃对象之前中断连接。

你可以决定在诸如文件处理器中使用 finalize 方法来释放资源,但是这种用法是很糟糕的。由于它是在垃圾回收期间被调用的,而 GC 的时间并不确定,因此 finalize 被调用的时间将无法保证。

5. 使用原始类型而不是参数化的

根据 Java 文档描述:原始类型要么是非参数化的,要么是类 R 的(同时也是非继承 R 父类或者父接口的)非静态成员。在 Java 泛型被引入之前,并没有原始类型的替代类型。Java 从1.5版本开始支持泛型编程,毫无疑问这是一个重要的功能提升。然而,由于向后兼容的原因,这里存在一个陷阱可能会破坏整个类型系统。着眼下例:
 

List listOfNumbers = new ArrayList();
listOfNumbers.add(10);
listOfNumbers.add("Twenty");
listOfNumbers.forEach(n -> System.out.println((int) n * 2));

这是一个由数字组成的列表被定义为原始的 ArrayList。由于它并没有指定类型参数,因此可以给它添加任何对象。但是最后一行将其包含的元素映射为 int 类型并乘以 2,打印出翻倍之后的数据到标准输出。

此代码编译时不会出错,但是一旦运行就会抛出运行时错误,因为这里试图将字符类型映射为整形。很显然,如果隐藏了必要信息,类型系统将不能帮助写出安全代码。

为了解决这个问题,需要为存入集合中的对象指定具体类型:

List<Integer> listOfNumbers = new ArrayList<>();

listOfNumbers.add(10);
listOfNumbers.add("Twenty");

listOfNumbers.forEach(n -> System.out.println((int) n * 2)); 

与之前代码的唯一差别即是定义集合的那一行:

复制代码 代码如下:

 List<Integer> listOfNumbers = new ArrayList<>();

修改之后的代码编译不可能被通过,因为这里试图向只期望存储整形的集合中添加字符串。编译器将会显示错误信息,并指向试图向列表中添加 Twenty 字符的那一行。参数化泛型类型是个不错的主意。这样的话,编译器就能够检查所有可能的类型,从而由于类型不一致而导致的运行时异常几率将大大降低。

主要总结了以上五个Java程序员常犯的错误,希望大家能够喜欢。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索java常见错误
java常见异常
程序员犯罪、程序员犯错、java程序员年终总结、java程序员、java程序员培训费用,以便于您获取更多的相关知识。

时间: 2025-01-19 15:04:58

Java程序员常犯的五个错误_java的相关文章

PHP程序员常犯的11个错误

1.使用myisam而不是innodb mysql教程有很多数据库教程引擎,但是你最可能碰到的就是myisam和innodb. mysql默认使用的是myisam.但是,很多情况下这都是一个很糟糕的选择,除非你在创建一个非常简单抑或实验性的数据库.外键约束或者事务处理对于数据完整性是非常重要的,但myisam都不支持这些.另外,当有一条记录在插入或者更新时,整个数据表都被锁定了,当使用量增加的时候这会产生非常差的运行效率. 结论很简单:使用innodb.   2.使用php教程的mysql函数

Java程序员在写SQL程序时候常犯的10个错误

  Java程序员编程时需要混合面向对象思维和一般命令式编程的方法,能否完美的将两者结合起来完全得依靠编程人员的水准: 技能(任何人都能容易学会命令式编程) 模式(有些人用"模式-模式",举个例子,模式可以应用到任何地方,而且都可以归为某一类模式) 心境(首先,要写个好的面向对象程序是比命令式程序难的多,你得花费一些功夫) 但当Java程序员写SQL语句时,一切都不一样了.SQL是说明性语言而非面向对象或是命令式编程语言.在SQL中要写个查询语句是很简单的.但在Java里类似的语句却不

作为Java程序员应该掌握的10项技能_java

本文详细罗列了作为Java程序员应该掌握的10项技能.分享给大家供大家参考.具体如下: 1.语法:必须比较熟悉,在写代码的时候IDE的编辑器对某一行报错应该能够根据报错信息知道是什么样的语法错误并且知道任何修正. 2.命令:必须熟悉JDK带的一些常用命令及其常用选项,命令至少需要熟悉:appletviewer.HtmlConverter.jar.java.javac.javadoc.javap.javaw.native2ascii.serialver,如果这些命令你没有全部使用过,那么你对jav

Java程序员应该遵守的10条纪律_java

有哪些"纪律"是Java程序员所要遵守的? 1. 为代码添加注释(Add comments to your code). – 每个人都知道这一点,但不是每个人都会这么做.你有多少次"忘记"添加注释了?确实,注释不会为你的程序增加任何函数功能.但是,有多少次,看到2周前写的代码,你都记不起它是干什么的?你很幸运,那些未注释的代码是你自己写的,你脑海中还会有残存的印象.非常不幸,大多时候,代码是别人写的,并且那个人很可能已经离开公司了.有句谚语说的好:"有来有

Java程序员面试中的多线程问题总结_java

很多核心 Java 面试题来源于多线程(Multi-Threading)和集合框架(Collections Framework),理解核心线程概念时,娴熟的实际经验是必需的.这篇文章收集了 Java 线程方面一些典型的问题,这些问题经常被高级工程师所问到. 0.Java 中多线程同步是什么? 在多线程程序下,同步能控制对共享资源的访问.如果没有同步,当一个 Java 线程在修改一个共享变量时,另外一个线程正在使用或者更新同一个变量,这样容易导致程序出现错误的结果. 1.解释实现多线程的几种方法?

C++程序员容易犯的十个C#错误

c++|程序|程序员|错误 我们知道, C#的语法与C++非常相似,实现从C++向C#的转变,其困难不在于语言本身,而在于熟悉.NET的可管理环境和对.NET框架的理解. 尽管C#与C++在语法上的变化是很小的,几乎不会对我们有什么影响,但有些变化却足以使一些粗心的C++编程人员时刻铭记在心.在本篇文章中我们将讨论C++编程人员最容易犯的十个错误. 错误1: 析构函数上的差异 几乎可以完全肯定地说,对于大多数C++编程人员而言,C#与C++最大的不同之处就在于垃圾收集.这也意味着编程人员再也无需

机器学习入门阶段程序员易犯的5个错误

怎样进入机器学习领域没有定式.我们的学习方式都有些许不同,学习的目标也因人而异. 但一个共同的目标就是要能尽快上手.如果这也是你的目标,那么这篇文章为你列举了程序员们在通往机器学习高手道路上常见的五种错误. 1.将机器学习看得高不可攀 机器学习不过是另一堆技术的集合,你可以用它来解决复杂问题.这是一个飞速发展的领域,因此,机器学习的学术交流一般出现在学术期刊及研究生的课本里,让它看起来高不可攀又难于理解. 要想高效掌握机器学习,我们需要转变观念,从技术转到方法,由精确变为"足够好",这

Java程序员需要了解的五种开源协议

五种开源协议的比较(BSD,Apache,GPL,LGPL,MIT). 当Adobe.Microsoft.Sun等一系列巨头开始表现出对"开源"的青睐时,"开源"的时代即将到来! 现今存在的开源协议很多,而经过Open Source Initiative组织通过批准的开源协议目前有58种(http://www.opensource.org/licenses/alphabetical).我们在常见的开源协议如BSD, GPL, LGPL,MIT等都是OSI批准的协议.

Java程序员容易犯的常见十大错误

1. Array 转 ArrayList 一般开发者喜欢用: List<String> list = Arrays.asList(arr);  Arrays.asList() 会返回一个ArrayList,这是Arrays里内嵌的一个私有静态类,而并不是java.util.ArrayList类 java.util.Arrays.ArrayList 有set(), get(), contains()方法,但并支持添加元素,所以大小是固定的,想要创建一个真正的ArrayList,你应该: Arra