空指针异常与Optional类
一、什么是空指针异常
当程序需要对象实例的时候返回null
就会抛出空指针异常(NullPointerException
,简称NPE)。包括以下情况:
- 调用一个
null
对象实例的方法 - 访问或修饰
null
对象的字段 - 获取数组为
null
时的长度 - 访问或修饰数组为
null
时的索引值 - 抛出
Throwable
对象为null
时的异常
虽然代码很难万无一失地避免所有NPE,但是也要尽量减少。所以一些防御性的编程技巧,可以将NPE控制在一个很好的水平上。
空指针案例
1. 调用业务方法的返回值对象
在不清楚一个方法的返回值是否存在返回null
的情况,直接使用对象返回值。
People people = new People();
People user = people.getUser("name");
String name = user.getName();
System.out.println(name);
上面示例中的
people.getUser("name");
调用返回的对象不清楚是否为null
,后面直接调用该对象的方法造成NPE。
2. 包装类自动拆箱的值
在包装类对象的值为null
的情况下,进行自动拆箱操作。
Integer a = null;
int b = a;// System.out.println(1 == a);
上面示例中包装类对象
a
定义时的初始化值为null
,在将a
赋值给基本数据类型的b
的时候,以及与基本数据类型1
进行相等逻辑操作的时候,都进行了自动拆箱操作,a
为null
时造成NPE。
3. 集合和数组空对象的遍历
在不清楚集合或数组是否为null
的时候,对它们进行遍历操作。
List<String> list = null;// String[] list = null;
for (String string : list) {
System.out.println(string);
}
在方法返回或者自己定义的数组和集合,只要有
null
的情况(不包括数组和集合长度为0的情况),进行遍历操作时造成NPE。
4. Spring没注入实例的使用
在使用Spring框架时,如果注入对象实例失败,此时该对象也是null
。
public class BeanExample {
@Autowired
private BeanProvider beanProvider;
public void run() {
this.beanProvider.sayHello(this.name, this.age);
}
}
当因为操作不当导致
beanProvider
没有注入,在调用sayHello()
方法的时候造成NPE。
5. ConcurrentHashMap和Hashtable的值
对某些不支持null
值的集合添加null
值元素,比如ConcurrentHashMap
和Hashtable
。
ConcurrentHashMap<String, String> map = new ConcurrentHashMap();
map.put("a", null);
Hashtable<String, String> hashtable = newHashtable<>();
hashtable.put("a", null);
这些集合的底层
put(K key,V value)
方法中,在key
或者value
为null
的情况下会造成NPE。
二、怎样防止空指针异常
既然NPE难以避免,我们就要去找各种方法来解决。既要有良好的编码习惯,也要细心的去把控业务。
1. 普通处理
在针对调用业务方法进行NPE普通地防御,可以简单的添加非空判断。
People people = new People();
People user = people.getUser("name");
if (user != null) {
String name = user.getName();
System.out.println(name);
}
2. 定义对象时的初始化
在自己定义对象的时候,注意初始化的值可不可以为null
。
// String str = "";
// 初始化为空字符串People people = newPeople();
// 初始化为对象
People user = people.getUser("name");
3. 使用 equals() 方法注意
已知非空对象为调用方,比如将常量值、枚举值作为调用方,避免使用未知对象去调用方法,可有效避免NPE。
String str = "123", string = null;
System.out.println(str.equals(string));
4. 使用 valueOf() 代替 toString() 方法
使用toString()
方法要利用对象去调用方法,而对象在不清楚是否为null
的情况下,会抛出NPE。使用valueOf()
方法可以避免使用未知对象去调用方法来避免。
People people = null;
System.out.println(String.valueOf(people)); // print:null
System.out.println(people.toString()); // NPE
5. 使用开源库非空判断方法
推荐使用各大开源库的StringUtils
字符串和CollectionUtils
集合等工具进行非空判断。
String str = null;
List<String> list = null;
if (!StringUtils.isEmpty(str)) {
System.out.println(str);
}
if (!CollectionUtils.isEmpty(list)) {
System.out.println(list);
}
6. 方法返回空集合或空数组
在方法中返回空数组和空集合而不是返回null
,JDK自带的Collections
集合工具类提供了多种空集合的定义。
public People[] getUsersArr() {
return new People[]{};
}
public List<People> getUsers() {
// return Collections.emptyMap();
// return Collections.emptySet();
return Collections.emptyList();
}
7. 定义数据库字段是否为空
在一些特定字段根据业务确定是否可为空,以及合理设置默认值。比如:表示业务状态的字段。
CREATE TABLE user{ ... status NOT NULL DEFAULT 0 ...}
8. 使用JDK1.8的Optional类
在JDK1.8后提供了防止NPE特定的容器,后面会讲到。
三、Optional
Optional
是一个可以包含null
或者非null
的容器对象。根据源码分析方法功能:
1. 定义方法
1.1 empty
返回一个空的Optaional
实例,在这个实例中没有值存在。
public static<T> Optional<T> empty() {
@SuppressWarnings("unchecked")
Optional<T> t = (Optional<T>) EMPTY;
return t;
}
1.2 of
返回一个值不能为空的Optional
实例,在这个实例中值为null
时抛出NPE。
public static <T> Optional<T> of(T value) {
return new Optional<>(value);
}
private Optional(T value) {
this.value = Objects.requireNonNull(value);
}
1.3 ofNullable
返回一个值可以为空的Optional
实例,在这个实例中值为null
时返回一个空的Optaional
实例。
public static <T> Optional<T> ofNullable(T value) {
return value == null ? empty() : of(value);
}
2. 功能方法
2.1 isPresent
如果有值存在,返回true
,否则返回false
。
People people = new People();
System.out.println(Optional.ofNullable(people).isPresent());// print: true
people = null;
System.out.println(Optional.ofNullable(people).isPresent());// print: false
2.2 get()
如果有值存在,返回值,否则抛出NoSuchElementException
。
People people = new People();
System.out.println(Optional.ofNullable(people).get());// print: People@61bbe9ba
people = null;
System.out.println(Optional.ofNullable(people).get());// print: Exception in thread "main" java.util.NoSuchElementException: No value present
尽量不要使用该方法获取对象。
2.3 orElse
如果有值存在,返回值,否则返回该 orElse
方法的参数,可以用来定义默认值。
String str = null;
String result = Optional.ofNullable(str).orElse("default");//print:default
System.out.println(result);
2.4 orElseGet
如果有值存在,返回值,否则返回提供者函数,可以用函数返回值来定义默认值。
String str = null;
String result = Optional.ofNullable(str).orElseGet(() -> "ajn");//print:ajn
System.out.println(result);
2.5 orElseThrow
如果有值存在,返回值,否则返回函数接口参数提供的异常。
String str = null;
String result = Optional.ofNullable(str).orElseThrow(IllegalArgumentException::new);// print: Exception in thread "main" java.lang.IllegalArgumentException
System.out.println(result);
关于更多函数接口的内容,关注Java函数式编程
2.6 ifPresent
如果有值存在,方法参数提供的函数接口会进行处理,否则不做任何操作。
Optional.ofNullable(getPeople()).ifPresent(people -> {
System.out.println("the people is nut null: " + people);
});
上面代码等价于:
People people = getPeople();
if (people != null) {
System.out.println("the people is nut null: " + people);
}
2.7 filter
如果有值存在,并且值符合给定的函数条件,返回当前Optional
,否则返回一个空的Optaional
实例,可以用来过滤特殊值。
String name = "AiJiangnan";// 给定的条件是字符串包含Ai
String result = Optional.of(name).filter(str -> str.contains("Ai")).orElse("字符串不包含Ai");
System.out.println(result);
2.8 map
如果有值存在,可以将该值的类型转换成其他类型,并返回转换后类型的Optional
实例,否则返回一个空的Optaional
实例,可以链式判空,非常实用。
People people = null;
String result = Optional.ofNullable(people)
.map(People::getName)
.map(String::toUpperCase)
.orElse("default");
System.out.println(result);
只有当
people
对象不为null
,并且people.getName()
不为null
,才返回name全部转换为大写的字符串,否则都返回 default。
2.9 flatMap
如果有值存在,可以将该值的类型转换成其他类型,但最终只能转成 Optional
实例,否则返回一个空的Optaional
实例。该方法与map
方法类似,只是该方法返回的Optional
实例由函数参数返回。
People people = new People();
people.setName(" ajn ");
String result = Optional.ofNullable(people)
.flatMap(peo -> Optional.ofNullable(peo.getName()))
.flatMap(str -> Optional.of(str.toUpperCase()))
.orElse("default");
System.out.println(result);
空指针异常与Optional类的更多相关文章
- Java8新特性——Optional类的使用(有效的避免空指针异常)
OPtional类的使用 概述 到目前为止,臭名昭著的空指针异常是导致Java应用程序失败的最常见原因.以前,为了解决空指针异常,Google公司著名的Guava项目引入了Optional类,Guav ...
- Java 8 Optional类使用的实践经验
前言 Java中空指针异常(NPE)一直是令开发者头疼的问题.Java 8引入了一个新的Optional类,使用该类可以尽可能地防止出现空指针异常. Optional 类是一个可以为null的容器对象 ...
- Java8 新特性 Optional 类
Optional 类的简介 Optional类的是来自谷歌Guava的启发,然后就加入到Java8新特性中去了.Optional类主要就是为子决解价值亿万的错误,空指针异常. Optional ...
- 【Java8新特性】不了解Optional类,简历上别说你懂Java8!!
写在前面 最近,很多读者出去面试都在Java8上栽了跟头,事后自己分析,确实对Java8的新特性一知半解.然而,却在简历显眼的技能部分写着:熟练掌握Java8的各种新特性,能够迅速使用Java8开发高 ...
- Java8新特性之空指针异常的克星Optional类
Java8新特性系列我们已经介绍了Stream.Lambda表达式.DateTime日期时间处理,最后以"NullPointerException" 的克星Optional类的讲解 ...
- 【Java8新特性】Optional类在处理空值判断场景的应用 回避空指针异常 编写健壮的应用程序
一.序言 空值异常是应用运行时常见的异常,传统方式为了编写健壮的应用,常常使用多层嵌套逻辑判断回避空指针异常.Java8新特性之Optional为此类问题提供了优雅的解决方式. 广大程序员朋友对空值异 ...
- (八)jdk8学习心得之Optional类
八.Optional 类 1. 作用:可以存放空指针null,主要用于解决空指针问题. 2. 使用方法 1) 创建对象(2种方法) Optional optional = Optional.of(非n ...
- Java 8 Optional 类
转自:https://www.runoob.com/java/java8-optional-class.html Optional 类是一个可以为null的容器对象.如果值存在则isPresent() ...
- Java8 Optional类
概述 到目前为止,著名的NullPointerException是导致Java应用程序失败的最常见原因.过去,为了解决空指针异常,Google公司著名的Guava项目引入了Optional类,Guav ...
随机推荐
- zabbix4.2配置监控TCP连接状态
1.使用命令查看TCP连接状态 (1)过去常用命令:netstat -antp [root@ansible-control zabbix]# netstat -antp Active Internet ...
- IE浏览器复选框遍历不兼容问题
obj = document.getElementsByName("userIdCheckbox"); ids = []; for(var k=0;k<obj.length; ...
- 1、Two Sum
Given an array of integers, return indices of the two numbers such that they add up to a specific ta ...
- Learn from Niu 2020.1.13
观念: 1. 把可视化的东西拾起来, 毕竟是做那个出身的. 2. 可视化里面时序数据,时空数据一直都是非常重要的. 3. know your data永远是最重要的一步, 我想更好的方式是,数据驱动, ...
- 参考 ZTree 加载大数据量。加载慢问题解析
参考 ZTree 加载大数据量. 1.一次性加载大数据量加载说明 1).zTree v3.x 针对大数据量一次性加载进行了更深入的优化,实现了延迟加载功能,即不展开的节点不创建子节点的 DOM. 2) ...
- python3练习100题——025
原题链接:http://www.runoob.com/python/python-exercise-example25.html 题目:求1+2!+3!+...+20!的和. 我的代码: s =[] ...
- 【蓝桥杯/算法训练】Sticks 剪枝算法
剪枝算法 大概理解是通过分析问题,发现一些判断条件,避免不必要的搜索.通常应用在DFS 和 BFS 搜索算法中:剪枝策略就是寻找过滤条件,提前减少不必要的搜索路径. 问题描述 George took ...
- vs2015制作一个超级简单的MVC项目
使用vs2015制作一个超级简单的MVC项目 本文链接:https://blog.csdn.net/qq_40919762/article/details/100705314 直奔主题一,创建一个 ...
- golang中的net/rpc包
本文先介绍RPC,然后go原生对RPC的使用,之后是介绍go语言中有哪些RPC框架以及一些其他常见的框架,最后是探究go语言中rpc的源码. (1)首先介绍下什么RPC? (2)RPC可以做什么? ( ...
- DOM的方法和属性
HTML DOM 方法是我们可以在节点(HTML 元素)上执行的动作. HTML DOM 属性是我们可以在节点(HTML 元素)设置和修改的值. 编程接口 可通过 JavaScript (以及其他编程 ...