go基础面试题汇总第一弹

news/2024/12/22 2:21:22/

init函数是什么时候执行的?

init的函数的作用是什么?

通常作为程序执行前包的初始化,例如mysql redis 等中间件的初始化

init函数的执行顺序是怎样的?

分不同情况来回答:

  1. 在同一个go文件里面如果有多个init方法,它们会按照代码依次执行。
  2. 同一个package文件里面,它们会按照文件名的顺序执行。
  3. 不同的package且不是相互依赖的情况下,按照Import导入的顺序执行。
  4. 不同的package是相互依赖的情况,会优先执行最后被依赖的init函数。

go文件的初始化顺序是怎样的?

  1. 执行引入的包。
  2. 当前包里面的常量变量。
  3. 执行init函数。
  4. 执行main函数。

注意点:

  1. 包相互依赖避免循环导入问题。
  2. 所有文件的init函数都是在一个goroutine内执行的。
  3. 如果一个包在不同的地方被引入多次,但是它的Init函数只会执行一次。

go如何获取项目根目录

可以通过os go内置函数库来获取 os.Getwd()
os.Args[0]
os.Executable
也可以通过环境变量自定义根路径

go中new和make有什么区别?

go和new的区别

meke分配内存的同时会初始化,new只会分配零值填充。
make主要用来给slice,map,channel来初始化,new万能没有限制。
make返回的是原始类型,new返回的是类型的指针*T。
new 申请的值均为零值,对创建映射和切片没有意义。
new可以为任何类型的值开辟内存并返回此值的指针。

数组和切片的区别?

数组和切片的相同点

数组和切片 全部元素类型都必须相同。
数组和切片 所有元素都是连续存储在一块内存中,并且是紧挨着的。

数组和切片的不同点

数组的零值是每个元素类型的零值。
切片的零值为nil。
指针类型的数组和切片直接用类型声明后是nil,不能直接使用。

slice和map有什么区别

map中的元素所在内存不一定是连续的
访问元素的时间复杂度都是O1,但相对于slice来说map更慢。
map的优势
map的key值的类型是任何可以比较的类型。
对于大多数元素为零值的情况,map可以节省大量内存。

切片的底层数据结构是什么?有什么特性?

它的底层主要有三个部分

  1. array 是一个指向底层数组的指针
  2. len 切片的长度,是指当前切片包含元素数量
  3. cap 切片的容量,是指底层数组能够容纳的元素数量
    切片的动态特性
    当切片的len到达cap时,切片需要扩容,在这个扩容的过程中,go内存机制会分配一个新的更大的底层数组,并且将原数组的内容复制到新数组里面,这个过程是runtime包的growlice函数实现的。

切片是如何扩容的?

1.7和1.8后截然不同

切片是否并发安全

需要手动管理并发安全可以用sync.Mutex来确保切片在追加元素的时候避免并发问题。

如何判断两个切片是否相等

在Go语言中,判断两个切片是否相等,需要考虑切片的元素值是否相等以及顺序是否一致。可以使用reflect.DeepEqual函数来进行比较,它能够深度比较两个值是否相等

package mainimport ("reflect""fmt"
)func main() {slice1 := []int{1, 2, 3}slice2 := []int{1, 2, 3}slice3 := []int{3, 2, 1}// 使用reflect.DeepEqual比较两个切片if reflect.DeepEqual(slice1, slice2) {fmt.Println("slice1 and slice2 are equal")} else {fmt.Println("slice1 and slice2 are not equal")}// 使用reflect.DeepEqual比较两个切片if reflect.DeepEqual(slice1, slice3) {fmt.Println("slice1 and slice3 are equal")} else {fmt.Println("slice1 and slice3 are not equal")}
}
package mainimport ("fmt"
)func main() {slice1 := []int{1, 2, 3}slice2 := []int{1, 2, 3}slice3 := []int{3, 2, 1}// 比较元素值是否相等equal := len(slice1) == len(slice2) && len(slice1) == len(slice3)if equal {for i := range slice1 {if slice1[i] != slice2[i] {equal = falsebreak}}}if equal {fmt.Println("slice1 and slice2 have the same elements")} else {fmt.Println("slice1 and slice2 do not have the same elements")}equal = len(slice1) == len(slice3) && len(slice1) == len(slice3)if equal {for i := range slice1 {if slice1[i] != slice3[i] {equal = falsebreak}}}if equal {fmt.Println("slice1 and slice3 have the same elements")} else {fmt.Println("slice1 and slice3 do not have the same elements")}
}

slice作为参数传递是传值还是传指针

,切片(slice)作为参数传递时,实际上是传递了切片的副本,而不是原始切片的指针。
切片作为参数传递时,传递的是切片的副本。
对切片副本的修改不会影响原始切片,除非通过函数返回值获取新的切片。
切片副本和原始切片共享底层数组,直到副本进行了导致底层数组扩容的操作。

package mainimport "fmt"func modifySlice(s []int) {s[0] = 100 // 修改切片副本的第一个元素
}func modifyAndAppend(s []int) []int {s = append(s, 4) // 追加元素,如果容量不够,会重新分配底层数组return s
}func main() {original := []int{1, 2, 3}modifySlice(original)fmt.Println("After modifySlice:", original) // 输出: After modifySlice: [1 2 3]newSlice := modifyAndAppend(original)fmt.Println("After modifyAndAppend:", original, newSlice) // 输出: After modifyAndAppend: [1 2 3] [1 2 3 4]
}

切片的优化技巧

  1. 预先分配足够的容量
  2. 避免在循环中使用append
  3. 使用copy来复制切片
  4. 注意切片的并发还用 用锁机制保证它的并发安全

strings.TrimRight和strings.TrimSuffix有什么区别

strings.TrimRight(s string, cutset string) string
s := "!!!Hello, World!!!"
result := strings.TrimRight(s, "!")
fmt.Println(result) // 输出 "!!!Hello, World"
strings.TrimSuffix(s string, suffix string) string
s := "Hello, World!"
result := strings.TrimSuffix(s, "!")
fmt.Println(result) // 输出 "Hello, World"

区别:
TrimRight关注的是去除尾部的特定字符集,而TrimSuffix关注的是去除尾部的特定后缀字符串。
TrimRight需要两个参数:原始字符串和要去除的字符集;TrimSuffix只需要两个参数:原始字符串和要去除的后缀。
TrimRight会去除尾部所有指定的字符,直到遇到不在cutset中的字符为止;TrimSuffix则只会去除尾部的特定后缀字符串,如果字符串不是以这个后缀结尾的,那么原始字符串不会被修改。

go语言值溢出会发生什么?

在 Go 语言中,数值类型溢出是指当数值超出类型所能表示的范围时,结果会被截断,从而产生不正确的值。Go 语言中的整型(如 int, int8, int16, int32, int64, uint8, uint16, uint32, uint64)都会受到这种限制。

var x uint8 = 255 // uint8 的最大值是 255
x += 1
fmt.Println(x) // 输出 0

避免值溢出
使用适当的数据类型:确保选择的数值类型可以容纳你所需的数值范围。例如,如果你需要存储大整数,应使用 int64 或者 big.Int 而不是较小的整型。
手动检查值是否溢出:在执行操作前检查数值是否接近类型的最大值或最小值。如果快要溢出,可以采取相应的措施。

var x uint8 = 255
if x == math.MaxUint8 {fmt.Println("溢出警告")
} else {x += 1
}

发生溢出后解决方案
检测到错误并抛出异常或警告:通过条件语句来捕捉溢出场景,并采取相应措施。
选择更大范围的类型:在代码中可以考虑将类型更改为 int64、uint64,或者对特别大的数值使用 math/big 包中的 big.Int 类型,这个类型没有固定的大小限制。

import "math/big"
a := big.NewInt(1)
b := big.NewInt(1)
result := new(big.Int).Mul(a, b) // 执行大数乘法
fmt.Println(result)

go语言中每个值 在内存中只分布在一个内存块上的类型有哪些?

总结来说,Go 语言中每个值只分布在一个内存块上的类型主要包括基本类型(如整型、浮点型、布尔型等)、数组和结构体。
有一些类型的值在内存中并不只占据一个连续的内存块,它们涉及到指向其他内存区域的指针或引用:
切片(slice):切片本身是一个描述符,包含指向底层数组的指针、长度和容量,切片描述符存储在一块内存中,但底层数组的元素可能分布在不同的内存块上。
映射(map):映射是一种引用类型,键值对并不会存储在一个连续的内存块中。
字符串(string):字符串是一个包含指向底层字节数组的指针和长度的结构,字符串的底层数据可能不在连续的内存块中。
接口(interface):接口是一个复杂的类型,包含两个部分:类型信息和数据指针。这两部分的内存布局通常不是连续的。

go语言中哪些类型可以使用cap和和len?

len 可用于:array、slice、string、map、channel。
cap 可用于:slice、array、channel。

go语言的指针有哪些限制?

  1. go指针不支持直接进行算术运算。
  2. 一个指针类型的值不能随意转换为另一个指针类型。
  3. 一个指针的值不能随意跟其他类型指针的值进行比较的。
  4. 一个指针的值不能随意被赋值给其它任意类型的指针值。

指针比较需要满足两个条件

  1. 这两个指针类型相同。
  2. 这两个指针之间可以隐式转换。

go语言中哪些类型的零值可以用nil表示

指针、切片、映射、通道、接口类型、函数都可以用nil表示
不同数据类型的nil值的尺寸是不同的。
nil值不一定是可以相互比较的,主要取决于该类型是否可以比较。
可以比较的两个Nil值不一定相等。

go调用函数传入结构体时,是传值还是传指针?

函数的传递参数只有值传递,且传递的实参都是原始数据的一份拷贝
在go语言中赋值操作和函数调用传参都是将原始值的直接部分赋值给了目标值

如何判断两个对象是否完全相同

  1. 基本类型的比较
    对于基本类型(如整型、布尔型、浮点型、字符、字符串等),你可以直接使用 == 运算符比较两个对象的值是否相同。
a := 10
b := 10
fmt.Println(a == b) // true
str1 := "hello"
str2 := "hello"
fmt.Println(str1 == str2) // true
  1. 数组的比较
    Go 语言中的数组支持使用 == 直接比较,前提是数组的元素类型必须是可比较的(如基本类型)。
arr1 := [3]int{1, 2, 3}
arr2 := [3]int{1, 2, 3}
fmt.Println(arr1 == arr2) // true
  1. 结构体的比较
    结构体可以使用 == 进行比较,但前提是结构体的所有字段都是可比较的类型。如果结构体包含了不可比较的字段(如切片、映射、函数等),直接比较会引发编译错误。
type Person struct {Name stringAge  int
}
p1 := Person{Name: "Alice", Age: 25}
p2 := Person{Name: "Alice", Age: 25}
fmt.Println(p1 == p2) // true
    1. 切片(slice)、映射(map)和通道(channel)的比较
      切片、映射、通道 不能直接使用 == 进行比较,除非与 nil 进行比较。要比较两个切片或映射是否完全相同,可以手动遍历每个元素或使用第三方库(如 reflect.DeepEqual)。
var s1 []int = nil
var s2 []int = nil
fmt.Println(s1 == s2) // true, 因为都为 nils1 = []int{1, 2, 3}
s2 = []int{1, 2, 3}
// fmt.Println(s1 == s2) // 编译错误,切片不能直接比较
  1. 接口的比较
    接口可以用 == 进行比较,只有当两个接口的动态类型和动态值都相同的时候,才会被认为是相同的。
var i1 interface{} = 123
var i2 interface{} = 123
fmt.Println(i1 == i2) // true
  1. 使用 reflect.DeepEqual 进行深度比较
    对于无法直接用 == 比较的复杂类型(如切片、映射等),你可以使用 Go 标准库中的 reflect.DeepEqual 进行深度比较。它会递归地比较对象的每个字段和元素,适用于结构体、切片、映射等复杂数据结构。
import ("fmt""reflect"
)s1 := []int{1, 2, 3}
s2 := []int{1, 2, 3}
fmt.Println(reflect.DeepEqual(s1, s2)) // truem1 := map[string]int{"a": 1, "b": 2}
m2 := map[string]int{"a": 1, "b": 2}
fmt.Println(reflect.DeepEqual(m1, m2)) // true

总结
基本类型、数组、结构体(可比较字段)可以使用 == 直接比较。
切片、映射、通道等引用类型不能直接比较,需要使用 reflect.DeepEqual 或手动比较。
接口类型可以使用 == 比较,前提是其动态类型和动态值相同

用两种方法判断一个对象是否拥有某个方法

  1. 利用类型断言(Type Assertion)判断对象是否实现某个接口
    Go 是静态类型语言,但它通过接口提供了动态的特性。通过定义一个接口,并使用类型断言,能够判断某个对象是否实现了该接口(即是否拥有某个方法)。
    示例代码:
    假设要判断某个对象是否有 Speak() 方法:
type Speaker interface {Speak()
}type Person struct{}func (p Person) Speak() {fmt.Println("I can speak!")
}func main() {var obj interface{} = Person{}if speaker, ok := obj.(Speaker); ok {fmt.Println("This object has the Speak method.")speaker.Speak() // 调用该方法} else {fmt.Println("This object does not have the Speak method.")}
}
  1. 使用 reflect 包进行反射判断
    Go 的 reflect 包允许在运行时检查和操作对象的类型和方法。通过反射可以检查一个对象是否拥有某个方法。
import ("fmt""reflect"
)type Person struct{}func (p Person) Speak() {fmt.Println("I can speak!")
}func main() {var obj = Person{}method := reflect.ValueOf(obj).MethodByName("Speak")if method.IsValid() {fmt.Println("This object has the Speak method.")method.Call(nil) // 调用该方法} else {fmt.Println("This object does not have the Speak method.")}
}

总结:
类型断言:通过接口类型断言来判断对象是否实现某个接口(从而拥有对应的方法)。
反射(reflect):利用反射在运行时动态检查对象是否具有指定名称的方法。


http://www.ppmy.cn/news/1534386.html

相关文章

IDEA在git提交时添加忽略文件

在IntelliJ IDEA中,要忽略target目录下所有文件的Git提交,你可以通过设置.gitignore文件来实现。以下是步骤和示例代码: 1、打开项目根目录下的.gitignore文件。也可以先下载这个.ignore插件。 2、如果不存在,利用上面的插件新建…

ElasticSearch备考 -- 多字段查询

一、题目 索引task有3个字段a、b、c,写一个查询去匹配这三个字段为mom,其中b的字段评分比a、c字段大一倍,将他们的分数相加作为最后的总分数 二、思考 通过题目要求对多个字段进行匹配查询,可以考虑multi match、bool query操作。…

从0学习React(5)---通过例子体会setState

上篇文章中,我们讲到了通过setState来更新组件的状态。我觉得上篇文章我讲的已经是比较详细的了,而且讲的很通俗易懂。但是上篇文章终归还是理论,没有实践,所以还是学了个表面而已。这篇文章我就结合一点实践来讲讲怎么使用这个se…

Spring Boot 进阶-Spring Boot 开发第一个Web接口

在前面的文章中我们对Spring Boot的配置以及日志配置有了大概的了解,在我们搭建完成第一个Spring Boot项目之后也提到了一个概念就是RestFul风格的接口开发。下面我们就来详细介绍一下使用Spring Boot如何去开发一个RestFul的Web接口。 准备 在开发接口之前,需要引入的就是W…

鸿蒙harmonyos next纯flutter开发环境搭建

公司app是用纯flutter开发的,目前支持android和iOS,后续估计也会支持鸿蒙harmonyos。目前谷歌flutter并没有支持咱们国产手机操作系统鸿蒙harmonyos,于是乎国内有个叫OpenHarmony-SIG的组织,去做了鸿蒙harmonyos适配flutter开发的…

云原生周刊:Argo CD v2.13 发布候选版本丨2024.9.30

开源项目推荐 Argo Events Argo Events 是一款事件驱动的工作流自动化框架,专门为 Kubernetes 环境开发。 UptimeFlare UptimeFlare 是一个基于 Cloudflare Workers 的免费无服务器监控和状态页开源项目 BunkerWeb BunkerWeb 是一个开源的下一代 Web 应用防火…

项目-坦克大战学习-人机随机生成

想要做到人机随机生成我们需要做到以下几点 1,确定随机生成的位置 2,确定随机生成人机的样式 3,实例化人机 4,绘制人机 在项目中我们由三个位置让人机生成,但是每次生成人机就需要在3个位置中随机挑选一个&#x…

DevExpress WinForms v24.1新版亮点:富文本编辑器、电子表格组件功能升级

DevExpress WinForms拥有180组件和UI库,能为Windows Forms平台创建具有影响力的业务解决方案。DevExpress WinForms能完美构建流畅、美观且易于使用的应用程序,无论是Office风格的界面,还是分析处理大批量的业务数据,它都能轻松胜…