告别空指针让代码变优雅,Optional使用图文例子源码解读
一、前言
我们在开发中最常见的异常就是NullPointerException
,防不胜防啊,相信大家肯定被坑过!
这种基本出现在获取数据库信息中、三方接口,获取的对象为空,再去get出现!
解决方案当然简单,只需要判断一下,不是空在去后续操作,为空返回!
所有在JDK8时出现了专门处理的方案,出来很早了,但是小编惭愧一直没有去使用它!
最近在看《Java开发手册》,一直想着提高自己的代码水平,文中就指出了使用Optional
来解决NullPointerException
!
二、Java开发手册规范
小编使用的是2022版的黄山版,29页写到:
【推荐】
防止 NPE,是程序员的基本修养,注意 NPE 产生的场景:
- 返回类型为基本数据类型,return 包装数据类型的对象时,自动拆箱有可能产生 NPE
反例:public int method() { return Integer 对象; },如果为 null,自动解箱抛 NPE。
- 数据库的查询结果可能为 null。
- 集合里的元素即使 isNotEmpty,取出的数据元素也可能为 null。
- 远程调用返回对象时,一律要求进行空指针判断,防止 NPE。
- 对于 Session 中获取的数据,建议进行 NPE 检查,避免空指针。
- 级联调用 obj.getA().getB().getC();一连串调用,易产生 NPE。
正例:使用 JDK8 的 Optional
类来防止 NPE 问题。
这份手册还是不错的,推荐反复阅读,虽然进不去大厂,也要自觉约束自己的代码风格,努力向大厂靠!
大家现在不知道哪里找的可以下载一下:
《Java开发手册》
三、Optional常用方法
小编带大家一起从api文档中的方法,一个个带大家慢慢去了解它!
1. empty()
返回一个空的Optional实例:Optional.empty
Optional<Object> empty = Optional.empty();
log.info("empty值:{}",empty);
2. of(T value)
传入一个参数,返回一个
Optional
对象,如果参数为空,报NullPointerException
!
Test testNew = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional对象:{}",optionalNew);
Optional<Test> optional = Optional.of(test);
源码查看:
我们看到参数为空会报NullPointerException
,我们去方法内部看一下就明白了:
public static <T> Optional<T> of(T value) {return new Optional<>(value);
}
private Optional(T value) {this.value = Objects.requireNonNull(value);
}
public static <T> T requireNonNull(T obj) {if (obj == null)throw new NullPointerException();return obj;
}
我们发现是在Objects
类中的requireNonNull
方法中判断了是否为空!
这个还会出现NullPointerException
,所以我们一般使用下面的这个方法!
3. ofNullable(T value)
参数传入一个对象,返回一个Optional对象,如果为空,将返回一个空的Optional对象,就等于Optional.empty
Test testNew = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional对象:{}",optionalNew);Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional对象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional对象中的ofNullable方法new返回值:{}",optionalTestNew);
源码查看:
我们发现是在方法开始进行非空判断,再去调用上面的of(T value)
方法
public static <T> Optional<T> ofNullable(T value) {return value == null ? empty() : of(value);
}
4. get()
如果此Optional中存在值,则返回该值,否则抛出
NoSuchElementException
。
Test testNew = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional对象:{}",optionalNew);
// Optional<Test> optional = Optional.of(test);Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional对象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional对象中的ofNullable方法new返回值:{}",optionalTestNew);Test test2 = optionalTestNew.get();
log.info("原来有值的:经过Optional包装后get后得到原来的值:{}",test2);
Test test1 = optionalTest.get();
log.info("原来没有值的:经过Optional包装后get后得到原来的值:{}",test1);
源码查看:
调用开始会进行值判断,如果为空则抛异常!
public T get() {if (value == null) {throw new NoSuchElementException("No value present");}return value;
}
5. isPresent()
如果存在值,则返回true,否则返回false。
这里代码就不加上面的,大家参考上面的获取一个Optional对象
boolean present = optionalTestNew.isPresent();
log.info("optionalTestNew调用是否为空:{}",present);
boolean present1 = optionalTest.isPresent();
log.info("optionalTest调用是否为空:{}",present1);
源码查看:
这就比较简单了!
public boolean isPresent() {return value != null;
}
6. ifPresent(Consumer<? super T> consumer)
如果存在值,则使用该值调用指定的使用者,否则不执行任何操作。
主要的就是入参数一个函数式接口,有值就会去执行,为空则不进行任何操作!
小技巧:
开始对lambda不了解时,可以先按照上面这种方式进行写,
大家可以看到Idea给置灰了,就是可以优化,我们Alt+Enter
然后再次Enter
就会变成后面的lambda!
optionalTest.ifPresent(new Consumer<Test>() {@Overridepublic void accept(Test test) {log.info("我是调用ifPresent执行后的打印=====");}
});
optionalTestNew.ifPresent(testInner -> log.info("我是调用ifPresent执行后的打印"));
源码查看:
还是先判断不为空才去执行函数式接口!
public void ifPresent(Consumer<? super T> consumer) {if (value != null)consumer.accept(value);
}
7. filter(Predicate<? super T> predicate)
如果存在值,并且该值符合规则,则返回描述该值的Optional,否则返回空Optional
是一个Predicate函数接口,可以传入实现了Predicate接口的lambda表达式!
如果不符合条件就会返回一个Optional.empty
testNew.setName("萧炎");
testNew.setAge(33);
Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);
log.info("过滤后的结果:{}",optionalTest1.get());
源码查看:
就是判断一下表达式和值是否为空,然后就是根据规则判断
public Optional<T> filter(Predicate<? super T> predicate) {Objects.requireNonNull(predicate);if (!isPresent())return this;elsereturn predicate.test(value) ? this : empty();
}
8. map(Function<? super T,? extends U> mapper)
如果存在值,则将提供的映射函数应用于该值,如果结果为非空,则返回描述结果的Optional。否则,返回空的Optional。
也是一个函数式接口!
Optional<String> stringOptional = optionalTestNew.map(Test::getName);
log.info("map后获得字段值:{}",stringOptional.get());
源码查看:
也是进行非空判断,然后执行lambda得到字段后放到ofNullable方法中!
public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {Objects.requireNonNull(mapper);if (!isPresent())return empty();else {return Optional.ofNullable(mapper.apply(value));}
}
9. flatMap(Function<? super T,Optional> mapper)
如果存在值,则将提供的Optional方位映射函数应用于该值,返回该结果,否则返回空的Optional。此方法类似于map,但提供的映射器的结果已经是可选的,并且如果调用,flatMap不会不会在最后进行任何包装。
Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);
log.info("flatMap后得到的字段:{}",optional.get());private static Optional<String> getFlatMap(Test test){return Optional.ofNullable(test).map(Test::getName);
}
源码查看:
也是进行非空判断,然后和map不同的是不执行ofNullable方法
public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {Objects.requireNonNull(mapper);if (!isPresent())return empty();else {return Objects.requireNonNull(mapper.apply(value));}
}
10. orElse(T other)
如果有值则将其返回,否则返回指定的其它值。
如果你是一个对象,orElse()
也要是相同对象!
String message = null;
String messageNew = "关注公众号:小王博客基地";String nullString = Optional.ofNullable(message).orElse("这是一个空字符串!");
log.info("这是空字符串打印的:{}",nullString);
String string = Optional.ofNullable(messageNew).orElse("=====这是一个空字符串!");
log.info("这是字符串打印的:{}",string);
源码查看:
简单的为空返回自己定义的,不为空直接返回!
public T orElse(T other) {return value != null ? value : other;
}
11. orElseGet(Supplier<? extends T> other)
返回值(如果存在),否则调用other并返回该调用的结果。
区别:
orElse方法将传入的参数作为默认值,orElseGet方法可以接受Supplier接口的实现用来生成默认值
如果没有复杂操作,Idea也会提醒我们不要使用这个,使用orElse即可!
String message = null;
String messageNew = "关注公众号:小王博客基地";
String orElseGet = Optional.ofNullable(message).orElseGet(() -> "这还是一个空的字符串");
log.info("orElseGet调用:这是空字符串打印的:{}",orElseGet);
String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "这还是一个空的字符串");
log.info("orElseGet调用:这是字符串打印的:{}",orElseGetString);
源码查看:
和orElse一样,只不过为空调用lambda执行!
public T orElseGet(Supplier<? extends T> other) {return value != null ? value : other.get();
}
12. orElseThrow(Supplier<? extends X> exceptionSupplier)
返回包含的值(如果存在),否则抛出由提供的供应商创建的异常。
String message = null;
String messageNew = "关注公众号:小王博客基地";
Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));
Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));
我们可以自定义异常,然后来引用!
源码查看:
为空则走自己写的异常!
public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {if (value != null) {return value;} else {throw exceptionSupplier.get();}
}
13. 例子汇总
/*** @author wangzhenjun* @date 2023/2/27 10:22*/
@Slf4j
public class OptionalTest {public static void main(String[] args) {Optional<Object> empty = Optional.empty();log.info("empty值:{}",empty);Test testNew = new Test();Test test = null;Optional<Test> optionalNew = Optional.of(testNew);log.info(" optional对象:{}",optionalNew);
// Optional<Test> optional = Optional.of(test);Optional<Test> optionalTest = Optional.ofNullable(test);log.info(" optional对象中的ofNullable方法返回值:{}",optionalTest);Optional<Test> optionalTestNew = Optional.ofNullable(testNew);log.info(" optional对象中的ofNullable方法new返回值:{}",optionalTestNew);Test test2 = optionalTestNew.get();log.info("原来有值的:经过Optional包装后get后得到原来的值:{}",test2);// Test test1 = optionalTest.get();// log.info("原来没有值的:经过Optional包装后get后得到原来的值:{}",test1);boolean present = optionalTestNew.isPresent();log.info("optionalTestNew调用是否为空:{}",present);boolean present1 = optionalTest.isPresent();log.info("optionalTest调用是否为空:{}",present1);optionalTest.ifPresent(new Consumer<Test>() {@Overridepublic void accept(Test test) {log.info("我是调用ifPresent执行后的打印=====");}});optionalTestNew.ifPresent(testInner -> log.info("我是调用ifPresent执行后的打印"));testNew.setName("萧炎");testNew.setAge(33);Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);log.info("过滤后的结果:{}",optionalTest1.get());Optional<String> stringOptional = optionalTestNew.map(Test::getName);log.info("map后获得字段值:{}",stringOptional.get());Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);log.info("flatMap后得到的字段:{}",optional.get());String message = null;String messageNew = "关注公众号:小王博客基地";String nullString = Optional.ofNullable(message).orElse("这是一个空字符串!");log.info("这是空字符串打印的:{}",nullString);String string = Optional.ofNullable(messageNew).orElse("=====这是一个空字符串!");log.info("这是字符串打印的:{}",string);String orElseGet = Optional.ofNullable(message).orElseGet(() -> "这还是一个空的字符串");log.info("orElseGet调用:这是空字符串打印的:{}",orElseGet);String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "这还是一个空的字符串");log.info("orElseGet调用:这是字符串打印的:{}",orElseGetString);Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("为空了,还不看看!"));}private static Optional<String> getFlatMap(Test test){return Optional.ofNullable(test).map(Test::getName);}}
四、总结
这里就不在演示实战了,基本上组合使用:
Optional.ofNullable(需要判断的对象).ifPresent(具体操作)
其实和if相比就是显得优雅一些,主要是防止某处没考虑到,忘记if判断,那么后续可能会导致空指针,如果使用Optional的话,那么这个问题能够得到避免。
就像多使用设计模式一样,让自己的代码更加健壮优雅,还是要多使用一些的!当然不能过渡使用!!
对你有帮助,还请不要吝啬你的发财小手点点关注哈!、
写作不易,大家给点支持,你的支持是我写作的动力哈!
关注小编的微信公众号,一起交流学习!文章首发看哦!
相关文章:
告别空指针让代码变优雅,Optional使用图文例子源码解读
一、前言 我们在开发中最常见的异常就是NullPointerException,防不胜防啊,相信大家肯定被坑过! 这种基本出现在获取数据库信息中、三方接口,获取的对象为空,再去get出现! 解决方案当然简单,只…...
【C++】哈希——unordered系列容器|哈希冲突|闭散列|开散列
文章目录一、unordered系列关联式容器二、哈希概念三、哈希冲突四、哈希函数五、解决哈希冲突1.闭散列——开放定址法2.代码实现3.开散列——开链法4.代码实现六、结语一、unordered系列关联式容器 在C98中,STL提供了底层为红黑树结构的一系列关联式容器,…...
mysql-面试
锁: mysql的锁分为全局锁、表锁、行锁、间隙锁 全局锁:Flush tables with read lock 可以全局设计库为只读 表锁:一种是表锁,一种是元数据锁(meta data lock,MDL) lock tables t1 read,t2 wi…...
【夏虫语冰】Win10局域网下两台电脑无法ping通: 无法访问目标主机
文章目录1、简介2、修改高级共享设置3、启用防火墙规则4、局域网内的其他主机访问NAT模式下的虚拟机4.1 虚拟机网络设置4.2 访问测试4.2.1 http测试4.2.2 curl测试4.2.3 telnet测试4.2.4 端口占用测试5、其他结语1、简介 ping 192.168.31.134ping主机ip时,访问无法…...
大数据框架之Hadoop:MapReduce(三)MapReduce框架原理——Join多种应用
3.7.1Reduce Join 1、工作原理 Map端的主要工作:为来自不同表或文件的key/value对,打标签以区别不同来源的记录。然后用连接字段作为key,其余部分和新加的标志作为value,最后进行输出。 Reduce端的主要工作:在Reduc…...
SSRF漏洞原理、危害以及防御与修复
一、SSRF漏洞原理漏洞概述SSRF(Server-side Request Forge,服务端请求伪造)是一种由攻击者构造形成由服务端发起请求的安全漏洞。一般情况下,SSRF攻击的目标是从外网无法访问的内部系统。正是因为它是由服务端发起的,所…...
CV学习笔记-ResNet
ResNet 文章目录ResNet1. ResNet概述1.1 常见卷积神经网络1.2 ResNet提出背景2. ResNet网络结构2.1 Residual net2.2 残差神经单元2.3 Shortcut2.4 ResNet50网络结构3. 代码实现3.1 Identity Block3.2 Conv Block3.3 ResNet网络定义3.4 整体代码测试1. ResNet概述 1.1 常见卷积…...
百亿数据,毫秒级返回查询优化
近年来公司业务迅猛发展,数据量爆炸式增长,随之而来的的是海量数据查询等带来的挑战,我们需要数据量在十亿,甚至百亿级别的规模时依然能以秒级甚至毫秒级的速度返回,这样的话显然离不开搜索引擎的帮助,在搜…...
cpp之STL
STL原理 STL ⼀共提供六⼤组件,包括容器,算法,迭代器,仿函数,适配器和空间配置器,彼此可以组合套⽤。容器通过配置器取得数据存储空间,算法通过迭代器存取容器内容,仿函数可以协助算…...
基于Spring Boot开发的资产管理系统
文章目录 项目介绍主要功能截图:登录首页信息软件管理服务器管理网络设备固定资产明细硬件管理部分代码展示设计总结项目获取方式🍅 作者主页:Java韩立 🍅 简介:Java领域优质创作者🏆、 简历模板、学习资料、面试题库【关注我,都给你】 🍅文末获取源码联系🍅 项目…...
Markdown总结
文字的着重标记与段落的层次划分 Tab键可以缩进列表; shift Tab:取消缩进列表 加粗(****)、斜体(**)高亮:xxx$$:特殊标记删除:~~xxx~~多级标题:######无序列…...
字节跳动软件测试岗4轮面经(已拿34K+ offer)...
没有绝对的天才,只有持续不断的付出。对于我们每一个平凡人来说,改变命运只能依靠努力幸运,但如果你不够幸运,那就只能拉高努力的占比。 2021年10月,我有幸成为了字节跳动的一名测试工程师,从外包辞职了历…...
docker - 搭建redis集群和Etcd
概述 由于业务需要,需要把之前的分布式架构调整成微服务,把老项目迁移到k8s的服务中,再开始编码之前,需要再本地环境里做相应的准备工作,使用docker搭建redis集群,Etcd主要是注册本地的rpc服务。 Liunx O…...
Java程序开发中如何使用lntelliJ IDEA?
完成了IDEA的安装与启动,下面使用IDEA创建一个Java程序,实现在控制台上打印HelloWorld!的功能,具体步骤如下。 1.创建Java项目 进入New Project界面后,单击New Project选项按钮创建新项目,弹出New Project对话框&…...
【Linux】理解进程地址空间
🍎作者:阿润菜菜 📖专栏:Linux系统编程 我们在学习C语言的时候,都学过内存区域的划分如栈、堆、代码区、数据区这些。但我们其实并不真正理解内存 — 我们之前一直说的内存是物理上的内存吗? 前言 我们…...
Unity脚本 --- 常用API(类)--- GameObject类 和
第一部分 --- GameObject类 1.在Hierarchy 层级面板中添加游戏物体其实就相当于在场景中添加游戏物体 2.每一个场景都有一个自己的Hierarchy层级面板,用来管理场景中的所有游戏物体 3.是的,我们可以创建多个场景 1.首先上面这两个变量都是布尔变量&am…...
HTML标签——表格标签
HTML标签——表格标签 目录HTML标签——表格标签一、表格标题和表头单元格标签场景:注意点:案例实操小结二、表格的结构标签场景:注意点:案例实操:三、合并单元格思路场景:代码实现一、表格标题和表头单元格…...
Telerik JustMock 2023 R1 Crack
Telerik JustMock 2023 R1 Crack 制作单元测试的最快、最灵活和模拟选项。 Telerik JustLock也很简单,可以使用一个模拟工具来帮助您更快地生成更好的单元测试。JustLock使您更容易创建对象并建立对依赖关系的期望,例如,互联网服务需求、数据…...
筑基八层 —— 问题思考分析并解决
目录 零:移步 一.修炼必备 二.问题思考(先思考) 三.问题解答 零:移步 CSDN由于我的排版不怎么好看,我的有道云笔记相当的美观,请移步有道云笔记 一.修炼必备 1.入门必备:VS2019社区版&#x…...
【面试题】当面试官问 Vue2与Vue3的区别,你该怎么回答?
大厂面试题分享 面试题库后端面试题库 (面试必备) 推荐:★★★★★地址:前端面试题库被问到 《vue2 与 vue3 的区别》应该怎么回答Vue 内部根据功能可以被分为三个大的模块:响应性 reactivite、运行时 runtime、编辑器…...
使用Python对excel中的数据进行处理
一、读取excel中的数据首先引入pandas库,没有的话使用控制台安装 —— pip install pandas 。import pandas as pd #引入pandas库,别名为pd#read_excel用于读取excel中的数据,这里只列举常用的两个参数(文件所在路径ÿ…...
TCP协议原理三
文章目录七、延时应答八、捎带应答九、面向字节流粘包问题十、TCP异常情况总结七、延时应答 如果说滑动窗口的关键是让窗口大一些,传输速度就快一些。那么延时应答就是在接收方能够处理的前提下,尽可能把ack返回的窗口大小尽可能大一些。 如果在接受数据…...
mac在命令行里获取root权限
1、为什么要获取root权限? 答:一些命令在正常状态下没有权限会报错,只有获取了root权限才能正常操作。 比如我们想修改一些系统的文件: vim /etc/shells 1 修改后保存,发现没权限,报错了。如下图…...
文献阅读 Improving Seismic Data Resolution with Deep Generative Networks
题目 Improving Seismic Data Resolution with Deep Generative Networks 使用深度生成网络提高地震数据分辨率 摘要 叠前数据的使用,通常可以来解决噪声迹线、覆盖间隙或不规则/不适当的迹线间距等问题。但叠前数据并不总是可用的。作为替代方案,叠后…...
mysql数据库之子查询练习
1、查询员工的姓名、年龄、职位、部门信息(隐式内连接)。 #emp、dept #连接条件:emp.dept_iddept.id select e.name,e.age,e.job,d.name from emp e,dept d where e.dept_idd.id; 2、查询年龄小于30岁的员工姓名、年龄、职位、部…...
西电计算机通信与网络(计网)简答题计算题核心考点汇总(期末真题+核心考点)
文章目录前言一、简答计算题真题概览二、网桥,交换机和路由器三、ARQ协议四、曼彻斯特编码和差分曼彻斯特编码五、CRC六、ARP协议七、LAN相关协议计算前言 主要针对西安电子科技大学《计算机通信与网络》的核心考点进行汇总,包含总共26章的核心简答。 【…...
【博学谷学习记录】超强总结,用心分享丨人工智能 Python基础 个人学习总结之列表排序
目录前言简述list.sort()语法返回值实例无参参数key参数reversesorted()语法返回值实例无参参数key参数reverseoperator.itemgetter功能简述实例List.sort与sored区别sorted原理:Timsort算法扩展list原理数据结构心得前言 经过一周的学习,对Python基础部…...
深度讲解React Props
一、props的介绍 当React遇到的元素是用户自定义的组件,它会将JSX属性作为单个对象传递给该组件,这个对象称之为“props”。 函数声明的组件,会接受一个props形参,获取属性传递的参数 function ComponentA(props) { return <…...
WebRTC现状以及多人视频通话分析
1.WebRTC 概述WebRTC(网页实时通信技术)是一系列为了建立端到端文本或者随机数据的规范,标准,API和概念的统称。这些对等端通常是由两个浏览器组成,但是WebRTC也可以被用于在客户端和服务器之间建立通信连接࿰…...
【Windows】Windows下wget的安装与环境变量配置
1 wget安装 GNU Wget常用于使用命令行下载网络资源(包括但不限于文件、网页等)。 GNU Wget官网:GNU Wget GNU Wget for Windows:GNU Wget for Windows 安装时首先下载主安装包:Complete package, except sources&…...
外贸建立网站怎么做/有没有推广app的平台
首先我们看下面视图: 这种效果大家都不陌生,网上好多都说是仿人人网的,估计人家牛逼出来的早吧,我也参考了一一些例子,实现起来有三种方法,我下面简单介绍下: 方法一:其实就是对Gest…...
wordpress升级插件出现意外错误/seo长尾关键词
实验步骤: 1、将SD卡插在SD卡槽中。 2、将跳线冒跳至USB_OTG,将USB_OTG通过Micor USB线与USB主机(电脑)相连。 3、烧写程序,我的电脑中将出现一个磁盘。 实验现象: 核心代码: int main(void) {/…...
怎么做网站设计/百度网站排名怎么提高
今天带给大家的是一道中国台湾某著名cpu生产公司的面试题。储备知识要看懂今天的代码需要一点储备知识。// C语言的符号意义 ~ 按位取反 例如 ~(111b) (000b) >> 按位右移 例如 (0011b)>>2 (0000b) ~、>>、 的…...
北极寒流wordpress/百度提问登录入口
当前位置:我的异常网 数据库 oracle-15树形查询oracle-15树形查询www.myexceptions.net 网友分享于:2013-06-13 浏览:0次oracle--15树形查询Oracle查询树形结构oracle中的select语句可以用START WITH...CONNECT BY PRIOR子句实现递归查询,…...
莱芜二手房网站/博客可以做seo吗
作为物联网细分领域最具潜力市场的智能家居,未来几年内在全球范围内将迅速增长。据预测,在未来五年,全球智能家居设备市场实现2倍增长,从2012年的不足2000万个节点增长至2017年的9000多万个节点。而国内,智能家居市场拥…...
靠网站建设成功/深圳网络推广服务公司
定时器类Timer在java.util包中。使用时,先实例化,然后使用实例的schedule(TimerTask task, long delay)方法,设定指定的任务task在指定的延迟delay后执行。定时器任务类TimerTask是抽象类,继承并重写其run()方法,可实现…...