人走茶凉|如何避免用动态语言的思维写Go代码( 二 )

  • panic: assignment to entry in nil map
  • panic: invalid memory address or nil pointer dereference
  • 第一个错误是因为对一个未初始化的map进行赋值导致的 , 所以使用map类型的变量前要记得用make函数对变量进行初始化 , 与map类似的切片在使用append函数 向nil slice追加新元素就可以 , 原因是append函数会生成新的切片 , 在底层为切片分配了底层数组 。
    第二个错误是对nil指针进行了解引用导致的 , 指针的零值nil与*T{}并不相等 。 所以指针类型的变量在使用前要注意使用new函数进行初始化 。
    还有就是前端同学们非常不喜欢接口返回值的字段有数据的时候是个列表 , 没数据的时候是Null , 这也是切片未初始化导致的 , 如果数据库里没查到数据 , 那么在代码逻辑里就执行不到给切片append数据的循环里 , 所以就会出现这个问题 。 这是一个保持接口字段类型一致性的一个很重要的细节 。
    使用error返回函数错误在使用PHP时 , 函数的错误是通过抛出异常 , 甚至是通过返回0 , false之类的值来表示函数遇到的错误(这种 , 即使写PHP也不推荐这种做法)
    比如好的写法 , 可这样写:
    public function updateUserFavorites(User $user, $favoriteData){try {// database execution......} catch (QueryException $queryException) {throw new UserManageException(func_get_args(), 'Error Message', '501' , $queryException);}return true;}但很多的人会这么写:
    public function updateUserFavorites(User $user, $favoriteData){// database executionif ($conn.AffectedRows <= 0) {return false}return true;}在Go语言里虽然没有异常机制 , 但是可以让函数返回error明确遇到的错误 。 所以除非确定函数不需要返回error , 多数情况下我们的函数都是需要返回error的 , 所以在定义函数时要明确 , 返回的数据和error的区别 , 两种返回值的职责范围不一样 。 要通过函数返回的error是否为空 , 而不是返回数据是0或者false之类的值判断函数是否执行成功 。
    谨慎使用map[string]interface{}做参数写过PHP的同学都知道 , PHP里的数组近乎万能 , 可以用来当列表、字典 , 而且当字典用时还能保证字典key的遍历顺序 , 这点是很多语言的字典类型办不到的事情 。
    很多刚从PHP转到用Go开发的同学还是带着在PHP里使用数组参数的习惯 , 那么在Go语言里 , 最像PHP数组的可能就是map[string]interface{}了 。
    这种还是典型的动态语言编程的思维 , 在使用Go的时候 , 针对比较复杂的代表一类事物的参数 , 我们也是应该先定义结构体 , 然后使用结构体指针或者结构体指针切片作为参数 。 尽量不使用map[string]interface{}这种类型的参数 , IDE也没法帮助提示这些参数的内部结构 , 这让其他人使用这个代码时就会很苦恼 , 还得先看看函数实现里具体用到了字典的哪些键 。 比如下面这两个函数的对比:
    type UserInput struct{NamestringAgeint32Password string}func AuthenticateUser(input *UserInput) error {findUser(input.Name, input.Password)...}func DummyAuthenticateUser(input map[string]interface{}) error {findUser(input["name"], input["password"])...}一般在业务级别的程序开发里 , 我们要传递存储在数据表里的额外信息的时候才会使用到map[string]interface{}类型的参数 。 写表前把这部分数据编码成JSON格式再写入 , 当然这个主要看使用场景 , 凡事没有绝对 , 这里只是强调一些在编码习惯上的问题 。
    总结最近两年在学习中我写了不少Go语言的文章 , 其中Web编程入门和Go并发编程这两个系列我自认为还是对新手很有帮助的 。
    来源:网管叨bi叨