本文始发于个人公众号: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. Slam笔记I

    视觉Slam笔记I 第二讲-三位空间刚体运动 点与坐标系: 基础概念: 坐标系:左手系和右手系.右手系更常用.定义坐标系时,会定义世界坐标系,相机坐标系,以及其他关心对象的坐标系.空间中任意一点可由空 ...

  2. Contest 160

    2019-10-29 16:36:24 总体感受:有一段时间没有打比赛,手居然有生疏的感觉,这次肯定是要掉分了,然后在做combination问题的时候没有敲对代码,很伤. 注意点:依然需要多练习,很 ...

  3. TensorFlow 中文资源精选,官方网站,安装教程,入门教程,实战项目,学习路径。

    Awesome-TensorFlow-Chinese TensorFlow 中文资源全集,学习路径推荐: 官方网站,初步了解. 安装教程,安装之后跑起来. 入门教程,简单的模型学习和运行. 实战项目, ...

  4. MySQL count知多少

    统计一个表的数据量是经常遇到的需求,但是不同的表设计及不同的写法,统计性能差别会有较大的差异,下面就简单通过实验进行测试(大家测试的时候注意缓存的情况,否则影响测试结果). 1. 准备工作 为了后续测 ...

  5. Matlab——m_map指南(4)——实例

    1. 全球/地区温度图 (1)读取数据 clear all setup_nctoolbox %调用工具包 tic %计时 %% nc=ncgeodataset('tmpsfc.gdas.199401. ...

  6. EXPLAIN 查看 SQL 执行计划

    EXPLAIN 查看 SQL 执行计划.分析索引的效率: id:id 列数字越大越先执行: 如果说数字一样大,那么就从上往下依次执行,id列为null的就表是这是一个结果集,不需要使用它来进行查询. ...

  7. ovirt 替换自主签署证书

    需求我自己写了一个python后台,添加上了ovirt 引擎web上,如图 但第一次访问时需要,需要接受两次不安全连接,ovirt  web使用https,我往里面加http,加不进去. 只能同样使用 ...

  8. cxk不会二进制 (贪心)

    cxk不会二进制 Description 最近cxk迷上了二进制,他很菜,有道简单的题不会做,挂在这里求大佬做一下: 以二进制形式给出两个数字:x,y.令s = x + y * 2 ^ k.输出能使 ...

  9. 为何关键字static在面试中频频被问?

    关键字static的神奇妙用在今天的学习中,我了解到关键字static的作用,下面我来给大家分享一下.①static 修饰局部变量只改变了变量的生命周期,让静态局部变量出了作用域依然存在,到程序结束生 ...

  10. Leetcode力扣45题 跳跃游戏 II

    原题目: 跳跃游戏 II 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 你的目标是使用最少的跳跃次数到达数组的最后一个位置. 示例: 输入: ...