本文始发于个人公众号: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. 编译原理-第三章 词法分析-3.7 从正则表达式到自动机-DFA最简化

    DFA最简化 一.构造最简DFA 1.输入输出 2.步骤 3.注意点 4.代码 二.示例 例1: 例2: 参考--慕课-苏州大学

  2. [leetcode] 位操作题解

    子集 题目[78]:给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集). 示例: 输入: nums = [1,2,3] 输出: [ [3],   [1],   [2],   [ ...

  3. 谷歌2019 学术指标发榜:CVPR首次进入Top 10,何恺明论文引用最高!

    [导读]今天,谷歌发布了2019最新版学术指标,对收录的会议和期刊的影响力进行排名.AI类的多个顶会进入榜单Top 100,CVPR更是进入前10,而何恺明的"深度残差网络"单篇引 ...

  4. 50行代码实现GAN | 干货演练

    2014年,Ian Goodfellow和他的同事发表了一篇论文,向世界介绍了生成对抗网络(GAN).通过对计算图和博弈论的创新性组合,他们表明如果有足够的建模能力,两个相互对抗的模型可以通过普通的反 ...

  5. HDU - 1962 二分图最大匹配模板(扑克牌得分最大)

    题意: 直接说数据,第一行给定几组数据,每一组数据的第一行是两个人扑克牌分别的数量,第一行是亚当的扑克牌,第二行是夏娃的扑克牌,每一个扑克牌的大小用两个字符来表示,第一个表示是几号扑克牌,第二个表示扑 ...

  6. socket,实现服务器和客户端对话

    服务器: #define _CRT_SECURE_NO_WARNINGS#include<stdio.h>#include<string>#include<WinSock ...

  7. python学习第二节 数据类型、字符编码、文件处理

    标准数据类型 Python3 中有六个标准的数据类型: Number(数字) String(字符串) List(列表) Tuple(元组) Sets(集合) Dictionary(字典) 数字 #整型 ...

  8. 基于Java的数字货币交易系统的架构设计与开发

    前言 无论是股票交易系统,还是数字货币交易系统,都离不开撮合交易引擎,这是交易平台的心脏.同时,一个优秀的架构设计也会让交易平台的运维和持续开发更加容易.本文基于对开源项目的深入研究,总结了数字货币交 ...

  9. Java 程序该怎么优化?(工具篇)

    程序员:为什么程序总是那么慢?时间都花到哪里去了? 面试官:若你写的 Java 程序,出现了性能问题,该怎么去排查呢? 工欲善其事必先利其器,为你呈上一箩筐性能优化工具,必有一款满足你,废话不多说,直 ...

  10. [原创] 关于步科eview人机界面HMI的使用 - HMI做Slave - Modbus RS485通讯

    做测试设备,或者自动化设备常常用到HMI 触摸屏 我有个案子用到了 步科的eview 触摸屏 型号 ET070 我的是单片机主板 控制 HMI显示,通讯用485  MODBUS 单片机板充当 主控 , ...