[面试] Golang 面试题

[面试] Golang 面试题本文章收录于:后端工程师面试题目总结(提供参考答案)目录1.make与new的区别2.简要描述go中的main和init函数的区别3.下面的代码输出什么,若会报错报什么错?4.这段代码会输出什么?5、简述channel和mutex锁机制的原理异同与使用场景6、sync.WaitGroup的使用场景?7、写一段闭包代码,阐述其作用8、执行这段代码会发生什…

大家好,又见面了,我是你们的朋友全栈君。

本文章收录于:后端工程师面试题目总结(提供参考答案)

目录

1. make与new的区别

2. 简要描述go中的main和init函数的区别

3. 下面的代码输出什么,若会报错报什么错?

4.  这段代码会输出什么?

5、简述channel和mutex锁机制的原理异同与使用场景

6、sync.WaitGroup的使用场景?

7、写一段闭包代码,阐述其作用

8、执行这段代码会发生什么?

9、单例实现

10、这段代码输出什么?

11、这段代码可以编译过吗,如果会错是在哪一行?

12、ABCD哪一行会报错?

13、下面的代码会怎样输出?每次输出结果一样吗?


1. makenew的区别

Make 用于map、slice 和channel几种类型的内存分配。并且返回一个有初始值的对象,注意不是指针。

注:channel在make之后打印出来的也是内存地址,是个特殊类型。

New 用于使用type声明的类型的内存分配。new(T)分配了零值填充的T类型的内存空间,并且返回其地址,即一个T类型的值。用Go的术语说,它返回了一个指针,指向新分配的类型T的零值。有一点非常重要:new返回指针。

2. 简要描述go中的maininit函数的区别

首先,这两个函数应用位置不同,init函数可以应用于所有的package,main只能应用于 package main,需要注意的是虽然一个package中可以写任意多个init,但是无论是从可读性还是可维护性来说,都是不推荐的; 其次,这两个函数定义时都不能有任何的参数和返回值, 最后,个人理解,init函数为初始化操作,main函数为程序入口。

一图胜千言

[面试] Golang 面试题

3. 下面的代码输出什么,若会报错报什么错?

func main(){
	/*
	先defer的后执行
	recover后输出panic中的信息
	*/
	defer func(){
		If err:=recover();err!=nil{
			fmt.Print(err)
		} else {
			fmt.Print("no")
		}
	}()
	defer func(){
		panic("1111111111111")
	}()
	panic("22222222222")
}

答案:不会报错,而是被recover捕捉,打印出1111111111111

4.  这段代码会输出什么?

[面试] Golang 面试题

答案:先打印出111111111,然后报错(死锁)

说明:两个协程都是读取c2中的元素,然后塞入c1, 然而c2又是无缓冲且没有任何协程在往c2中写数据,所以第一个协程读c2的时候就导致死锁,因为c2永远读不出数据。 如何改能让程序不报错?在第一个协程前加一个协程:

gofunc(){ 
    c2<-1 
}()

5、简述channel和mutex锁机制的原理异同与使用场景

channel原理: 当channel能存放的元素数量为0时表示为阻塞型channel。当管道无数据时,需要从管道取数据的协程会被阻塞,不会向下执行。 所以可以通过多个协程应用同一个channel,从而实现协程间的同步。

channel使用场景: 1. 需要协程通信时     2. 需要管道传输数据时。

Mutex原理: 互斥锁用来防止资源竞争,多个协程使用同一把排它锁时,原来是并发运行的话将变为线性运行。

排它锁针对任意操作都是排它的,没有读写区分。 若一个goroutine获得锁,则其他goroutine会一直阻塞到他释放锁后才能获得锁。

mutex使用场景: 解决协程并发时对同一资源的竞争问题。

6、sync.WaitGroup的使用场景?

程序中需要并发,需要创建多个goroutine,并且一定要等这些并发全部完成后才继续接下来的程序执行.WaitGroup的特点是Wait()可以用来阻塞直到队列中的所有任务都完成时才解除阻塞,而不需要sleep一个固定的时间来等待.但是其缺点是无法指定固定的goroutine数目(也就是协程池功能).

7、写一段闭包代码,阐述其作用

package main

import "fmt"

type logClosure func(format string, v...interface{})

func LoggerWrapper(logType string)logClosure{
	return func(format string, v...interface{}) {
		fmt.Printf(fmt.Sprintf("[%s] %s", logType, format), v...,)
		fmt.Println()  // 换行
	}
}

func main(){
	info_logger := LoggerWrapper("INFO")
	warning_logger := LoggerWrapper("WARNING")
	info_logger("this is a %s log", "info")
	warning_logger("this is a %s log", "warning")
}

作用:给访问外部函数定义的内部变量创造了条件,将关于的函数的一切封闭到了函数内部,减少了全局变量。

使用场景:

每次调用函数A时都要改变全局变量B,且B只与A相关,以往没有闭包时只能将B定义为全局变量;而现在可以将B定义为A的内部变量,将真正的执行函数作为闭包放在A内部去执行。

8、执行这段代码会发生什么?

type ConfigOne struct {
	Daemon string
}

func (c *ConfigOne) String() string {
	return fmt.Sprintf("print: %v", c)
}
c := &ConfigOne{}
_ = c.String()

答案:如果类型实现 String(),%v 和%+v 格式将使用 String() 的值, 这里会造成死递归

9、单例实现

var once sync.Once

type manager struct {name string}
var single *manager

func Singleton() *manager{
	once.Do(func() {
		single = &manager{"a"}
	})
	return single
}

注意一个once实例只能用一次。

10、这段代码输出什么?

fmt.Println(len(“你好bj!”)) 

答案:是9, len方法返回字符串的字节长度。

11、这段代码可以编译过吗,如果会错是在哪一行?

type Test struct {
	Name string
}
var list map[string]Test
func main() {
	list = make(map[string]Test)
	name := Test{"xiaoming"}
	list["name"] = name
	
	fmt.Println(list["name"].Name)
	list["name"].Name = "Hello"
	fmt.Println(list["name"])
}

不能!倒数第三行报错: cannot assign to struct field list[“name”].Name in map

因为map的value不是指针。首先,map本来存储的就是value的“初始指针值”,可以打印list[“name”].Name,  但不能通过取值的方式来修改。 因为当map扩容时,内部元素会在内存中移动, 移动之后list[“name”].Name获取到的值依然有效,但获取到的指针是无效的,如果允许这样赋值,那之后再打印list[“name”].Name 是获取不到修改后的值的。 而当value是指针时,也就是说list[“name”]是指针,list[“name”].Name就是指针内部的指针,值改变后,list[“name”]仍然获取到的是原始数据指针,也就仍然可以获取到list[“name”].Name。

12、ABCD哪一行会报错?

type S struct {
}

func f(x interface{}) {
}

func g(x *interface{}) {
}

func main() {
	s := S{}
	p := &s
	f(s) //A
	g(s) //B
	f(p) //C
	g(p) //D
}

B和D行。interface可以接受任意类型参数,包括指针。但是*interface{} 就只能接受*interface{}

13、下面的代码会怎样输出?每次输出结果一样吗?

const N = 10
var wg = &sync.WaitGroup{}
func main() {

	for i := 0; i < N; i++ {
		go func(i int) {
			wg.Add(1)
			println(i)
			defer wg.Done()
		}(i)
	}
	wg.Wait()
}

大部分时候可以输出0~9,但是顺序都不一样,有时只能输出部分数字。 因为wg.Add被包含在协程中,到达wg.Wait时,已经执行到wg.Add(1)的协程数量是无法确定的,这中间没有耗时也没有等待的操作。 虽然不会报错,但waitgroup不应该被这样使用!Wg.Add语句应该在协程之外。

 

文中的题目部分来自网络,由个人整理而成,如有错误请指出。

 

—————————————————————-  saosao的分割线  ————————————————————————

博主的Coding部落群 588757596,快来一起玩耍!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/148315.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Mac(OSX)下媲美XShell的神器Termius「建议收藏」

    Mac(OSX)下媲美XShell的神器Termius「建议收藏」文章目录简介特点软件环境配置配置项配置密钥配置说明配置主机配置项简介XShell的大名不用多说,称它为Windows平台最好用的远程终端不为过吧。唯一不足的地方就是它只有Windows版本。所以今天跟大家介绍一款全平台的远程终端——Termius。Termius不仅涵盖了Windows、Linux、OSX,还变态得支持Android和iOS(以后在地铁、公交上都可以随时拿出手机来排查线上问题啦…………

    2022年7月20日
    22
  • draggable的用法_draggable

    draggable的用法_draggable一、概述通过前面几节学习,大家应该都知道了,一个div对象是可以通过拖拉来改变大小,也可以通过拖动来改变其位置的。如何改变大小已经讲解过了,那么怎么实现拖动改变位置呢?现在就开始讲解如何实现拖动-

    2022年8月6日
    3
  • ireport 分页_sql组内分组

    ireport 分页_sql组内分组1、创建订单表et_order,并插入数据2.创建订单明细表et_order_detail,并插入数据3.不分组显示,将字段放入detail部分预览效果4.按照订单ID分组打印报表展示,点击模板名称,然后右键选择addreportgroup5.创建分组名称和分组字段6.分组包含了3部分,头部。明细。尾部,标题想要每张纸都显示,则需要放在pageheader块中7.最终效果…完美的达到了自己需要的效果。…

    2022年9月10日
    0
  • 物联网数据采集

    物联网数据采集大致方案为:硬件采集数据(包含采集协议和通讯协议)硬件与网络通讯(传输数据和传输方式)网络前端的显示和展示1、硬件采集数据我们现在用到的传感器大都是有固定通讯协议的,例如串口通讯(https://blog.csdn.net/qq_36629451/article/details/76038673),模拟量与数据量的直接读取(需要硬件设备留有相应的接口)…

    2022年5月15日
    39
  • tomcat能正常启动,但是不能访问http://localhost:8080

    tomcat能正常启动,但是不能访问http://localhost:8080

    2021年5月24日
    198
  • 微信小程序如何实现支付功能?看官方文档头疼(使用云函数的方式操作)「建议收藏」

    先来个效果图^_^微信支付功能,个人公众号是没有办法进行开发支付功能的,需要是使用非个人公众号进行注册(如:营业执照等,可以去淘宝购买一个也行大概500左右)公众平台的配置可以参考文档,这里主要是微信官网注册非个体公众号的否需代码操作。(也就是和我们码农相关的操作了)耐心看下面操作,基本上就是复制下面的代码(整个操作也就一会,基础再差也就30分钟搞定支付)^_^1.创建微信小程序2.在微信小程中序创建云函数1).根目录下创建一…

    2022年4月18日
    320

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号