本文始发于个人公众号:TechFlow,原创不易,求个关注

今天是Golang专题的第二篇,我们来看看Go的语言规范。

在我们继续今天的内容之前,先来回答一个问题。

有同学在后台问我,为什么说Golang更适合分布式系统的开发?它和Java相比有什么优势吗?

其实回答这个问题需要涉及很多概念,比如操作系统当中关于进程、线程、协程等很多概念。我们将这些内容进行简化,举一个最简单的线程的例子。我们来写一段在java当中实现多线程的例子:

public class MyThread implemnts Runnable {
    public void run() {
      System.out.println("I am a thread")
    }
}

public class Test {
    public static void main(String args[]) {
        MyThread thread1 = new MyThread();
        thread1.start();
    }
}

我们再来看看Golang:

func run() {
    fmt.Println("I am a Thread")
}

func main() {
    go run()
}

这么一对比是不是简单很多?

Golang的语言规范

大家都知道程序员最大的分歧之一就是花括号到底应该写在哪一行,有另写一行的,也有跟在循环体后面的。这两拨人分成了两个流派,彼此征战不休,也衍生出了许多段子。

为了统一风格,很多语言对代码风格做了规范。比如Python就去掉了花括号,而使用空格来进行代码缩进。然而不幸的是,有些人缩进用四个空格,也有些人用tab,这双方又形成了阵营,彼此争吵不停……

也许Golang的开发者曾经饱受代码风格争吵的苦恼,所以Golang做了一个划时代的事情,它严格限制了代码风格,强行统一大家都必须使用同一套风格,否则就分分钟报错给你看。所以在我们进行具体的语法学习之前,先从语言规范开始,否则等我们后面养成了不好的习惯再想要改正就会成本很高。其实改正代码风格是一件很难的事情,老实说我的代码风格不是很好,总是使用一些cur、pnt、node、u、v这种简单的变量,这也是当年打acm留下来的习惯,想改一时半会蛮难的。所以大家一定要在初期就养成好习惯,坏习惯就留给我一个人吧(大雾)。

package规范

Golang的语言规范很多,涉及的面很广,有些我们暂时用不到,我们先挑基础的说。首先是package规范,对于package来说它的名字应该和目录保持一致,采取有意义的包名,不要起一些别人看不懂的名字。比如test、unit这种,并且不能和标准库冲突。

其次是我们在引包的时候,需要注意不要使用相对路径,而应该使用绝对路径。

// wrong
import "../../../repo"

// correct
import "github.com/repo/package

当然我们可以装一个goimport工具,帮助我们自动引包。但是自动引包也会有坑,尤其是当目录下存在两个包名称一样的时候,有可能会引入错误,需要我们自己留意。

代码风格规范

Go语言当中规定了我们应该使用驼峰标准来命名变量,不能使用_。在Go当中首字母大写表示结构体中的变量或者是包中的函数public,如果是小写则表示是private,这一点尤其需要注意。刚开始写go的时候都会很不习惯,因此踩坑是常有的事。

golang当中是有常量的,golang当中的常量一样用驼峰标准,首字母大写。比如我们起一个常量叫做app_env,表示当前app运行的环境,我们必须要这样定义:

const AppEnv = "env"

另一点是Golang的设计者认为行尾加上分号毫无必要,所以在编译器当中添加了会在行尾自动加上分号的功能。所以我们可以加也可以不加,但是一般认为没有必要这么做。所以普遍来说,除了在循环体或者是判断条件当中,我们一般是不写分号的。当然也有特殊情况,比如你想要把多条语句写在一行的时候:

var a int; var b float;
a = 3; b = 3.2;

当然还是一般不推荐这么干,建议分成多行,更加美观。

另外一点是golang当中所有的变量和包都必须用上,不允许定义没有使用的东西,否则也会报错。也就是说严格限制了我们写代码时候的谨慎。不能随意申请用不到的变量,大多数语言当中没有这样的限制,但是golang当中做了限制,所以我们写代码的时候要小心。

另外一点是关于花括号,在golang当中严格限制了花括号写在当前行,而不是另起一行。

// wrong
if expression 
{
  ...
}

// correct
if expression {
  ...
}

从上面这个例子我们还可以注意到一点,就是在golang当中if后面的条件不加括号,这点和Python一样。但是如果你写惯了java或者是C++刚开始可能会不太适应。

最后一点是golang的代码规范检测工具golint当中规定了所有的函数以及结构体头部必须要写注释,并且对注释的规范也进行了限制。注释的规范是名称加上说明,如果不写或者是不规范的话,代码虽然可以运行,但是无法通过golint的规范检测。一般来说公司的开发环境都会做限制,只有通过golint规范检测的代码才可以提交发布。

// HelloWorld print hello world
func HelloWorld() {
    fmt.Println("Hello World")
}

另外一点是golang不支持隐式类型转换,比如int和int32以及int64,会被视作是不同的类型。如果我们将一个int32的变量赋值给int类型,则会引起报错,必须要我们手动转换。这当然增加了编码时候的工作,但是也避免了很多由精度不一样产生的问题。

除了这些之外,golang当中还定义了对结构体定义以及错误处理等内容的规范。但是对于我们初学者而言,目前这些是必须要了解的,其他的内容可以等我们后续遇见了再熟悉。

一门语言对于代码风格做了严格的规范限制对于初学者而言可能是一件比较蛋疼的事情,因为要记的东西变多了,我们不仅要学会语法,还要搞清楚这些规范。但是当我们熟悉了或者是工作了之后,会发现这其实是一件好事。对于多人协作的场景而言,大家都遵守一样的规范会大大提升代码交流以及协作的效率。如果你们看过其他代码风格和自己完全不同的人的代码之后,相信你们对于这点一定会有更深的认识。

总结

从规范的严格程度以及对面向对象的阉割程度看起来,golang简直不像是一门新生的语言,倒有些上世纪老派语言的风格。但是偏偏golang又有很多新鲜特性,比如允许函数值返回多个结果,支持匿名函数以及部分函数式编程的功能等等。在初学的阶段,我也非常抗拒它,可能是因为Python写得太多了,习惯了动态语言。但是随着对这门语言了解的深入,我越来越多地发现了它这些设计理念背后的思考和智慧,慢慢对它改观,时至今日,我已经不再怀疑这是一门优秀的语言,这几年的流行并不是没有道理的。

另外很重要的一点是,因为golang太特立独行了,所以经常会让我思考它这么做背后的用意是什么?这么一思考,加上查阅一些资料,能够发现很多之前思维当中的盲点。在之前学习语言的时候,我是绝对不会去思考语言的设计者为什么要这么设计的,只会依葫芦画瓢,照着把相关的内容学会仅此而已。这样的思考除了能够提升对于语言本身的理解之外,也能够提升对问题场景的思考和理解,对于工程师而言,后者其实是更为重要的。

当然这些内容我光说是没有用的,也需要屏幕前的你用心去体会。

希望大家都能感受到golang的魅力,都能在此过程当中收货成长,加油!如果觉得有所收获,请顺手点个在看或者转发吧,你们的举手之劳对我来说很重要。

Golang——详解Go语言代码规范的更多相关文章

  1. 详解Go语言调度循环源码实现

    转载请声明出处哦~,本篇文章发布于luozhiyun的博客: https://www.luozhiyun.com/archives/448 本文使用的go的源码15.7 概述 提到"调度&q ...

  2. 详解go语言的array和slice 【二】

    上一篇已经讲解过,array和slice的一些基本用法,使用array和slice时需要注意的地方,特别是slice需要注意的地方比较多.上一篇的最后讲解到创建新的slice时使用第三个索引来限制sl ...

  3. 详解 Go 语言中的 time.Duration 类型

    swardsman详解 Go 语言中的 time.Duration 类型swardsman · 2018-03-17 23:10:54 · 5448 次点击 · 预计阅读时间 5 分钟 · 31分钟之 ...

  4. SQL Server 表的管理_关于事务的处理的详解(案例代码)

    SQL Server 表的管理_关于事务的处理的详解(案例代码) 一.SQL 事务 1.1SQL 事务 ●事务是在数据库上按照一定的逻辑顺序执行的任务序列,既可以由用户手动执行,也可以由某种数据库程序 ...

  5. SQL Server 表的管理_关于数据增删查改的操作的详解(案例代码)

    SQL Server 表的管理_关于数据增删查改的操作的详解(案例代码)-DML 1.SQL INSERT INTO 语句(在表中插入) INSERT INTO 语句用于向表中插入新记录. SQL I ...

  6. SQL Server 表的管理_关于表的操作增删查改的操作的详解(案例代码)

    SQL Server 表的管理_关于表的操作增删查改的操作的详解(案例代码) 概述: 表由行和列组成,每个表都必须有个表名. SQL CREATE TABLE 语法 CREATE TABLE tabl ...

  7. SQL Server 表的管理_关于事务操作的详解(案例代码)

    SQL Server 表的管理_关于事务操作的详解(案例代码) 1.概念 事务(transaction): 是将多个修改语句组合在一起的方法,这个方法中的所有语句只有全部执行才能正确完成功能.即要么全 ...

  8. http500:服务器内部错误案例详解(服务器代码语法错误或者逻辑错误)

    http500:服务器内部错误案例详解(服务器代码语法错误或者逻辑错误) 一.总结 服务器内部错误可能是服务器中代码运行的时候的语法错误或者逻辑错误 二.http500:服务器内部错误案例详解 只是一 ...

  9. 详解 ESLint 规则,规范你的代码

    在很久之前就想通过工具来规范自己的代码风格,减少程序出错的概率,如果看过我的 一个前端程序猿的Sublime Text3的自我修养 ,这篇博客的朋友,肯定知道在当时我使用 SublimeLinter- ...

随机推荐

  1. 微信小程序开发-小程序之间的跳转

    前几天开发微信小程序,其中有个需要联动宣传的业务,就是正在开发的小程序跳转到别的小程序去, 然后去看了下大家的做法与看法,总结下这小程序跳转之间应该注意到的几个问题 首先是跳转的方法, https:/ ...

  2. jupyter之配置自己喜欢的python环境

    之前安装anaconda,会自动安装jupyer笔记本,但是本人从来没有用过,近期查看github上的一些教程时,发现很多文件都是.ipynb为后缀的jupyter文件,于是自己准备琢磨一下把环境换成 ...

  3. hdu3973 AC's String 线段树+字符串hash

    题目链接:http://icpc.njust.edu.cn/Problem/Hdu/3973/ 题意是:给出一个模式串,再给出一些串组成一个集合,操作分为两种,一种是替换模式串中的一个字符,还有一种是 ...

  4. 数学-绝对值-Reverse Subarray To Maximize Array Value

    2020-02-11 12:01:21 问题描述: 问题求解: 本题的难度个人感觉还是蛮大的,主要是不容易想到O(n)的解. 对于 ...a, [b, ... , c], d, ...,如果我们将其中 ...

  5. MySQL datetime类型详解

    研发反馈问题,数据库中datetime数据类型存储的值末尾会因四舍五入出现不一致数据,影响查询结果,比如:程序中自动获取带毫秒精度的日期'2019-03-05 01:53:55.63',存入数据库后变 ...

  6. GB2312,GBK和UTF-8的区别

    GBK GBK包含全部中文字符, GBK的文字编码是双字节来表示的,即不论中.英文字符均使用双字节来表示,只不过为区分中文,将其最高位都定成1.至于UTF-8编码则是用以解决国际上字符的一种多字节编码 ...

  7. PyTorch专栏(八):微调基于torchvision 0.3的目标检测模型

    专栏目录: 第一章:PyTorch之简介与下载 PyTorch简介 PyTorch环境搭建 第二章:PyTorch之60分钟入门 PyTorch入门 PyTorch自动微分 PyTorch神经网络 P ...

  8. 数据库(sqlserver 2005)优化排查之路

    查找问题过程是痛苦的,解决完问题是快乐! 兄弟帮助一个公司开发了一个旅游网站(asp.net+sqlsever2005),一直还算稳定,但是最近网站却慢的可以,让人头疼.登录服务器,进入任务管理器,发 ...

  9. CentOS6.10下安装mysql-5.7.24

    卸载原有mysql 因为mysql数据库在Linux上实在是太流行了 所以目前下载的主流Linux系统版本基本上都集成了mysql数据库在里面 我们可以通过如下命令来查看我们的操作系统上是否已经安装了 ...

  10. Python中类型的概念(一)

    本课程主要介绍6种Python语言中的类型:数字类型.字符串类型.元组类型.列表类型文件类型.字典类型 1.数字类型 Python语言包括三种数字类型:整数类型.浮点数类型.复数类型 (1)整数类型 ...