百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

刚从别的语言转到 golang,下面几个问题让我琢磨了好久

toyiye 2024-07-03 02:00 19 浏览 0 评论

之前是 PHP/Java 程序员,转到用 golang 开发,下面几个问题让我着实学习了好久才弄明白

  • 谜一般的 GOPATH,到底该怎么配!

  • 为什么有 error,还需要 panic?

  • golang 的接口是一种什么样的存在?

  • goroutine panic 居然会导致进程退出!!!

  • channel 满天飞,这样真的好么?

咱尽量简明扼要,只谈要点。

谜一般的 GOPATH,到底该怎么配!

官方一开始的设想是这样的

my-gopath/

└── src

├── github.com

│ └── project1

└── golang.com

└── project2

把 my-gopath 设置为 GOPATH,然后这个就是你开发的工作目录了。src目录下有你所有的代码。但是这种本机设置一个 GOPATH 的模式会导致非常多的问题

  • 项目之间的依赖不隔离,容易错误引用到不希望的依赖

  • 无法分清楚哪些代码是自己的,哪些代码是别人的

  • 提交代码到 git 仓库的时候怎么办?

所以我们实际的目录结构应该是

my-projects/

├── project1 (GOPATH)

│ └── src

│ └── github.com│ └── project1 (GIT根目录)

│ └── vendor

│ ├── github.com│ │ ├── others-lib1

│ │ └── others-lib2

│ └── vendor.json

└── project2 (GOPATH)

└── src

└── golang.com

└── project2 (GIT根目录)

这个结构是普通人类第一眼就能想出来的么!!!这是我经过无数次的试错之后发现的最佳的目录设置

  • 项目隔离的需求:每个项目有自己的单独的 GOPATH。

  • GIT 提交的需求:在 git clone 之前把 my-projects/project1/src/http://github.com给创建出来,然后 git clone project1 到这个目录里。

  • 区分自己和别人的代码:使用 vendor 目录

你以为这就完美了?如果你要提供一个build.sh在编译机上做打包怎么办?编译机上的GOPATH 怎么设置呢?最佳实践:在 build.sh 里自己创建一个完整的 GOPATH(下图里的tmp目录),用符号链接指向自己。

my-projects/

├── project1

│ └── src

│ └── github.com│ └── project1

│ ├── tmp

│ │ └── src

│ │ └── github.com│ │ └── project1 -> ../../../../project1

│ └── vendor

│ ├── github.com│ │ ├── others-lib1

│ │ └── others-lib2

│ └── vendor.json

└── project2

└── src

└── golang.com

└── project2

这是一种什么样的妖孽啊!另外友情提示一个坑,Intellij 的 golang 插件和指向上级目录的符号链接不兼容,会导致无法自动提示。把 GOPATH 调整到 IDE 高兴,git 高兴,项目隔离,build.sh 可工作,不知道花了多少时间。

为什么有 error,还需要 panic?

panic 表示进程内的错误。panic 的原因来自于代码的逻辑 bug,比如强制类型转换失败,比如数组越界。这个代表了程序员的责任不到位,导致了程序的panic。

error 代表进程外的错误。比如输入符合预期。比如访问外部的服务失败。这些都不是程序员可以设计控制的。这些情况的错误处理是业务逻辑的一部分。

Java 在设计的时候 checked exception 就是 error,runtime exception 就是 panic。但是玩崩了。checked exception 和 error 一样都是想强制让程序员思考 error 的业务逻辑,但是没有成功。

golang 的接口是一种什么样的存在?

public function myFunc(SomeClass someObj) SomeResponseClass {

someObj.method1();

someObj.method2();

}

Java 这样的语言设置会导致的问题是容易导致依赖于实现,而不是依赖于接口。也就是这个 myFunc 依赖的输入可能只需要一个 method1(), method2(),但是 SomeClass 上除了这两个方法之外还有很多其他的行为。把输入接口设置为 SomeClass,导致了接口的“扩大化”。

public myFunc($someObj) {

$someObj->method1();

$someObj->method2();

}

PHP 的写法其实要比 SomeClass 更好。动态语言是 “duck typing”的,也就是你只要给一个实现了 method1(),method2() 的方法的对象,那么就能够调用成功。也就是 myFunc 的接口恰到好处的,不会因为类型声明而使得接口扩大化。但是 PHP 的问题是,不看实现,你永远不知道应该传一个什么样的 obj 进来。接口模糊了,导致调用方要查看对方的实现。

public function myFunc(SomeInterface someObj) SomeResponseClass {

someObj.method1();

someObj.method2();

}

Java 为了解决传实体类的问题,创造了 interface。interface 很好的解决了接口”扩大化“的问题。你可以给myFunc的需求,创造一个恰到好处的interface。但是 Java 的 interface 的问题是,需要 class 定义方的配合。我如果声明了一个interface,需要传入的对象在定义的时候就写了”implement interface"。相比动态类型来说,这就很不方便了。

综上

  • Java Class,依赖具体实现,而不是接口。导致接口扩大化

  • PHP Object,依赖的是接口,不存在接口扩大化的问题。但是调用方很痛苦,需要来看你的内部实现才知道你的接口是什么。

  • Java Interface,依赖接口,但是需要提前定义。无法随时按需定制接口。实践中仍然会导致 interface 的扩大化。

golang 的实现兼顾了 Java 的静态类型,和 PHP 的 duck typing 的好处。它使得你可以给 myFunc 定制一个最精确的接口依赖。只要实现了 method1() 和 method2() 的对象,自动就符合了调用的条件,可以被传入。这个行为非常类似 duck typing。但是相比纯动态语言的 duck typing,golang interface 又有一个肉身的实体存在,可以很方便查看。其实当我们在动态语言里做 duck typing 的时候

public myFunc($someObj) {

$someObj->method1();

$someObj->method2();

}

// 这里对$someObj 的使用,隐式地定义了$someObj 的interface,只是这个interface缺少一个肉身

golang 的interface就是避免了duck typing缺陷的,duck typing。其鼓励地行为是给你的函数定义“精确”地依赖接口,不要过大,也不要过小,精确。

这种精确也体现在了返回值允许多个上面。如果不允许返回多个返回值,我们被迫返回一个结构体。而定义很多小的结构体是非常麻烦的。这个实践中,就会导致很多人写没有返回值的的函数,把返回值隐藏到对一个大的结构体的变更中。或者返回一个很宽泛的结构体(比如map),仅仅因为给每个方法定义一个struct作为response太麻烦了。

golang简单务实,让你返回多个返回值,这样你就可以避免去定义一大堆小的struct来代表函数的返回值。目标就是让你精确地定义函数的输入输出。

goroutine panic 居然会导致进程退出!!!

前面说了,checked exception 是 error。runtime exception 是 panic。

Java 的 thread 里抛里抛异常,thread 挂掉,但是进程不挂掉。

Go 的 goroutine 里panic,整个进程挂掉。

goroutine 必须经过包装使用。

goroutine 越多,代码的线索就越多。线索越多,线索打结的可能性就越高。千万不要随手搞一个 go,fork 一堆 goroutine 出来。

channel 满天飞,这样真的好么?

channel 是一个很新鲜的东西。只需要一天就可以学会channel,然后需要剩下的时间让你忘掉它。

  • channel 不是一种抽象手段,不要用 channel 来组装逻辑。channel 是并发的控制手段,不牵涉并发的,不要过度设计。

  • channel 不能用来搞进程内的微服务,你写一块逻辑,我写一块逻辑,我们之间用 channel 通信。因为 rpc 是同步的,相对好掌控。channel 是纯异步的,你们搞不定。别说我看不起你们。

  • channel 漫天飞导致 goroutine 漫天飞。如非必要,勿增实体。goroutine 越多,代码理解起来复杂度成指数增加

You have to be this tall to use go/channel

go 相比 thread,channel 相比 lock,再简化也是有复杂度的。这些东西能不碰就不碰。它们始终是控制并发的工具,组织业务逻辑还是靠朴实的函数套函数吧。

任何一个 go/channel 关键字出现的地方,想一下是不是必要的。再想一下是不是必要的。再仔细想一下是不是必要的。

总结

  • 按照我给的模板使用 GOPATH,虽然看起来很不优雅

  • 你控制不了的进程外错误用 error,凡是程序员的锅,用panic

  • golang 鼓励你精确定义函数的输入和输出,感谢 golang。error 作为函数接口的一部分,checked exception 未竟的事业在 golang 发扬光大。

  • 任何一个 go/channel 关键字出现的地方,三思再三思

免责声明:转载自网络 不用于商业宣传 版权归原作者所有 侵权删

相关推荐

为何越来越多的编程语言使用JSON(为什么编程)

JSON是JavascriptObjectNotation的缩写,意思是Javascript对象表示法,是一种易于人类阅读和对编程友好的文本数据传递方法,是JavaScript语言规范定义的一个子...

何时在数据库中使用 JSON(数据库用json格式存储)

在本文中,您将了解何时应考虑将JSON数据类型添加到表中以及何时应避免使用它们。每天?分享?最新?软件?开发?,Devops,敏捷?,测试?以及?项目?管理?最新?,最热门?的?文章?,每天?花?...

MySQL 从零开始:05 数据类型(mysql数据类型有哪些,并举例)

前面的讲解中已经接触到了表的创建,表的创建是对字段的声明,比如:上述语句声明了字段的名称、类型、所占空间、默认值和是否可以为空等信息。其中的int、varchar、char和decimal都...

JSON对象花样进阶(json格式对象)

一、引言在现代Web开发中,JSON(JavaScriptObjectNotation)已经成为数据交换的标准格式。无论是从前端向后端发送数据,还是从后端接收数据,JSON都是不可或缺的一部分。...

深入理解 JSON 和 Form-data(json和formdata提交区别)

在讨论现代网络开发与API设计的语境下,理解客户端和服务器间如何有效且可靠地交换数据变得尤为关键。这里,特别值得关注的是两种主流数据格式:...

JSON 语法(json 语法 priority)

JSON语法是JavaScript语法的子集。JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔花括号保存对象方括号保存数组JS...

JSON语法详解(json的语法规则)

JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔大括号保存对象中括号保存数组注意:json的key是字符串,且必须是双引号,不能是单引号...

MySQL JSON数据类型操作(mysql的json)

概述mysql自5.7.8版本开始,就支持了json结构的数据存储和查询,这表明了mysql也在不断的学习和增加nosql数据库的有点。但mysql毕竟是关系型数据库,在处理json这种非结构化的数据...

JSON的数据模式(json数据格式示例)

像XML模式一样,JSON数据格式也有Schema,这是一个基于JSON格式的规范。JSON模式也以JSON格式编写。它用于验证JSON数据。JSON模式示例以下代码显示了基本的JSON模式。{"...

前端学习——JSON格式详解(后端json格式)

JSON(JavaScriptObjectNotation)是一种轻量级的数据交换格式。易于人阅读和编写。同时也易于机器解析和生成。它基于JavaScriptProgrammingLa...

什么是 JSON:详解 JSON 及其优势(什么叫json)

现在程序员还有谁不知道JSON吗?无论对于前端还是后端,JSON都是一种常见的数据格式。那么JSON到底是什么呢?JSON的定义...

PostgreSQL JSON 类型:处理结构化数据

PostgreSQL提供JSON类型,以存储结构化数据。JSON是一种开放的数据格式,可用于存储各种类型的值。什么是JSON类型?JSON类型表示JSON(JavaScriptO...

JavaScript:JSON、三种包装类(javascript 包)

JOSN:我们希望可以将一个对象在不同的语言中进行传递,以达到通信的目的,最佳方式就是将一个对象转换为字符串的形式JSON(JavaScriptObjectNotation)-JS的对象表示法...

Python数据分析 只要1分钟 教你玩转JSON 全程干货

Json简介:Json,全名JavaScriptObjectNotation,JSON(JavaScriptObjectNotation(记号、标记))是一种轻量级的数据交换格式。它基于J...

比较一下JSON与XML两种数据格式?(json和xml哪个好)

JSON(JavaScriptObjectNotation)和XML(eXtensibleMarkupLanguage)是在日常开发中比较常用的两种数据格式,它们主要的作用就是用来进行数据的传...

取消回复欢迎 发表评论:

请填写验证码