深信服golang面经

在这里插入图片描述

for range 中赋值的变量,这个变量指向的是真实的地址吗,还是临时变量

不是真实地址,是临时变量

package mainimport "fmt"func main() {slice := []int{4, 2, 3}for _, v := range slice {fmt.Println(v, &v) // 这里的 v 是临时变量}for i := range slice{fmt.Println(i, &i)fmt.Println(slice[i], &slice[i])}fmt.Println("sjhdjaskhdjkashdkas")for _, v := range slice {fmt.Println(v, &v) // 这里的 v 是临时变量}for i := range slice{fmt.Println(i, &i)fmt.Println(slice[i], &slice[i])}
}

如果在for range里面有一个函数,这个函数需要传一个指针,这时候应该怎么写,这时候会进行拷贝吗

slice内容未改变, 换成另一个方法才会改变
for _,v := range x
是值拷贝

package mainimport "fmt"func main() {slice := []int{4, 2, 3}for _,v := range slice{fmt.Println(v, &v)doDubole(&v)fmt.Println(v, &v)}fmt.Println(slice)
}func doDubole(x *int){fmt.Println(*x, x)*x = *x*2
}

有用过go link?那么在什么情况下如果我不赋给一个新的变量,它也是没问题的?

在 Go 语言中,link 通常指的是链接操作,主要用于将多个包和可执行文件组合成一个单一的可执行文件。

如果我要在defer里面修改return里面的值呢?这时怎么写?

命名返回值
返回110

package mainimport "fmt"func main() {ans := get(10)fmt.Println(ans)
}
func get(x int)(ans int){defer func(){ans += 10}()ans = x*10return
}

map时协程安全吗?有什么是协程安全的?

在 Go 中,map 不是协程安全的。如果多个协程同时读写同一个 map,可能会导致数据竞争和不确定的行为。因此,使用 map 时需要确保并发安全。

协程安全的解决方案:

  1. 使用 sync.Mutexsync.RWMutex

    • 通过互斥锁确保对 map 的访问是安全的。
    var mu sync.Mutex
    m := make(map[string]int)func safeWrite(key string, value int) {mu.Lock()m[key] = valuemu.Unlock()
    }func safeRead(key string) int {mu.Lock()value := m[key]mu.Unlock()return value
    }
    
  2. 使用 sync.Map

    • Go 提供了一个内置的并发安全的 map 类型 sync.Map,适合并发读写。
    var m sync.Map// 存储值
    m.Store("key", 42)// 读取值
    value, ok := m.Load("key")
    
  3. 使用通道 (Channels)

    • 通过通道来同步对数据的访问,而不是直接操作 map
    type request struct {key   stringvalue intreply chan int
    }func mapHandler(requests <-chan request) {m := make(map[string]int)for req := range requests {m[req.key] = req.valuereq.reply <- m[req.key]}
    }
    

总结:

  • map 在并发环境下不安全。
  • 使用互斥锁、sync.Map 或通道可以实现协程安全。

channel有缓冲区和无缓冲区的区别?

无缓冲:必须有人接收才能发送,需要保证顺序性
在这里插入图片描述
以下是有缓冲区和无缓冲区通道的实际应用场景示例。

1. 有缓冲区的通道:任务队列

在一个工作池中,多个工作者从一个任务队列中获取任务。使用有缓冲区的通道可以存储一定数量的任务,提高效率。

package mainimport ("fmt""time"
)func worker(id int, tasks <-chan int) {for task := range tasks {fmt.Printf("工作者 %d 处理任务 %d\n", id, task)time.Sleep(time.Second) // 模拟任务处理时间}
}func main() {tasks := make(chan int, 5) // 创建一个缓冲区为 5 的通道// 启动多个工作者for i := 1; i <= 3; i++ {go worker(i, tasks)}// 发送任务for i := 1; i <= 10; i++ {fmt.Printf("发送任务 %d\n", i)tasks <- itime.Sleep(500 * time.Millisecond) // 模拟任务生成时间}close(tasks) // 关闭通道time.Sleep(5 * time.Second) // 等待所有任务完成
}

2. 无缓冲区的通道:同步信号

在两个 goroutine 之间使用无缓冲区通道进行同步,确保一个 goroutine 在另一个完成后再继续执行。

package mainimport ("fmt"
)func main() {done := make(chan struct{}) // 创建一个无缓冲区的信号通道go func() {fmt.Println("工作者正在处理任务...")// 模拟任务处理// ...done <- struct{}{} // 发送完成信号}()// 等待工作者完成<-donefmt.Println("任务完成,主程序继续执行。")
}

总结

  • 有缓冲区的通道适用于任务队列等场景,可以提高并发性能。
  • 无缓冲区的通道适用于需要同步的场景,确保操作的顺序。

我能不能在写入channel的时候,先判断它阻不阻塞再写?

在 Go 中,无法直接检查一个通道是否会阻塞。通道的发送和接收都是阻塞操作,具体行为取决于通道的状态和接收方是否准备好。

解决方案

如果你想在写入通道之前判断是否会阻塞,可以使用非阻塞发送配合 select 语句。例如:

package mainimport ("fmt"
)func main() {ch := make(chan int, 1) // 创建一个缓冲通道,容量为 1// 先尝试发送select {case ch <- 42: // 非阻塞发送fmt.Println("写入成功")default:fmt.Println("通道已满,写入阻塞")}// 再次尝试发送select {case ch <- 99:fmt.Println("写入成功")default:fmt.Println("通道已满,写入阻塞")}// 从通道读取值value := <-chfmt.Println("读取值:", value)
}

关键点:

  1. select 语句:允许你进行非阻塞操作。
  2. default 分支:在通道不能发送或接收时执行,避免阻塞。

通过这种方式,你可以在写入通道之前判断通道的状态,从而避免阻塞。

mutex是个悲观锁还是乐观锁,乐观锁和悲观锁的区别?

Mutex 是悲观锁,适用于高冲突的场景。
乐观锁 适用于低冲突的场景,减少了加锁带来的性能开销。

需要知道一个goroutine里面的一个函数是否执行成功还是失败,应该怎么写

要在一个 goroutine 中检查一个函数是否执行成功,可以使用通道(channel)来传递结果或错误信息。以下是一个示例,演示如何实现这一点:

示例代码

package mainimport ("errors""fmt""time"
)func doWork(resultChan chan<- string, errChan chan<- error) {// 模拟一些工作time.Sleep(2 * time.Second)// 假设工作成功,返回结果// resultChan <- "工作成功"// 假设工作失败,返回错误errChan <- errors.New("工作失败")
}func main() {resultChan := make(chan string)errChan := make(chan error)go doWork(resultChan, errChan)select {case result := <-resultChan:fmt.Println("结果:", result)case err := <-errChan:fmt.Println("错误:", err)case <-time.After(3 * time.Second): // 设置超时fmt.Println("操作超时")}
}

关键点

  1. 通道

    • 使用 resultChan 来传递成功结果。
    • 使用 errChan 来传递错误信息。
  2. select 语句

    • 通过 select 等待通道中的结果或错误。
    • 如果有结果返回,则打印结果;如果有错误返回,则打印错误。
  3. 超时处理

    • 可以使用 time.After 设置超时处理,避免等待太久。

通过这种方式,你可以有效地在 goroutine 中检查函数的执行状态。

了解过Go的内存逃逸吗?

是的,Go 的内存逃逸(Escape Analysis)是一个重要的功能,用于优化内存分配和管理。

什么是内存逃逸?

内存逃逸指的是在 Go 中某个变量的生命周期超出了其原本的作用域。这意味着该变量不能在栈上分配,而必须在堆上分配,以确保在函数返回后仍然能够访问。

如何工作?

  1. 栈 vs 堆

    • :用于存储局部变量,分配和释放速度快,但生命周期受限于函数调用。
    • :用于动态分配内存,生命周期更长,但分配和释放速度较慢。
  2. 逃逸分析

    • Go 编译器会在编译时分析变量的使用情况,判断它是否会“逃逸”到堆上。
    • 如果变量的地址被返回,或在函数外部使用,它将被分配到堆上。

示例

以下是一个简单的例子,展示了内存逃逸的情况:

package mainimport "fmt"type Person struct {Name string
}func NewPerson(name string) *Person {return &Person{Name: name} // 逃逸到堆
}func main() {p := NewPerson("Alice")fmt.Println(p.Name)
}

在上面的代码中,NewPerson 函数返回一个指向 Person 结构体的指针,这导致 Person 实例被分配到堆上,因为它的生命周期超出了函数的作用域。

逃逸分析的好处

  • 性能优化:通过避免不必要的堆分配,提升性能。
  • 内存管理:减少内存泄漏的风险,帮助开发者更好地管理内存。

如何检查逃逸分析

可以使用 go build -gcflags -m 命令来查看编译器的逃逸分析信息:

go build -gcflags -m yourfile.go

这将显示哪些变量逃逸到堆上,以及相关的分析信息。

总结

内存逃逸分析是 Go 的一项重要特性,帮助编译器优化内存分配,提高程序性能。理解内存逃逸有助于编写更高效的 Go 代码。

场景:1GB文件,每个单词不超过16字节,在1M的内存里,得到出现频率最高的100个单词

由于文件大小为1GB,而内存的大小只有1MB,因此不能一次把所有的词读入到内存中去处理,可以采用分治的方法进行处理:把一个文件分解为多个小的子文件,从而保证每个文件的大小都小于1MB,进而可以直接被读取到内存中处理。

第一步:使用多路归并排序对大文件进行排序,这样相同的单词肯定是挨着的

第二步:
① 初始化一个 100 个节点的小顶堆,用于保存 100 个出现频率最多的单词
② 遍历整个文件,一个单词一个单词的从文件中取出来,并计数
③ 等到遍历的单词和上一个单词不同的话,那么上一个单词及其频率如果大于堆顶的词的频率,那么放在堆中,否则不放

最终,小顶堆中就是出现频率前 100 的单词了

多路归并排序对大文件进行排序的步骤如下:
① 将文件按照顺序切分成大小不超过 512KB 的小文件,总共 2048 个小文件
② 使用 1MB 内存分别对 2048 个小文件中的单词进行排序
③ 使用一个大小为 2048 大小的堆,对 2048 个小文件进行多路排序,结果写到一个大文件中

json和protobuf的区别

JSON(JavaScript Object Notation)和 Protobuf(Protocol Buffers)是两种常用的数据序列化格式。它们各自有不同的特点和适用场景。以下是它们之间的主要区别:

1. 数据格式

  • JSON

    • 人类可读的文本格式,易于调试和查看。
    • 数据使用键值对的结构表示,支持基本数据类型(字符串、数字、布尔值、数组和对象)。
  • Protobuf

    • 二进制格式,不易于人类直接阅读。
    • 需要预先定义数据结构(消息格式)并编译生成代码来处理数据。

2. 性能

  • JSON

    • 解析和序列化速度较慢,因为它是文本格式。
    • 数据体积相对较大,尤其是在传输大量数据时。
  • Protobuf

    • 解析和序列化速度快,效率高。
    • 数据体积较小,适合网络传输和存储。

3. 可扩展性

  • JSON

    • 结构灵活,不需要预定义模式,可以随意添加字段,但对版本控制支持较差。
  • Protobuf

    • 需要预定义结构,但支持向后兼容和向前兼容,可以安全地添加或删除字段。

4. 语言支持

  • JSON

    • 几乎所有编程语言都支持 JSON 解析和生成。
  • Protobuf

    • 支持多种编程语言,但需要使用特定的工具生成代码。

5. 使用场景

  • JSON

    • 适用于配置文件、Web API(如 RESTful API)、轻量级数据传输等场景。
  • Protobuf

    • 适用于高性能网络通信、大规模数据存储、微服务架构等场景。

总结

  • JSON:易读、灵活,但性能较低,适合简单的应用场景。
  • Protobuf:高效、可扩展,适合对性能和数据体积有较高要求的应用场景。

Go怎么调试的,会Goland远程调试吗?

  1. 安装dlv
    在Linux服务器上执行:go install github.com/go-delve/delve/cmd/dlv@latest,安装dlv调试工具,因为是go编译的可执行程序,可以随意复制,其他环境甚至都可以不安装go语言环境。

  2. 按照goland提示添加远程调试

  3. 添加编译配置

  4. 在服务器运行
    将可执行程序上传到服务器,并使用dlv运行:
    dlv --listen=:2345 --headless=true --api-version=2 --accept-multiclient exec ./test001_linux

带命令行参数,在可执行程序后面带上 --,再后面就是命令行参数:

dlv --listen=:2345 --headless=true --api-version=2 --accept-multiclient exec ./test001_linux – -s 123

  1. 然后再window的goland上运行调试

算法:查找字符串子串,有哪些算法?

暴力匹配:简单,但效率低。
KMP:高效,适合单个子串匹配。O(m+n)
部分匹配表(LPS 数组):

在开始匹配之前,KMP 算法首先构建一个部分匹配表(Longest Prefix Suffix,LPS)。
LPS 数组的每个元素 lps[i] 表示子串 pattern[0…i] 中的最长相等前后缀的长度。
LPS 数组可以帮助我们在匹配失败时,不必回溯主串的指针,而是根据已匹配的部分直接跳到下一个可能匹配的位置。
匹配过程:

使用两个指针,i 指向主串,j 指向子串。
逐个比较主串和子串的字符:
如果字符匹配,两个指针同时向后移动。
如果不匹配且 j > 0,则根据 LPS 数组调整子串指针 j,而不移动主串指针 i。
如果 j 达到子串的长度,说明找到匹配,记录匹配位置。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/81644.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PLC双人舞:profinet转ethernet ip网关奏响施耐德与AB的协奏曲

PLC双人舞&#xff1a;ethernet ip转profinet网关奏响施耐德与AB的协奏曲 案例分析&#xff1a;施耐德PLC与AB PLC的互联互通 在现代工业自动化中&#xff0c;设备之间的互联互通至关重要。本案例旨在展示如何通过北京倍讯科技的EtherNet/IP转Modbus网关&#xff0c;将施耐德P…

链接家里电脑

要在外网访问家里的电脑&#xff08;或NAS&#xff09;&#xff0c;主要有 5种主流方法&#xff0c;各有优缺点&#xff0c;适用于不同需求。以下是详细方案和操作指南&#xff1a; 一、方案对比速查表 方法适用场景速度安全性难度是否需要公网IP远程桌面&#xff08;RDP&…

VS Code开源AI编辑器:一场编程革命的新起点

在2025年5月19日&#xff0c;微软发布了一则激动人心的消息——VS Code将开源其AI编辑器组件&#xff0c;特别是GitHub Copilot Chat扩展。正如微软官方博客所宣告的&#xff1a;“我们相信代码编辑器的未来应该是开放的&#xff0c;并由AI驱动。” 为什么现在开源&#xff1f…

51c嵌入式※~合集7~Linux

我自己的原文哦~ https://blog.51cto.com/whaosoft/13926843 一、u-boot和bootloader~区别 Bootloader 比Bootloader从字面上来看就是启动加载的意思。用过电脑的都知道&#xff0c;windows开机时会首先加载bios&#xff0c;然后是系统内核&#xff0c;最后启动完毕。那…

深度学习实战 04:卷积神经网络之 VGG16 复现三(训练)

在后续的系列文章中&#xff0c;我们将逐步深入探讨 VGG16 相关的核心内容&#xff0c;具体涵盖以下几个方面&#xff1a; 卷积原理篇&#xff1a;详细剖析 VGG 的 “堆叠小卷积核” 设计理念&#xff0c;深入解读为何 332 卷积操作等效于 55 卷积&#xff0c;以及 333 卷积操作…

Ubuntu 20.04之Docker安装ES7.17.14和Kibana7.17.14

你需要已经安装如下运行环境: Ubuntu 20.04 docker 28 docker-compose 1.25 一、手动拉取镜像 docker pull docker.elastic.co/kibana/kibana:7.17.14docker pull docker.elastic.co/elasticsearch/elasticsearch:7.17.14 或者手动导入镜像 docker load -i es7.17.14.ta…

实时技术方案对比:SSE vs WebSocket vs Long Polling

早期网站仅展示静态内容,而如今我们更期望:实时更新、即时聊天、通知推送和动态仪表盘。 那么要如何实现实时的用户体验呢?三大经典技术各显神通: SSE(Server-Sent Events):轻量级单向数据流WebSocket:双向全双工通信Long Polling(长轮询):传统过渡方案假设目前有三…

测试开发面试题:Python高级特性通俗讲解与实战解析

前言&#xff1a;为什么测试工程师必须掌握Python高级特性&#xff1f; 通俗比喻&#xff1a; 基础语法就像“锤子”&#xff0c;能敲钉子&#xff1b;高级特性就像“瑞士军刀”&#xff0c;能应对复杂场景&#xff08;如自动化框架、高并发测试&#xff09;。面试官考察点&a…

C语言-9.指针

9.1指针 9.1-1取地址运算:&运算符取得变量的地址 运算符& scanf(“%d”,&i);里的&获取变量的地址,它们操作数必须是变量int i;printf(“%x”,&i);地址的大小是否与int相同取决于编译器int i;printf(“%p”,&i); &不能取的地址不能对没有地址的…

【C++】Vcpkg 介绍及其常见命令

Vcpkg 简介 Vcpkg 是微软开发的一个跨平台的 C/C 依赖管理工具&#xff0c;用于简化第三方库的获取、构建和管理过程。 主要特点 跨平台支持&#xff1a;支持 Windows、Linux 和 macOS开源免费&#xff1a;MIT 许可证大型库集合&#xff1a;包含超过 2000 个开源库简化集成&…

Unity3D 动画文件优化总结

前言 在Unity3D中&#xff0c;动画文件的压缩和优化是提升性能的重要环节&#xff0c;尤其在移动端或复杂场景中。以下是针对Animation Clip和Animator Controller的优化方法总结&#xff1a; 对惹&#xff0c;这里有一个游戏开发交流小组&#xff0c;希望大家可以点击进来一…

前端工程的相关管理 git、branch、build

环境配置 标准环境打包 测试版&#xff1a;npm run build-test 预生产&#xff1a;npm run build-preview 正式版&#xff1a;npm run build 建议本地建里一个 .env.development.local 方便和后端联调时修改配置相关信息。 和 src 同级有一下区分环境的文件&#xff1a; .env.d…

VAPO:视觉-语言对齐预训练(对象级语义)详解

简介 多模态预训练模型(Vision-Language Pre-training, VLP)近年来取得了飞跃发展。在视觉-语言模型中,模型需要同时理解图像和文本,这要求模型学习二者之间的语义对应关系。早期方法如 VisualBERT、LXMERT 等往往使用预先提取的图像区域特征和文本词嵌入拼接输入,通过 T…

docker运行Redis

创建目录 mkdir -p /home/jie/docker/redis/{conf,data,logs}添加权限 chmod -R 777 /home/jie/docker/redis创建配置文件 cat > /home/jie/docker/redis/conf/redis.conf << EOF # 基本配置 bind 0.0.0.0 protected-mode yes port 6379# 安全配置 密码是root require…

初识 java

目录 前言 一、jdk&#xff0c;JRE和JVM之间的关系 二、JVM的内存划分 前言 初步了解 jdk&#xff0c;JRE&#xff0c;JVM 之间的关系&#xff0c;JVM 的内存划分。 一、jdk&#xff0c;JRE和JVM之间的关系 jdk 是 java 开发工具集&#xff0c;包含JRE&#xff1b; JRE 是…

关于百度地图JSAPI自定义标注的图标显示不完整的问题(其实只是因为图片尺寸问题)

下载了几个阿里矢量图标库里的图标作为百度地图的自定义图标&#xff0c;结果百度地图显示的图标一直不完整。下载的PNG图标已经被正常引入到前端代码&#xff0c;anchor也设置为了图标底部中心&#xff0c;结果还是显示不完整。 if (iconUrl) {const icon new mapClass.Icon(…

系统安全及应用深度笔记

系统安全及应用深度笔记 一、账号安全控制体系构建 &#xff08;一&#xff09;账户全生命周期管理 1. 冗余账户精细化治理 非登录账户基线核查 Linux 系统默认创建的非登录账户&#xff08;如bin、daemon、mail&#xff09;承担系统服务支撑功能&#xff0c;其登录 Shell 必…

02-前端Web开发(JS+Vue+Ajax)

介绍 在前面的课程中&#xff0c;我们已经学习了HTML、CSS的基础内容&#xff0c;我们知道HTML负责网页的结构&#xff0c;而CSS负责的是网页的表现。 而要想让网页具备一定的交互效果&#xff0c;具有一定的动作行为&#xff0c;还得通过JavaScript来实现。那今天,我们就来讲…

AXXI4总线协议 ------ AXI_FULL协议

https://download.csdn.net/download/mvpkuku/90855619 一、AXI_FULL协议的前提知识 1. 各端口的功能 2. 4K边界问题 3. outstanding 4.时序仿真体验 可通过VIVADO自带ADMA工程观察仿真波形图 二、FPGA实现 &#xff08;主要用于读写DDR&#xff09; 1.功能模块及框架 将…

React系列——nvm、node、npm、yarn(MAC)

nvm&#xff0c;node&#xff0c;npm之间的区别 1、nvm&#xff1a;nodejs版本管理工具。nvm 可以管理很多 node 版本和 npm 版本。 2、nodejs&#xff1a;在项目开发时的所需要的代码库 3、npm&#xff1a;nodejs包管理工具。nvm、nodejs、npm的关系 nvm 管理 nodejs 和 npm…