Java最困扰你的那些事
英文原文:What bugs you most about Java?
啊哈 Reddit(某知名国外在线问答社区),没了你我们还能在哪里从鱼目混珠的网络中提炼真正的精华?就在这杂乱无章的论坛中,的的确确存在着这样一些精辟的讨论。
比如有个叫 Shambloroni 的兄弟发起了一个引起广泛共鸣的话题(收到了 150 个回复和投票)。话题的开始是他在吐槽 Java 有哪些方面最让他感到厌烦,同时他又向其他程序猿征集关于编写 Java 时令人蛋疼(伤感……)的故事。
下面我列出了一些最有意思的回复——你会同意吗?
Try-catch 声明之殇
虽然我才刚刚开始编写 Java 程序,但是在 Java 中实现让程序停顿一些时间然后继续运行这件事情已经复杂的令我震惊。除了写下基本的指令完成停顿外,我还必须要用 try-catch 来包裹 的这些操作。没准对于大牛来说这不算什么,但是对我来讲这太令人难过了。
还有跟所有 IO 相关的异常也让许多简单的事情变得繁琐起来。比如我仅仅想简洁地做些事情,但最后总是会搞砸,。然后我不得不在所有方法上加上“throws Exception”。
没有,没有,还是没有
- 没有无符号的整形。
- 没有操作符重载。
- 没有对象属性。
- 没有代理 (尽管我听说 Java 8 中可能引入了代理或类似机制,但我还没细看)。
我同样痛恨很多 Java 库中引入模块的方式(比如根据名称加载模块)。相对而言,我更希望在编译时就能检查我需要的依赖库是否已经加载了。
找不到爱…
我恨 Java,就是单纯的恨。负责任地说,我从不使用 Java 写程序的最大因素就是因为我恨它。还有一点就是 JRE 糟糕的模块化。就算你用了一些加载工具比如 Launch4j,你仍然需要让用户安装一个超过 200 MB 的框架来运行程序。
没有 getter、setter
Java 还缺少 getter 和 setter 注解。这样可以更简单的添加和移除模板代码。
缺乏亮点
Java 缺少一些杀手级的模块。虽然听说一些不错的模块将被加入到 Java 9.x 中,但目前为止这是我最大的槽点了。
非暴力不合作
我遇到的最大的麻烦就是如何一起使用基本元素和对象。例如, 把一个 char[] 类型的变量转换成一个列表别提有多费劲了,而且我感觉这种操作根本没必要这么麻烦。
心愿单
- 我最想要的就是像 Perl 中那种上下文相关的函数了,以及当处理真正的异常时,这些函数可以在一个语言中所扮演的角色。
- 我还希望 Java 可以支持智能打包返回值,这样我们就可以像在 Perl 中那样返回一个包含多个值的列表了。
- 我希望使用异常的标准库还可以在不适合抛出异常的场景下使用并能够处理失败。
- 还有,另一个烦人的地方就是我在用 StringWriter 时候还要处理 IO 异常。
还缺些什么
- 缺少宏对我来说使 Java 减分不少(我并不说在 C/C++ 中使用的预处理器宏,而是在 Lisp/Scheme 中使用的那种宏)。
- 不论做什么事情你都需要定义一个类,尽管你可能肯本不需要一个类。比方说,我想把一段经常使用的代码提取出来,然后在需要地方使用——为了达到这 个目的,我必须要把这段代码封装成一个 final static 的方法,并放在一个类中。这样一来我还得费劲去给类取一个方便理解的名字……本来这事儿可以很简单(这确实很简单, 尤其是当你可以定义宏的时候)。
有没有搞错
- 缺乏对泛型的支持。C++ 中的模板要强大的多。
- 事实上,在 Java 中你根本不能在泛型中实例化一个类,除非你把这个类作为参数来声明一个泛型。
- 你很难给一个类加上结构函数并让它销毁这个类。RAII(一种资源管理模式,见 C++)却一直非常有用。
- 没有操作符重载。C++ 允许你是将 == 操作符用于比较字符串。同样的,大整数运算因为同样的原因变的很难使用。
呃, 好吧
没有无符号的基础类型。这尼玛是闹哪样啊!
还是 getter、setter
1、null(最大槽点)。
2、没有 getter 和 setter 注解(例如,没有属性)。
3、Java 只支持位置参数。我喜欢像 Smalltalk 那种支持多样化的参数形式,或者是强制使用关键字参数的 Python 3。
比如在 Samlltalk 中调用一个具有两个参数的函数,可以这样做:
myInstance myMethodWithFoo: arg1 Bar: arg2
在 Python 中你可以使用下面的语法来调用函数并给函数参数赋值:
my_inst.my_method (foo=arg1, bar=arg2)
4、……不支持多分派(Multiple Dispatch)?
这些是我最先想到的,不过觉得应该还有更多。认真地说,使用回调函数一直是一个大问题,因为它太麻烦了。不过 Java 8 中解决了这个问题,我还是很开心的 =)
愚蠢的默认值
- 默认的可见性。如果没有给变量或方法一个修饰符,那么这个方法应该是私有的,而不是包内可见。
- 默认的修改能力。最终类型(在所有情况下)应该是默认的,并用“var”作为修饰符。目前的情况是,程序员很少会把一个方法的参数设置为最终类型,因为那样会让变量很快变得不可读。同时,在一个方法中重写参数也是很少见的情况。
- 集合接口。Java 中应该提供一个可写的集合接口,现在集合继承自这个可写的接口,只是把所有改变集合内容的方法屏蔽掉。这样就会减少现在你会 在 Collections.unmodifiable……() 和一些第三方的 API 中见到的那些令人困惑的历史遗留方法。有了可写的集合接口,Java 将会变得更加类型安全。
- 缺少表达能力。在用过 Scala (或是最新的 PL)之后, 你会觉得 Java 非常的繁琐。这是最常见的关于 Java 的吐槽,但它这就是事实。
说说异常
被强迫的处理异常——真主保佑你。谁能告诉我为什么我非要用 try-catch 来包裹每一个 Thread.sleep () ……?我从来就没有真正见过那个我被要求去处理的 InterruptedException。
我知道我要说的可能不是一个广泛认可的问题,但是我真的同意 checked exception(应被检查的异常)很烦人。这些异常让代码变得面目全非还让重构变得不可能。我明白他们为什么存在(而且理论上也说的 通),但是他们没为开发者带来什么实质好处。不论你做什么,都不要留一个空的 catch 块,就算你认为这个异常永远不会发生。你大可以把这个 checked exception 用 RuntionException(运行时异常)重新封装一下,再抛出去。
翻译: ImportNew.com - 靳禹