为什么用 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工作了,也许现在是一个好机会再次尝试它。只要不是像下面这样做:

时间: 2024-10-08 19:34:54

为什么用 Java:一个 Python 程序员告诉你的相关文章

一个老程序员告诉你:中国程序员为什么要跳槽

程序员 一些人总是发出一些错误的声音,形成了劣胜优汰可怕的现象.他们在误导着中国,把我们的后继军训练成软件蓝领,胸无大志,目光短浅,稍有点成绩就自满就自高自大的人,浑不知天外有天,外国正在虎视眈眈盯着中国的庞大市场. 由于软件蓝领的呼声人们不再致力于培养大批的高精尖人才,掌握国际尖端技术的人.而是花费心思培训一群猪出来给外国人利用.把自己的命运交给了外国人手里.结果,在最容易的管理软件领域也输给外国人,外国人派几个人过来,利用中国的劳动力,开发出软件,再卖给中国.大量的利润到了外国人手里.而我们

一个Python程序员的跨界之旅

时间过得真快,感觉15年年终总结刚写完,16年就结束了,看了blog,16年就写了可怜的8篇,对我来说16年还算顺风顺水.真正可能出乎意料的是年底我离开了呆了2年半的龙图游戏,暂时放弃了用了3年半的Python,转向了Java领域和数据仓库,这也算是跨界了吧... 家庭 16年老婆和闺女来北京住了小半年,那段时间是很幸福的一段时光,最终还是不适应北京的天气和雾霭,老婆和闺女逃离了北京,我又恢复了"单身"生活.16年在家庭上面比15年要有所进步,但恐怕17年...,看北京最近这雾霭,逃离

一个老师程序员说:这是学Java 必知必会的 20 种常用类库和 API

一个有经验的Java开发人员特征之一就是善于使用已有的轮子来造车.<Effective Java>的作者Joshua Bloch曾经说过:"建议使用现有的API来开发,而不是重复造轮子".在本文中,我将分享一些Java开发人员应该熟悉的最有用的和必要的库和API.顺便说一句,这里不包括框架,如Spring和Hibernate因为他们非常有名,都有特定的功能.最后,如果大家如果在自学遇到困难,想找一个java的学习环境,可以加入我们的java学习圈,点击我加入吧,会节约很多时

好程序员告诉你Java架构师学习路线

好程序员告诉你Java架构师学习路线,从初级Java工程师成长为Java架构师,你需要走很长的路,很多有计划的人在学习之初就在做准备.你知道Java架构师学习路线该怎么走吗?成为一个优秀的Java架构师究竟需要学什么?接下来就跟小编一起揭晓答案. 架构师是一个充满挑战的职业,知识面的宽窄往往决定着一个架构师的架构能力.架构师在整个软件开发过程中都起着重要的作用,并随着开发进程的推进而其职责或关注点不断地变化. 想要成为Java架构师,你首先要是一个高级Java工程师,可以熟练使用各种框架,并知道

一个平庸程序员自白:我不牛逼但那又怎样?

转自:http://www.managershare.com/post/189966 不是谁都能飚高音,但他们也有自己的声音. Jacob Kaplan-Moss是著名 Python 框架 Django 的 Co-Creater 和核心开发者,在 Heroku 担任安全部门的 Director, 他常年参加 Python 社区的年度聚会 PyCon ,并经常上台演讲.在这次 PyCon 2015 上,他声称自己顶多只是一个平庸的程序员,幻灯片上也写着: “Hi, I’m Jacob, and I

一个平庸程序员的自白(转)

转自豆瓣http://www.douban.com/note/502532701/ Jacob Kaplan-Moss 是著名 Python 框架 Django 的 Co-Creater 和核心开发者,在 Heroku 担任安全部门的 Director, 他常年参加 Python 社区的年度聚会 PyCon ,并经常上台演讲.在这次 PyCon 2015 上,他声称自己顶多只是一个平庸的程序员,幻灯片上也写着: "Hi, I'm Jacob, and I'm a mediocre program

Coding girl一个老程序员谈到的一个女程序员的故事

因为有人说我给一个女程序员的建议不靠谱,我不服,因为我的工作经历中的一些女程序员都很不错,比那些男程序员都强,所以,我在新浪微博和twitter上征集女程序员的故事和想法,这两天来,我收到了好几封邮件,让我很感动.其中,有一个故事让我回味很久,在脑海里挥之不去,可能是因为她的经历和我很相似,她的想法和我很有共鸣. 本来,我想通过收到的这些故事然后编辑成一篇关于女程序员的文章,但是我觉得这个故事已经足够好了,任何的编辑都是对这个故事的不尊重,所以,我原封不动,一字不改地把这个故事转到这里.我把一些

介绍Python程序员常用的IDE和其它开发工具

概述 “工欲善其事,必先利其器”,如果说编程是程序员的手艺,那么IDE就是程序员的吃饭家伙了. IDE 的全称是Integration Development Environment(集成开发环境),一般以代码编辑器为核心,包括一系列周边组件和附属功能.一个优秀的IDE,最重要的就是在普通文本编辑之外, 提供针对特定语言的各种快捷编辑功能,让程序员尽可能快捷.舒适.清晰的浏览.输入.修改代码.对于一个现代的IDE来说,语法着色.错误提示.代码折 叠.代码完成.代码块定位.重构,与调试器.版本控制

Python 程序员的 Golang 学习指南(III): 入门篇

基础语法 类型和关键字 类型 // 基础类型 布尔类型: bool 整型: int8,uint8,int16,uint16,int32,uint32,int64,uint64,int,rune,byte,complex128, complex64,其中,byte 是 int8 的别名 浮点类型: float32 . float64 复数类型: complex64 . complex128 字符串: string 字符类型: rune(int32的别名) 错误类型: error // 复合类型 指