我不喜欢Go语言的十个理由
链客,专为开发者而生,有问必答!
此文章来自区块链技术社区,未经允许拒绝转载。
Go 语言有多火爆?国外如 Google、AWS、Cloudflare、CoreOS 等,国内如七牛、阿里、知乎等都已经开始大规模使用 Go 语言开发相关产品,可以说它是近来风头最盛的编程语言之一。但再好的编程语言也不会是完美的编程语言,本文作者 Lawrence 使用了三年 Go 语言,并在这三年间参与了几个大型 Go 语言项目,但三年后他彻底放弃 Go 语言,而且不打算在新项目中使用它。Go 语言带给他的总体印象是:“好的方面非常好,不好的方面实在令人无法忍受”。Lawrence 在一篇博客文章中列出了他不再喜欢 Go 语言的十大理由,这篇文章很快在 HackerNews 上引发热议,下面就让我们一起来看看这十个理由到底有哪些。
不喜欢 Go 语言的十个理由
- Go 语言使用首字母大小写来决定标识符的可见性。
以小写字母开头的标识符在包内可见,以大写字母开头的标识符是公开的。这样做可能是为了省掉 public 和 private 关键字,但是,首字母大写已经被用来表示其他意思——比如类名的首字母是大写的,常量完全是大写的。对于我来说,使用完全小写的字母来定义全局常量实在是有点不适应。
如果你要定义私有结构体,情况会变得更加糟糕,因为你必须使用小写字母。例如,你可能有一个结构体叫作 user,那么你会怎么定义这个结构体的变量呢?你可能会把它叫作 user,这样看起来是不是有点奇怪?因为变量的名字跟结构体的名字是一样的,而且如果编译器也搞不清楚哪个是哪个,就会抛出编译错误。
type user struct { name string}func main() { var user *user user = &user{} // Compile error}
Go 语言通常会使用比较短的命名方式,比如 u。但是,35 年前,当我还在使用古董机 TRS-80 Basic 的时候就已经不再使用这种单字母的命名方式了。
当然,也有一些与实际开发相关的考虑。比如经常会出现这样的情况:刚开始定义了一个私有字段或结构体,后来需要把它们变成公开的,这样就不得不把所有使用这些标识符的地方都修改一遍。即使你不需要把它们变成公开的,但如果要使用 json 包来序列化它们,也不得不这么做。我就曾经定义过一个包含 74 个私有字段的结构体,因为要使用 json 序列化,不得不把所有字段都变成公开的,并且修改了所有使用这些字段的地方。
通过首字母大写的方式来决定可见性有一定的局限性(要么包内可见,要么公开)。例如,我经常需要使用文件内部可见的私有标识符,但目前 Go 语言不太可能引入这样的东西。
- 在 Go 语言里,结构体不会显式声明它实现了哪些接口,而是通过匹配方法签名来辨别。
这种设计犯了一个根本性的错误:它假设两个方法如果有相同的签名就表示有相同的契约。在 Java 中,如果一个类实现了一个接口,它会告诉编译器它实现了接口的所有方法。如果一个方法返回布尔类型,接口的注释会写清楚它的值代表什么意思(比如,true 表示成功,false 表示失败)。
但是,Go 语言的结构体可能一方面实现了同样的接口,但返回值的意思却是相反的。具体怎么实现可以自由发挥,因为并没有接口声明约束。当然,在 Java 里也可以这么做,但这很显然就是一个 bug。而在 Go 语言中,一个程序员可能在没有验证方法兼容性的情况下将一个对象转成某个接口,这样很容易引入潜在的 bug。验证兼容性的负担不应该强加给 API 使用者,应该由结构体的实现者承担,并在代码中声明清楚。
- Go 语言中没有异常,而是通过多个返回值来返回错误。
开发者会容易忘了检查返回值里是不是包含了错误。
db.Exec(“DELTE FROM item WHERE id = 2”)
在这个语句里,DELETE 拼写错误,但没有任何消息告诉你出了什么问题。如果这个语句是一个大型事务的一部分,那么整个事务就什么事都不会做。通过返回值表示错误不是个问题,但程序员必须去检查返回值,或者把它赋值给 _。
另外,看一下这个语句:
user, err := getUserById(userId)
它并不保证 user 或 err 会包含正确的值。user 有可能没有被赋值(于是读取这个变量就会出现警告),而如果使用了联合类型,编译器只保证正确的那个可以被访问。
- Go 语言里有很多“神奇”的行为。
例如,如果我把源码文件命名为 i_love_linux.go,在 Mac 上就编译不过去。而如果我凑巧把一个函数名定义成 init(),在运行时它会自动被调用。这些都是“约定俗成而非配置(convention over configuration)”的表现。对于小型项目来说,这些都无关键要,但在大型项目中,它们会给你带来大麻烦。
- 因为 Go 语言的首字母大写约定,很容易出现很多相同的标识符。
比如,一些包名、结构体名和变量名都叫作 item。在 Java 中,包名使用了全限定名,类名首字母是大写的。有时候,我觉得 Go 代码不好阅读,因为可能无法一下子看出一个标识符的作用域是怎样的。
- 要进行 Go 代码自动生成并不容易。
编译器太过敏感,一些未被使用的导入和变量也会导致构建失败。在生成大型文件时,它在一开始可能并不知道需要导入那些包,而且可能会出现包名冲突,这种冲突也不好处理,因为即使你知道包名,却不知道导入的符号来自哪里。即使你知道,生成的代码为了避免冲突也会强制使用别名。在 Java 中,这些问题可以通过使用全限定类名来解决,而在 Go 语言中是不能这样做的。
- Go 语言没有三元运算符(?:)。
C 语言风格的编程语言都提供了这个运算符。在使用 Go 语言时,我无时不刻都在想念着这个运算符。当每个人都觉得这个运算符很有用的时候,Go 语言却把它移除掉了。原来很优雅的语句,比如:
var serializeType = showArchived ? model.SerializeAll : model.SerializeNonArchivedOnly
不得不写成这样:
var serializeType model.SerializeTypeif showArchived { serializeType = model.SerializeAll} else { serializeType = model.SerializeNonArchivedOnly}
- sort 接口很笨。
如果你有 10 个不同的结构体需要排序,必须写 30 个函数,其中有 20 个是几乎一样的。而且代码写起来很麻烦,你或许可以把它们委托给另一个 Less(),但你必须保证 Len() 和 Swap() 是兼容的。而且这样的代码看起来很奇怪,转型看起来像是在调用函数:
sort.Sort(sort.Reverse(UsersByLastSignedInAt(users)))
- 缺少泛型。
我真的不想使用一门无法实现 Stack 泛型类的语言。临时的解决方案是使用 append() 函数替代,但现在都什么年代了,我可不想写像下面这样的代码:
stack = append(stack, object)object = stack[len(stack) - 1]stack = stack[:len(stack) - 1]
我很好奇究竟有多少第三方库使用了 interface{}。可见 Go 语言的类型系统设计得非常糟糕。
- 这是个小问题,但足以说明 Go 语言设计者没有完全站在程序员的角度考虑问题。
append() 函数用于扩展一个数组,然后返回新数组:
users = append(users, newUser)
但下面这行代码总是能够调用成功:
append(users, newUser)
append() 函数在可能的情况下进行原地数组修改,如果没有足够的空间就返回一个新的数组。这样的 API 设计真是再糟糕不过了。有多少 bug 是因为忘记把结果赋值给变量引起的?有很多。因为即使是在测试时也不一定会触发数组调整大小(这里需要说明一下,后来他们修改了编译器,如果 append() 函数没有被赋值给变量,就抛出错误)。
总 结
这是我个人对使用 Go 语言的一些建议:如果你的程序很小,基本上可以说清楚要实现什么功能,并且不会与太多外部数据(数据库、Web 等)打交道,那么使用 Go 语言是没有问题的。如果程序很大,有很复杂的数据结构,或者需要处理大量的外部数据,那么 Go 语言的类型系统会让你抓狂,你最好选择其他静态类型的语言(类型系统比较完善)或者动态语言(不会碍手碍脚)。
相关文章:

写扩展性好的代码:函数
http://blog.jobbole.com/107442/转载于:https://www.cnblogs.com/answercard/p/8862006.html

PHP学习笔记:万能随机字符串生成函数(已经封装好)
做验证码用到的,然后就把这个函数封装起来,使用时候要设置2个参数: $str设置里要被采集的字符串,比如: $strefasfgzsrhftjxjxjhsrth; 则在函数里面生成的字符串就回从efasfgzsrhftjxjxjhsrth里面随机抓取; …

python中getopt函数_python getopt模块详解
getopt这个函数 就是用来抽取 sys.argv 获得的用户输入来确定执行步骤。 getopt是个模块,而这个模块里面又有getopt 函数,所以getopt需要这样这样用。 getopt.getopt( [命令行参数列表], "短选项", [长选项列表] ) 该函数返回两个值. opts 和a…

Go语言的前景分析
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 抓住时代的趋势 在上篇文章中,也就是那个跨维度的打击,是可以直接秒杀的中,提到这个时代的问题,这次…

SQL获取当月天数的几种方法
SQL获取当月天数的几种方法 原文:SQL获取当月天数的几种方法 日期直接减去int类型的数字 等于 DATEADD(DAY,- 数字,日期) 下面三种方法: 1,日期加一个月减去当前天数,相当于这个月最后一天的日期。然后获取天数。(注意,…

移动端zepot媒体查询media queries
使用zepot做轮播图<head> <meta charset"utf-8"> <meta name"viewport" content"widthdevice-width, user-scalableno, initial-scale1.0, maximum-scale1.0, minimum-scale1.0"> <title>jd首页</title> <link…

前端try catch是如何捕获异常的_一文告诉你如何优雅处理前端异常?
前端一直是距离用户最近的一层,随着产品的日益完善,我们会更加注重用户体验,而前端异常却如鲠在喉,甚是烦人。一、为什么要处理异常?异常是不可控的,会影响最终的呈现结果,但是我们有充分的理由…

区块链热度不断,那么究竟是泡沫还是未来?
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 区块链究竟是泡沫还是未来,需要落脚到实际运用中去判断。区块链区别于传统服务器,其宗旨不是为现实世界带来某个特定的产品&…

集成 Kendo UI for Angular 2 控件
伴随着 Angular 2 的正式 release,Kendo UI for Angular 2 的第一批控件已经发布了,当前是 Beta 版本,免费使用。 官方站点:Kendo UI for Angular 2 Kendo UI for Angular 被打包成独立的多个 NPM package,在 Progres…

按钮垂直居中_带下拉按钮的动态图表
小伙伴们好啊,今天和大家一起分享一个图表制作的技巧,先来看看效果:这个图表里,其实有三个数据系列,分别是一深一浅两个颜色的条形图,再就是大大的圆圈儿,其实是用散点图模拟出来的。先来看数据…

第一周Access课总结
第一周Access课总结 1:这节课学到了什么? 这节课重点学了数据库是用来干什么 做什么的 老师怕我们理解不了 用了很长时间向我们举了很多的例子 让我们终于知道了数据库是用来干嘛的了 顾名思义 数据库就是存放数据的仓库 是长期存放在计算机内 有组织…

以太坊,EOS和其他DApps的总数达到2,432,但没有大规模采用
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 根据分散应用监测网站StateOfTheDApps,每月创建的新DApps数量的最高水平是2018年12月。去年最后一个月共有179个新的DApps上线。 以太…

docker logstash_用于监视Kubernetes和Docker的六大开源工具
Kubernetes和Docker是在DevOps圈中最常听到的两个词。Docker是一个工具,它使你能够以容器化的方式运行应用程序,Kubernetes是一个用于编排、管理容器的平台——如果你想使用Docker CLI去手动地管理数千个容器,这是不切实际的。然而࿰…

大道至简第一章读后感
当今社会,信息化飞速发展,软件的需求也越来越高,而《大道至简》给我们点透了编程的精义。 大道至简第一是讲的便是编程的精义,文章借愚公移山这个故事来阐释编程的基本思路:出现一个问题(惩山北之塞&#x…

nginx resolver 指令的使用
官方解释下就是:反向代理的场景,upstream后端用域名时,配置resolver以便于nginx能够解析该域名 Configures name servers used to resolve names of upstream servers into addresses 官网地址:http://nginx.org/en/docs/http/ngx…

以太坊和EOS DApp数量上升
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 目前,在最受欢迎的智能合约平台以太坊和EOS上,每月大约有180个新的去中心化应用程序(DApps),该数量处于历史最高…

转载iOS开发中常见的警告及错误
iOS警告收录及科学快速的消除方法 前言:现在你维护的项目有多少警告?看着几百条警告觉得心里烦么?你真的觉得警告又不是错误可以完全不管么? 如果你也被这些问题困惑,可以和我一起进行下面的操作。其实大部分的警告都是…

数学思想 —— 低维与高维的统一
在平面几何有勾股定理:“假设三角形ABC的两边AB、AC互相垂直,则有关系: AB2AC2BC2当我们拓展到空间,类比平面几何的勾股定理并研究三棱锥的侧面面积与底面面积的关系时,我们可得到相应结论:假设三棱锥A-BCD…

python键_在Python中创建键命令
我正在根据我在网上找到的一个脚本编写我自己的简单键记录器。但是,我正在尝试编写一个键命令,以便在键入此命令时记录器程序将关闭。我该怎么做?(我也知道它一点也不安全,但这与这个程序无关) 例如&#x…

对标以太坊的EOS再火,也拼不过InterValue的区块链4.0
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 2017年6月26日上线的数字货币EOS,仅用5天时间,就融了1.85亿美元,一举打破ICO的融资记录。 至2018年4月12日&a…

python中ttk和tkinter_Python tkinter与ttk日历
我正在使用 this代码在我的Tkinter上创建一个简单的日历.当我在主根窗口上放置日历时,日历显示正常.因此,我决定放置另一个按钮,它将创建一个Tkinter顶层窗口并在顶层窗口上放置另外一个日历.但这次它无法显示日历,而是它给了我这个错误,“TclError:无法打包. 18913…

Ubuntu dns
在Ubuntu系统网络设备启动的流程中,会依赖/etc/network/interface的配置文件初始化网络接口,所以直接在/etc/network/interface之中配置好对应的dns服务器会最先写入/etc/resolv.conf。所以我们可以在其中添加如下内容: dns-nameserve…

web图片识别
<!doctype html><html lang"en"><head> <meta charset"UTF-8"> <title>图像识别</title> <script src"../js/jquery.js"></script></head><body><canvas id"canv…

通俗易懂,到底什么是区块链?
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 2017年9月4日,中国政府正式明令禁止ICO和数字货币交易行为,随即关闭了多个数字货币交易所。同时政府也多次声明࿰…

select三级联动 怎么删除前一个的_python测试开发django57.xadmin选项二级联动
前言当我们选择项目分类的时候,一个项目下关联多个模块,同时有这两个选项框的时候,需要实现选中一个项目,模块里面自动删除出该项目下的模块,如下图这种解决基本思路:1.写个jqeury脚本监听change事件2.ajax…

安装wdcp的方法和bug
1.Wdcp安装SSH登录系统,这里讲解源码编译安装和RPM包安装两种 a.源码编译安装 下载源码包wget http://dl.wdlinux.cn:5180/lanmp_laster.tar.gz 解压并安装 tar zxvf lanmp_laster.tar.gz sh in.sh linux安装wdcp之后mysql找不到my.cnf,locate my-medium.cnf。 cp /…

第七周读书笔记
《深入理解计算机系统》从程序设计与性能优化的角度介绍了计算机系统,让我从程序员的角度了解了计算机系统,更深入地理解了硬件、操作系统和编译系统等对应用程序性能和正确性的影响,并掌握了基本的程序优化设计技术,为编写更高效…

是什么限制了区块链技术的应用?
链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载。 2017年已经匆匆离去,回顾过去一整年,似乎区块链应用一直处于隐忍未发的状态,很多项目的落地已处于验证阶段&…

软件包管理(rpmyum)
一、rpm包管理器 rpm是一个功能强大的包管理工具,可用于构建,安装,查询,验证,更新和卸载软件包。 用法: rpm [OPTION...] 配置文件: /var/lib/rpm/ 已安装rpm包的元数据 选项: -i&am…

python yield理解_对Python中Yield的理解
看到下面这段程序的时候,有点不明白这个yield到底是个啥东西,看了网上很多的博客,大致理解了yield的含义,所以记录下来。要说yield首先要说python中的生成器,那么什么是生成器?假设有一个包含一百万个元素的列表&…