2009 年 4 月 20 日,Java 的亲生父亲 Sun 被养父 Oracle 以 74 亿美元收购,这在当时可是一件天大的事。有不少同学都担心 Java 的前途,我当时傻不啦叽地也很担心:自己刚学会如何通过记事本编写 Java 代码,然后通过 cmd 打印 Hello World 呢,这一下难道白学了?

但其实这种担心是多余的,因为 Java 并不会陪葬,毕竟行业内有太多基于 Java 的软件系统在运行,Java 牵扯了太多人的饭碗。10 年过去了,Java 果然没有陪葬,我仍然坚守在 Java 的阵线上。

2011 年 7 月 7 日,代号「海豚(Dolphin)」的 Java 7 首次推出,这也是 Java 历史上一次非常重要的版本更新。同时推出了非常多实用的新特性,比如说创建泛型实例时自动类型推断、switch-case 语句支持字符串类型、新增 try-with-resources 语句等等。

这么多年过去了,Java 7 的“新”特性显然都变成老古董了——它们似乎也不需要我再赘述了,但好像不是这样的。前几天我发了一篇文章,用到了其中一个新特性,竟然有同学表示从来没见过这个新特性,特意在交流群里@我,要我说清楚怎么回事(代码折叠了,随后贴出来)。

当时我就在想啊,原来技术从来没有“新与旧”之说,只有知不知道。所以借此机会,我就再来“赘述”一下 Java 7 的那些最经常使用的新特性吧。

01、数值中可使用下划线分隔符联接

之前图片中的代码没有展示全,现在我把具体的代码贴出来。

try {
    AsynchronousFileChannel channel = AsynchronousFileChannel.open(file);
    Future<Integer> result = channel.read(ByteBuffer.allocate(100_000), 0);
} catch (IOException e) {
    e.printStackTrace();
}

其中 100_000 就是读者要我解释清楚的那个特性:在数值类型的字面值中使用下划线分隔符联接。

人脑不总是很善于记住很长串的数字,所以在处理长串数字时会采用分割法,比如说电话号码要用一个分隔符“-”隔开,银行卡号会每隔四位有一个空格等等。

数字中没有用逗号(,)和中划线(-)作为分隔符,是因为它们可能会引发歧义,取而代之的是下划线(_)。这个不起眼的特性,让我们开发人员在处理数字上轻松多了,毕竟 100_000100000 (忍不住查了一遍 0 的个数,害怕多写或者少写)辨识度高得多。

下划线(_)位置并不固定,你可以随意摆放,举例如下:

int a = 100_000, b = 100000, c = 10_0000;
System.out.println(a==b); // true
System.out.println(b==c); // true
System.out.println(a==c); // true

需要注意的是,下划线仅仅能在数字中间,编译器在编译的时候自己主动删除数字中的下划线。反编译后的代码如下所示:

int a = 100000;
int b = 100000;
int c = 100000;
System.out.println(a == b);
System.out.println(b == c);
System.out.println(a == c);

02、 switch-case 语句支持字符串类型

我们都知道,switch 是一种高效的判断语句,比起 if/else 真的是爽快多了。示例如下:

String wanger = "王二";

switch (wanger) {
case "王二":
    System.out.println("王三他哥哥王二");
    break;
case "王三":
    System.out.println("王二他弟弟王三");
    break; default:
    System.out.println("王二他妹妹王六");
    break;
}

switch-case 语句在处理字符串的时候,会先将 switch 括号中的字符串和 case 后的字符串转成 hashCode,所以字符串不能为 null,否则会抛出 NullPointerException。反编译后的代码如下所示:

String wanger = "王二";
switch (wanger.hashCode()) {
    case 936926 :
        if (wanger.equals("王三")) {
            System.out.println("王二他弟弟王三");
            return;
        }
        break;
    case 937057 :
        if (wanger.equals("王二")) {
            System.out.println("王三他哥哥王二");
            return;
        }
} System.out.println("王二他妹妹王六");

03、try-with-resources 语句

try-with-resources 的基本设想是把资源(socket、文件、数据库连接)的作用域限定在代码块内,当这块代码执行完后,资源会被自动释放。

在此之前,资源的释放需要在 finally 中主动关闭,不管 try 中的代码是否正常退出或者异常退出。就像下面这样:

BufferedReader in = null;
try {
    in = new BufferedReader(new FileReader("cmower.txt"));
    int charRead;
    while ((charRead = in.read()) != -1) {
        System.out.printf("%c ", (char) charRead);
    }
} catch (IOException ex) {
    ex.printStackTrace();
} finally {
    try {
        if (in != null)
            in.close();
    } catch (IOException ex) {
        ex.printStackTrace();
    }
}

这样的代码看起来就像老太婆的裹脚布,又臭又长;有了 try-with-resources 之后,情况大有改观,不信你看:

try (BufferedReader in = new BufferedReader(new FileReader("cmower.txt"));) {
    int charRead;
    while ((charRead = in.read()) != -1) {
        System.out.printf("%c ", (char) charRead);
    }
} catch (IOException ex) {
    ex.printStackTrace();
}

是不是清爽多了!把需要释放的资源放在 try 后的 () 中,连 finally 也不需要了。不过,需要注意的是,上面的代码还需要优化,应该为每一个资源声明独立的变量,否则的话,某些特殊的情况下,资源可能无法正常关闭。

try (FileReader fr = new FileReader("cmower.txt"); 
    BufferedReader in = new BufferedReader(fr);) {
    int charRead;
    while ((charRead = in.read()) != -1) {
        System.out.printf("%c ", (char) charRead);
    }
} catch (IOException ex) {
    ex.printStackTrace();
}

try-with-resources 特性依赖于一个新定义的接口 AutoCloseable,需要释放的资源必须要实现这个接口。

不过,try-with-resources 在本质上仍然使用了 finally 去释放资源,只不过这部分工作不再由开发者主动去做——从反编译后的结果可以看得出来:

try {
    Throwable var1 = null;
    Object var2 = null;     try {
        FileReader fr = new FileReader("cmower.txt");         try {
            BufferedReader in = new BufferedReader(fr);             int charRead;
            try {
                while ((charRead = in.read()) != -1) {
                    System.out.printf("%c ", (char) charRead);
                }
            } finally {
                if (in != null) {
                    in.close();
                }             }

04、创建泛型实例时自动类型推断

在这个特性出现之前,有关泛型变量的声明略显重复,示例如下:

Map<String, ArrayList<String>> wanger = new HashMap<String, ArrayList<String>>();

这样的代码简直太长了,很多重复的字符,难道编译器不能推断出泛型的类型信息吗?Java 7 实现了这个心愿。

Map<String, List<String>> wanger = new HashMap<>();
List<String> chenmo = new ArrayList<>();
wanger.put("chenmo", chenmo);

这个看似简单的特性省去了不少敲击键盘的次数。

05、最后

除了上面我列出的这 4 个常用的新特性,Java 7 还有一些其他的特性,比如说 multi-catch,可以在一个 catch 语句中捕获多个异常;比如说对集合(Collections)的增强支持,可以直接采用 []、{} 的形式存入对象,采用 [] 的形式按照索引、键值来获取集合中的对象等等。

但总体上,我列出的那 4 个特性最为常用,其学习的意义更大。如果你觉得漏掉了某些更为常用的特性,欢迎你在文末提出来。

温故而知新,重温 Java 7 的那些“新”特性的更多相关文章

  1. JAVA JDK1.5-1.9新特性

    1.51.自动装箱与拆箱:2.枚举(常用来设计单例模式)3.静态导入4.可变参数5.内省 1.61.Web服务元数据2.脚本语言支持3.JTable的排序和过滤4.更简单,更强大的JAX-WS5.轻量 ...

  2. Java 8 正式发布,新特性全搜罗

    经过2年半的努力.屡次的延期和9个里程碑版本,甲骨文的Java开发团队终于发布了Java 8正式版本. Java 8版本最大的改进就是Lambda表达式,其目的是使Java更易于为多核处理器编写代码: ...

  3. Java引入的一些新特性

    Java引入的一些新特性 Java 8 (又称为 jdk 1.8) 是 Java 语言开发的一个主要版本. Oracle 公司于 2014 年 3 月 18 日发布 Java 8 ,它支持函数式编程, ...

  4. 使用示例带你提前了解 Java 9 中的新特性

    使用示例带你提前了解 Java 9 中的新特性 转载来源:https://juejin.im/post/58c5e402128fe100603cc194 英文出处:https://www.journa ...

  5. Java学习之==>Java8 新特性详解

    一.简介 Java 8 已经发布很久了,很多报道表明Java 8 是一次重大的版本升级.Java 8是 Java 自 Java 5(发布于2004年)之后的最重要的版本.这个版本包含语言.编译器.库. ...

  6. JAVA笔记 之 JDK新特性

    JDK1.5新特性1.泛型(Generics) 为集合(collections)提供编译时类型安全,无需每刻从Collections取得一个对象就进行强制转换(cast) 2.增强的for循环(for ...

  7. Java系列 - 用Java8新特性进行Java开发太爽了

    本人博客文章网址:https://www.peretang.com/using-java8s-new-features-to-coding-is-awesome/ 前言 从开始写博客到现在已经过去3个 ...

  8. Java系列 – 用Java8新特性进行Java开发太爽了(续)

    本人博客文章网址:https://www.peretang.com/using-java8s-new-features-to-coding-is-awesome-2/ 前言 上周, 我们谈论了关于Ja ...

  9. Java各个版本的新特性

    原链接:http://blog.csdn.net/shareus/article/details/50736159 1.5 1.自动装箱与拆箱: 2.枚举(常用来设计单例模式) http://www. ...

随机推荐

  1. Docker最全教程之MySQL容器化 (二十四)

    前言 MySQL是目前最流行的开源的关系型数据库,MySQL的容器化之前有朋友投稿并且写过此块,本篇仅从笔者角度进行总结和编写. 目录 镜像说明  运行MySQL容器镜像  1.运行MySQL容器  ...

  2. python算法与数据结构-什么是数据结构

    一.什么是数据结构 数据结构,直白地理解,就是研究数据的存储方式. 我们知道,数据存储只有一个目的,即为了方便后期对数据的再利用,就如同我们使用数组存储 {1,2,3,4,5} 是为了后期取得它们的值 ...

  3. spring源码解析之IOC容器(四)——属性注入

    上一篇跟踪了bean的创建过程,接下来,我们继续跟踪bean的属性填充的过程.先回到doCreateBean方法,代码如下: protected Object doCreateBean(final S ...

  4. 前端Vue基础学习

    Vue基础 对与vue的简洁明了的干货分享,适合小白学习观看,如果有笔误或者想了解更多的,请给笔者留言,后续会继续分享,从零开始的vue相关经验 1.创建vue实例 <div id=" ...

  5. 中转Webshell 绕过安全狗(一)

    前言 听说中国菜刀里有后门.抓包我是没有监测到异常数据包.为了以防万一,且更好使用中国菜刀硬杠安全狗.笔者收集了一下资料.无耻的copy大佬的源码,只是在大佬的基础上简单修改了一下,达到Webshel ...

  6. 【python3两小时根本不够】入门笔记04:线程+Lock安全同步

    有了简单爬虫,但是效率实在是太慢,于是决定启用线程进行爬取数据 但是对于临界资源的定义不好把握,思路如下: 1.定义队列(Queue的数据结构,List也可,安全性待考究) demo:https:// ...

  7. SQLPLUS执行PL/SQL语句块

    1.首先登录Oracle HR schema: 2.对于PL/SQL程序,分号表示语句的结束:而使用 "."  号表示整个语句块的结束,也可以省略.按回车键后,该语句块不会执行,即 ...

  8. spring源码深度解析— IOC 之 bean 创建

    在 Spring 中存在着不同的 scope,默认是 singleton ,还有 prototype.request 等等其他的 scope,他们的初始化步骤是怎样的呢?这个答案在这篇博客中给出. s ...

  9. iOS中 分类(category)与扩展(Extension)的区别?

    1.分类(category)的作用 (1).作用:可以在不修改原来类的基础上,为一个类扩展方法.(2).最主要的用法:给系统自带的类扩展方法. 2.分类中能写点啥? (1).分类中只能添加“方法”,不 ...

  10. python数据库-安装问题总结(48)

    一.ERROR1698(28000):Access denied for user root@localhost错误 我的操作系统是ubuntu: 我的MySQL版本是: 安装完成后,登录mysql的 ...