为什么用 Java:一个 Python 程序员告诉你
这篇文章专门给程序员写的,普通读者慎入。原作者:Kevin Sookocheff 译者:Celia Zhen,原文点击文末链接。
每当我告诉别人我一直在用Java工作时,大家的反应都是:
“纳尼!Java?为啥是Java?”
说实话,本人刚开始的时候也是同样的反应。但是由于Java的类型安全,执行性能和坚如磐石的工具,我渐渐地开始欣赏Java。同时我注意到,现在的Java已今非昔比——它在过去的10年间稳健地改善着。

缘何是Java?
假 设每天都用Java的想法还没有让君恶心到食不下咽,我在此重申Java已非你所了解的“吴下阿蒙”了。当Python, Ruby, 和Javascript在“动态类型语言革命”™(我自己造的名词)中大放异彩时,Java已经悄悄地借鉴了动态语言和函数式语言的很多吸引人的特性,同 时保留了让Java和JVM晋级一流开发环境的先贤的努力成果。凭借大约9百万Java攻城狮的基层群体,Java仍然是世界上最受欢迎的编程语言。我们 不能仅仅因为Java的语法有一点点繁琐,就抹杀掉它所有的历史和开发工作。但是流行不等同于正确。下面我们就来看看是什么让Java如此大放异彩。
Java虚拟机(JVM)
Java虚拟机(JVM) 已经诞生20年了。在此期间,它被部署在成千上万的系统上,历经了无数的漏洞修复和性能提升。JVM的优点有以下几个方面。首先,JVM完美支持日志和监控, 这使你可以很方便地监控小到单个线程的性能指标。JVM有世界上最优化的垃圾回收器之一,你可以根据优化吞吐量等因素灵活选择垃圾回收算法。最 后,Java承诺的“write once, run anywhere”终于得已实现——你可以轻松地在任何架构上部署一个Java应用(大家还是承认applet从来没有过吧)。为什么用Scala和 Clojure这样新式语言的聪明人会选择JVM作为他们的执行环境呢?——因为JVM为你的代码提供了一个无出其右的分发环境。抛弃像JVM这样坚如磐 石的工具是非常不合理的。
库的支持
如 果你需要做点什么,很可能已经有非常好用且经过测试的Java库在等着你。Java库大部分都是成熟并用于实际生产开发的。Google, Amazon, LinkedIn, Twitter和很多Apache项目都很倚重于Java。如果你用了Java,你可以参考这些库和公司,从而借鉴伟大的程序员先驱们的工作。
类型安全
Java的类型系统,虽然有时很繁琐,但是这使得你可以写出“好用”的代码。不再有运行调试,它使你可以依靠编译器而不是单元测试——单元测试只在你知道bug在哪里的时候才有用。类型安全也使你轻松的代码重构。Java同时支持范型——Go语言的最大诟病之一。再者,Guava这样的库I以最小的样板和开销,标准化了创建类型安全的API的方法。 Java编译器的改进也意味着你可以在享受类型安全的同时最小化范型所需的样板代码。
并发性
下面这条tweet总结了大多数动态语言的并行状态:

Most JS/Python/Ruby apps… pic.twitter.com/hkDkjdxpFH
— Reuben Bond (@reubenbond)
Java 却有着对多线程和并行的一流支持。对于Java 1.7, 许并行的immutable数据结构令你轻松地在线程间共享数据。Akka库更进一步的提供了Erlang型的Actors来写并发和分布式的程序。我并 不是在说Java比Go具有更好的并行支持,但是可以管理单个线程这一特性为Java应用提供了异步性能;而Python是做不到这点的。
用最新的Java来编程
现在你的心情可能已经从恶心变成好奇了,那么我们在2015年该如何写Java呢?从哪儿开始呢?首先,让我们回顾一些在Java 7和Java 8涌现的核心语言概念。
迭代
首先我们一起来看看迭代。下面是Java 8中的 for循环:
List<String> names = new LinkedList<>(); // compiler determines type of LinkedList // ... add some names to the collection names.forEach(name -> System.out.println(name));
或者是被大大简化的 for关键词?
for (String name : names) System.out.println(name);
这2种循环结构都比你平时看到的for循环简洁的多。
Lambda函数
上面提到的第一个for循环引入了Lambda函数这个新概念。Lamda函数,语法记作->, 是Java语言的一项重大改革,并从函数式编程中引入了一些概念。
下面来看几个Java中Lambda函数的例子。
// Lambda Runnable
Runnable r2 = () -> System.out.println("Hello world two!");
// Lambda Sorting
Collections.sort(personList, (Person p1, Person p2) -> p1.getSurName().compareTo(p2.getSurName()))
// Lambda Listener
testButton.addActionListener(e -> System.out.println("Click Detected by Lambda Listener"));
这里无法详细展开Lambda函数这个话题——http://www.drdobbs.com/jvm/lambda-expressions-in-java-8/240166764文章提供了一个很好的切入点来更多地了解Lambda函数。
流
Java 8引入了流(stream)的概念,这为Java提供了很多现代函数式语言的特性。流是一种对集合上的一系列转换延迟执行的机制。比如我们来数一下以’A’开头的名字。首先想到的方法肯定是像下面这样:
List<String> names = new LinkedList<>();
// ... add some names to the collection
long count = 0;
for (String name : names) {
if (name.startsWith("A"))
++count;
}
如果用流,上述就可以简化为首先将集合转换成流,然后使用函数:
List<String> names = new LinkedList<>();
// ... add some names to the collection
long count = names.stream()
.filter(name -> name.startsWith("A"))
.count();
Java同时支持用parallelStream()来进行流的并行处理。并行流允许流水线业务在独立的线程同时执行,这不仅改进了语法,同时提高了性能。在大多数情况下,你可以简单得用parallelStream()替换stream()实现并行。
Try-With-Resources结构
在Java 6之前,打开一个文件然后读取内容需要通过try/finally来完成:
static String readFirstLineFromFileWithFinallyBlock(String path)
throws IOException {
BufferedReader br = new BufferedReader(new FileReader(path));
try {
return br.readLine();
} finally {
if (br != null) br.close();
}
}
但是readLine和close都有可能抛出异常。在这种情况下,readLine抛出的异常被忽略,我们事实上并不知道readLine执行失败。
Java 7引入了 Try-With-Resources结构来克服这种缺陷:
static String readFirstLineFromFile(String path) throws IOException {
try (BufferedReader br =
new BufferedReader(new FileReader(path))) {
return br.readLine();
}
}
上例中,无论在何种失败情况下,BufferedReader都会自动关闭文件流。你可以通过用逗号分隔的方式,用一个try语句来打开多个资源。
多重catch
以往Java只允许一个catch代码块对应一个异常,这造成如下的代码冗余:
catch (IOException ex) {
logger.log(ex);
throw ex;
catch (SQLException ex) {
logger.log(ex);
throw ex;
}
从Java 7开始,你可以在一个代码块内捕捉多个异常,从而减少了代码冗余:
catch (IOException|SQLException ex) {
logger.log(ex);
throw ex;
}
数值字面常量(Numeric Literals)
数值字面常量可以添加下划线是Java语言的新特性。这允许你使用_作为大数字的视觉分隔符。下面的例子不言自明:
int thousand = 1_000; int million = 1_000_000;
使用Java
看到现代Java的语法如何简化并扩展了老Java之后,你可能已经摩拳擦掌跃跃欲试Java了。我整理了一下第三方的工具和库,这些可以用来帮助你们上手。
Maven
Maven是一个Java构建系统,相比于配置,它更重视规范。Maven定义了应用程序的结构,并提供了许多内置函数,比如运行测试,打包应用,部署你的库。使用Maven会显著降低管理Java项目的认知开销。 Maven Central是Java世界中的PyPI,为已发布的Java库提供一站式服务。
核心函数
谷歌的Guava library提供了谷歌Java开发中所使用的核心函数。这包括应用于集合,缓存,基础数据类型,并发,字符串处理工作,I/O等的常见函数。Guava为如何设计好的的Java API提供了绝佳的案例分析,提供最有效的从Java中推荐的最佳实践的具体例子一个很好的案例, Effective Java中推荐的最佳实践大部分都在Guava中得以体现。Guava被用于谷歌产品开发,进行了超过286,000个单元测试,可谓经受过实战测试的考验。
日期/时间函数
Joda-Time 已 经成为Java实际上的标准日期/时间函数库。事实上,Java 8几乎一字不差地采用了Joda-Time规范。自此,我们建议使用java.time中的日期/时间函数代替Joda-Time。但是,如果你需要使用 Java 8之前的版本,Joda-Time提供了无与伦比的API。
分布式系统
Akka 提供类似Erlang型的Actor模型的抽象层来编写分布式系统。Akka可以从容应对许多种不同的故障,为编写可靠的分布式系统提供了更高层次的抽象。
Web应用程序
需要用Java写一个功能完善的Web应用程序?莫怕,有Play Framework罩着你。Play基于Akka的非阻塞I/O,提供了编写Web应用程序的可扩展的异步框架。如果想使用不那么前沿但是被广泛应用于产品的框架,请尝试Jetty。
单元测试
JUnit 仍为编写单元测试的标准。最近几年,JUnit的匹配器有所扩展,允许你对集合作assertions。例如,您可以轻松地断言一个链表是否包含某个特殊值。
模拟框架(Mocking Framework)
Mockito是Java的标准模拟库。它提供了所有你能想到的且对编写测试非常重要的模拟库的功能。
然而不足的是。。。
目前为止,我一直在为Java说好话,但是有些方面它还是很烂。
它还是Java!
Java的历史遗留不可避免,Java仍然向下兼容其最早的版本,这意味着语言和标准库的最烂的部分还存在着。Guava是为了令Java语言更讨人喜欢而产生这个事实就证明了,Java和API存在不一致,令人困惑的问题,有时甚至是完全错误的。
JSON
Java缺少映射到JSON的object literal syntax(如Python的字典literal syntax)。正因如此,从Java对象映射到JSON有时需要繁复的对象实例化和映射,反之亦然。目前有各种JSON库在这个领域竞争,Jackson是当前的最受欢迎的,但是Jackson的文档需要进一步完善。
模拟(Mocking)
Mockito解决了测试Java代码中的很多痛点,但是从像Python语言的灵活转换到Java语言的严格,你需要更谨慎地来设计你的类用于模拟。
REPL
我之所以喜欢Python,其中一点就是它可以迅速地实现读取﹣求值﹣输出循环( read-eval-print loop),从而快速评估新的想法或检验假设。虽然一直有声音说要把读取﹣求值﹣输出循环添加到标准Java库,这一点目前还是不支持的。
语法累赘
虽然Java编译器的进步意味着明确的类型签名不再那么需要——尤其对于泛型——但是Java仍然比Python冗余的多。启动和运行一个项目需要更多的样板和开销——通常这意味更多的工作。
结论
Java拥有一个漫长而传奇的历史,其中有好有坏。如果你已经很多年没有使用Java工作了,也许现在是一个好机会再次尝试它。只要不是像下面这样做:

为什么用 Java:一个 Python 程序员告诉你的更多相关文章
- 做为一个Python程序员的基本素养
今天在学习的过程中,明白了一些不是Python标准所必须要做的事情,二是做为一个合格的Python程序员应该所遵从的一些规范 分享给大家,有不足的地方请大家指正,此下是我学习的一点心得: 1.在给变量 ...
- 520到了,作为一个python程序员,必须整点肤白貌美的爬虫代码给你们~
马上520就快到啦~ 整点好看的给你们看下~ 直接开搞~ 代码流程 模拟浏览器向服务器发送一个http请求,网站接收到请求后返回数据.在写爬虫代码的时候一定先要去模拟浏览器访问,因为现在的网站当接收到 ...
- 成为python程序员,对疫情过后的毕业生来说,真是一个不错的方向吗?
Python最近几年,一直被炒得很火,这其中有商业因素,但更重要的是即将到来的人工智能时代,而python就恰好是最适合的编程语言. 所以无论是在职的人,还是在校的学生,都想着跟上这一趋势,但,在今年 ...
- Coding girl一个老程序员谈到的一个女程序员的故事
因为有人说我给一个女程序员的建议不靠谱,我不服,因为我的工作经历中的一些女程序员都很不错,比那些男程序员都强,所以,我在新浪微博和twitter上征集女程序员的故事和想法,这两天来,我收到了好几封邮件 ...
- 介绍Python程序员常用的IDE和其它开发工具
概述 “工欲善其事,必先利其器”,如果说编程是程序员的手艺,那么IDE就是程序员的吃饭家伙了. IDE 的全称是Integration Development Environment(集成开发环境), ...
- python笔记:#002#第一个python程序
第一个 Python 程序 目标 第一个 HelloPython 程序 Python 2.x 与 3.x 版本简介 执行 Python 程序的三种方式 解释器 -- python / python ...
- Java技术开发程序员如果在2019年立足
2019年的互联网环境相对以往来说要更复杂一些,互联网领域也正在经历从消费互联网向产业互联网转型的阵痛期.其实不少公司从2018年开始已经在陆续进行结构化调整,这些调整中的重要内容就是岗位调整,而岗位 ...
- Python程序员常用的IDE和其它开发工具
概述 “工欲善其事,必先利其器”,如果说编程是程序员的手艺,那么IDE就是程序员的吃饭家伙了. IDE的全称是Integration Development Environment(集成开发环境),一 ...
- 人生苦短之---第一个Python程序
第一个 Python 程序 目标 第一个 HelloPython 程序 Python 2.x 与 3.x 版本简介 执行 Python 程序的三种方式 解释器 —— python / python ...
随机推荐
- [转载] PowerMokito 使用
一.为什么要使用Mock工具 在做单元测试的时候,我们会发现我们要测试的方法会引用很多外部依赖的对象,比如:(发送邮件,网络通讯,远程服务, 文件系统等等). 而我们没法控制这些外部依赖的对象,为了解 ...
- PHP - 《高性能php应用开发》学习笔记
一.基准测试 php网站优化最佳实践:优化前端(压缩js/css/images)--->程序优化(编码最佳实践.opcode缓存.变量/数据缓存)--->数据库.服务器调优-->操作 ...
- var t = a&&b;的问题
var a = "avalue";var b = "bvalue";var t = a&&b;console.info(t); // bvalu ...
- hadoop1中mapreduce原理详解
剖析Mapreduce作业运行机制:原理如下图: 原理图的解释的可以分为以下几个部分 1.客户端提交一个mapreduce的jar包给JobClient 2.JocClient通过RPC和JobTra ...
- HDU 4493 Tutor 水题的收获。。
题目: http://acm.hdu.edu.cn/showproblem.php?pid=4493 题意我都不好意思说,就是求12个数的平均数... 但是之所以发博客,显然有值得发的... 这个题最 ...
- POJ 3267 The Cow Lexicon 简单DP
题目链接: http://poj.org/problem?id=3267 从后往前遍历,dp[i]表示第i个字符到最后一个字符删除的字符个数. 状态转移方程为: dp[i] = dp[i+1] + 1 ...
- MySQL简单使用
1.启动MySQL服务器实际上上篇已讲到如何启动MySQL.两种方法:一是用winmysqladmin,如果机器启动时已自动运行,则可直接进入下一步操作.二是在DOS方式下运行 d:/mysql/bi ...
- 关于键盘冲突那点事(3键冲突/7键冲突/PS2/USB的各种原理)
转自关于键盘冲突那点事(3键冲突/7键冲突/PS2/USB的各种原理) 最近闲得无聊,正好看到有人发帖提问,于是就来详细说说所谓键位冲突和无冲突的各种原理--基本上这也是个老生常谈的话题了,但相关的技 ...
- PS命令删除所有EXCHANGE2013内用户邮件
因为在测试的时间产生了大量测试邮件,所以在正式上线前,要删除所有用户的邮件. 命令如下: Get-Mailbox | Search-Mailbox -DeleteContent
- PEP Index > PEP 339 -- Design of the CPython Compiler 译文
http://www.python.org/dev/peps/pep-0339/ PEP: 339 标题: CPython的编译器设计 版本: 425fc5598ee8 最后修改: 2011-01-1 ...