当前位置: 首页 > news >正文

Go学习第7天:Map集合 + 递归函数 + 类型转换

Go 语言:Map集合、递归函数、类型转换

    • 目录
  • 一、Map 集合
    • 1.1 核心特性
    • 1.2 Map 创建方式
      • 方式1:make 函数创建(工程首选)
      • 方式2:字面量直接初始化
      • 方式3:仅声明(nil Map)
    • 1.3 Map 基础操作
      • 1. 增 / 改 元素
      • 2. 查询元素
      • 3. 获取长度
      • 4. 遍历 Map
      • 5. 删除元素
    • 1.4 Map 高频踩坑
  • 二、递归函数
    • 2.1 递归两大必要条件
    • 2.2 基础语法
    • 2.3 经典递归示例
      • 示例1:计算阶乘 n!
      • 示例2:斐波那契数列
      • 示例3:递归遍历文件目录
    • 2.4 递归优缺点 & 对比迭代
    • 2.5 递归高频踩坑
  • 三、类型转换
    • 3.1 基础数值类型转换
      • 语法
      • 示例
      • 规则 & 踩坑
    • 3.2 字符串与基本类型互转
      • 1. 字符串 ↔ 整型
      • 2. 字符串 ↔ 浮点型
      • 踩坑
    • 3.3 接口类型转换(类型断言)
      • 基础语法
      • 示例1:基础类型断言
      • 示例2:类型选择(多类型判断)
    • 3.4 类型转换通用踩坑
  • 四、知识点速记

延续前文学习体系,本文讲解Map 集合、递归函数、类型转换三大核心知识点,每部分包含语法说明、完整示例、核心规则、高频踩坑,代码可直接在 VSCode + Go Modules 环境运行。

目录

  1. Map 集合
  2. 递归函数
  3. 类型转换
  4. 综合练习 & 知识点速记

一、Map 集合

Map 是 Go 语言无序键值对(key-value)集合,属于引用类型,底层采用哈希表实现。通过key快速查询、修改、删除对应value,是开发高频容器。

1.1 核心特性

  1. 无序性:遍历 Map 时,键值对输出顺序不固定,和写入顺序无关;
  2. 键唯一性:同一个 Map 中key不可重复,重复赋值会覆盖原有 value;
  3. 零值规则:查询不存在的 key,返回 value 对应类型的零值;
  4. 引用类型:Map 赋值、函数传参时,多个变量指向同一底层数据,修改互相影响;
  5. 键类型限制key必须是可比较类型(整型、字符串、布尔、数组等),切片、Map、函数不可作为 key

1.2 Map 创建方式

方式1:make 函数创建(工程首选)

语法:

make(map[键类型]值类型[初始容量])
  • 初始容量为可选参数,达到容量后 Map 自动扩容;
  • 仅声明未使用make初始化的 Map 为nil无法直接增删元素

示例:

packagemainimport"fmt"funcmain(){// 无初始容量m1:=make(map[string]int)// 指定初始容量 10m2:=make(map[string]int,10)m1["苹果"]=1fmt.Println(m1)}

方式2:字面量直接初始化

适合已知初始键值对的场景:

packagemainimport"fmt"funcmain(){// 初始化并赋值m:=map[string]int{"apple":1,"banana":2,"orange":3,}fmt.Println(m)}

方式3:仅声明(nil Map)

varmmap[string]string// 默认为 nil,不能直接存数据// m["test"] = "123" // 运行报错

1.3 Map 基础操作

1. 增 / 改 元素

直接通过map[key] = value赋值:

  • key 不存在 → 新增键值对;
  • key 已存在 → 覆盖原有 value。
siteMap:=make(map[string]string)siteMap["Runoob"]="菜鸟教程"// 新增siteMap["Runoob"]="新名称"// 修改覆盖

2. 查询元素

两种写法:

  1. 单返回值:直接取值,key 不存在返回零值,无法判断 key 是否存在
  2. 双返回值:value, ok := map[key]ok为布尔值,true表示 key 存在。
packagemainimport"fmt"funcmain(){m:=map[string]int{"a":10}// 写法1:单值获取v1:=m["a"]fmt.Println(v1)// 写法2:判断键是否存在(推荐)v2,ok:=m["b"]ifok{fmt.Println("存在,值:",v2)}else{fmt.Println("键不存在")}}

3. 获取长度

使用len(map)获取当前键值对总数:

m:=map[string]int{"a":1,"b":2}fmt.Println(len(m))// 输出 2

4. 遍历 Map

搭配for-range遍历,遍历顺序随机:

packagemainimport"fmt"funcmain(){m:=map[string]string{"France":"巴黎","Japan":"东京",}// 遍历 key + valuefork,v:=rangem{fmt.Printf("%s => %s\n",k,v)}// 只遍历 keyfork:=rangem{fmt.Println(k)}}

5. 删除元素

使用内置函数delete(map, key)key 不存在不会报错

packagemainimport"fmt"funcmain(){m:=map[string]string{"a":"111","b":"222"}delete(m,"a")// 删除键 afmt.Println(m)delete(m,"c")// 键不存在,无任何异常}

1.4 Map 高频踩坑

  1. nil Map 直接赋值var m map[k]v声明后未 make,直接m[k]=v运行 panic;
  2. 混淆“零值”和“真实值”:仅单值查询时,零值无法区分 key 不存在 & value 本身就是零值,必须用ok判断;
  3. key 类型非法:切片、Map、函数不能作为 Map 的键;
  4. 遍历中增删元素:遍历 Map 时修改/删除元素,会导致遍历结果异常;
  5. 引用传递特性:Map 传参/赋值后,多个变量共享底层数据,一处修改全局生效;
  6. Map 不支持下标遍历:不能使用for i := 0; i<len(m);i++方式遍历。

二、递归函数

递归是函数直接或间接调用自身的编程方式,适合拆解为重复子问题的场景(阶乘、数列、目录遍历等)。

2.1 递归两大必要条件

  1. 基准条件(终止条件):递归停止的出口,必须设置,否则无限递归、栈溢出;
  2. 递归体:函数调用自身,将大问题拆解为同类型小问题。

2.2 基础语法

func函数名(参数)返回值{// 1. 基准条件:终止递归if终止判断{return结果}// 2. 递归体:调用自身return函数名(新参数)}

2.3 经典递归示例

示例1:计算阶乘 n!

规则:0! = 1n! = n * (n-1)!

packagemainimport"fmt"// 递归计算阶乘funcfactorial(nint)int{// 基准条件ifn==0{return1}// 递归调用自身returnn*factorial(n-1)}funcmain(){fmt.Println(factorial(5))// 输出 120}

示例2:斐波那契数列

规则:f(0)=0,f(1)=1,f(n)=f(n-2)+f(n-1)

packagemainimport"fmt"funcfibonacci(nint)int{ifn<2{returnn}returnfibonacci(n-2)+fibonacci(n-1)}funcmain(){fori:=0;i<10;i++{fmt.Printf("%d ",fibonacci(i))}// 输出:0 1 1 2 3 5 8 13 21 34}

示例3:递归遍历文件目录

packagemainimport("fmt""os""path/filepath")// 递归遍历目录funcwalkDir(dirstring,indentstring){entries,err:=os.ReadDir(dir)iferr!=nil{return}for_,entry:=rangeentries{fmt.Println(indent+entry.Name())// 如果是文件夹,递归进入子目录ifentry.IsDir(){walkDir(filepath.Join(dir,entry.Name()),indent+" ")}}}funcmain(){walkDir(".","")}

2.4 递归优缺点 & 对比迭代

对比项递归迭代(循环)
代码简洁、逻辑贴近问题本身代码偏冗长
性能函数调用开销大,深度递归易栈溢出执行效率高,内存占用小
调试调用链路深,调试困难流程直观,易调试

适用场景

  • 优先递归:树、图遍历、分治算法、目录遍历;
  • 优先迭代:大数据、深度循环、性能敏感场景。

2.5 递归高频踩坑

  1. 缺少终止条件:无限递归,触发goroutine stack exceeds栈溢出崩溃;
  2. 终止条件错误:递归无法正常退出,结果异常;
  3. 递归深度过大:Go 栈空间有限,深度递归直接栈溢出;
  4. 重复计算:如斐波那契递归写法,存在大量重复运算,性能极差。

三、类型转换

Go 是强类型语言,不支持隐式类型转换,不同类型之间赋值、运算必须手动显式转换。分为:基础数值转换、字符串互转、接口类型断言/转换三大类。

3.1 基础数值类型转换

语法

目标类型(表达式/变量)

示例

packagemainimport"fmt"funcmain(){varaint=17varbint=5// 整型转浮点型后再运算,避免整数除法res:=float32(a)/float32(b)fmt.Println(res)// 3.4}

规则 & 踩坑

  1. 不同位数整型(int/int32/int64)必须手动转换,直接赋值编译报错;
  2. 大范围转小范围会截断数据(精度丢失);
  3. 浮点转整型会直接舍弃小数部分,不四舍五入。

错误示例:

varaint64=10varbint32// b = a 编译报错,不支持隐式转换b=int32(a)// 正确显式转换

3.2 字符串与基本类型互转

依赖标准库strconv包,不能直接用基础类型转换语法。

1. 字符串 ↔ 整型

  • strconv.Atoi(s):字符串转 int,返回(数值, 错误)
  • strconv.Itoa(n):int 转字符串。
packagemainimport("fmt""strconv")funcmain(){// 字符串转整型str1:="123"num,err:=strconv.Atoi(str1)iferr==nil{fmt.Println(num)}// 整型转字符串num2:=456str2:=strconv.Itoa(num2)fmt.Println(str2)}

2. 字符串 ↔ 浮点型

  • strconv.ParseFloat(字符串, 精度):字符串转浮点;
  • strconv.FormatFloat(浮点数, 格式, 小数位数, 精度):浮点转字符串。
packagemainimport("fmt""strconv")funcmain(){// 字符串转 float64s:="3.14"f,_:=strconv.ParseFloat(s,64)fmt.Println(f)// float64 转字符串,保留2位小数f2:=3.1415s2:=strconv.FormatFloat(f2,'f',2,64)fmt.Println(s2)}

踩坑

  1. 非数字字符串调用Atoi/ParseFloat,会返回转换错误;
  2. 不能直接用string(数字)做数字转字符串(会转为 ASCII 字符,不是数字文本)。

3.3 接口类型转换(类型断言)

空接口interface{}可接收任意类型值,使用类型断言还原原始类型。

基础语法

// 写法1:判断类型并取值,布尔标识:=接口变量.(目标类型)// 写法2:类型选择(switch 批量判断类型)switch变量:=接口变量.(type){case类型1:// 对应逻辑case类型2:// 对应逻辑default:}

示例1:基础类型断言

packagemainimport"fmt"funcmain(){variinterface{}="Go语言"// 断言为字符串类型str,ok:=i.(string)ifok{fmt.Println("转换成功:",str)}else{fmt.Println("类型不匹配")}}

示例2:类型选择(多类型判断)

packagemainimport"fmt"funcprintVal(vinterface{}){switchv:=v.(type){caseint:fmt.Println("整型:",v)casestring:fmt.Println("字符串:",v)casefloat64:fmt.Println("浮点型:",v)default:fmt.Println("未知类型")}}funcmain(){printVal(100)printVal("test")print(3.14)}

3.4 类型转换通用踩坑

  1. 禁止隐式转换:所有不同类型必须手动转换,编译强制校验;
  2. 数值精度丢失:大类型转小类型、浮点转整型会丢失数据;
  3. 字符串数字转换:非数字文本转换必然报错,必须捕获 error;
  4. 类型断言失败:不使用ok直接断言,类型不匹配会触发运行 panic;
  5. 区分转换语法:数值用类型(),字符串/数字互转用strconv包,接口用类型断言。

四、知识点速记

模块核心要点高频坑点
Map无序键值对、引用类型;make初始化;delete删除;range遍历nil Map直接赋值;仅用单值判断键是否存在;切片/Map做key
递归必须有终止条件;函数调用自身;适合分治/遍历无终止条件导致栈溢出;深度递归性能差
类型转换强类型、无隐式转换;数值直接转;字符串用strconv;接口用类型断言类型不匹配直接赋值;断言不判断ok;非法字符串转数字
http://www.cnnetsun.cn/news/2913336.html

相关文章:

  • 保姆级教程:用C语言和gSOAP从零实现一个ONVIF客户端(附完整源码)
  • 别被型号搞晕了!一文看懂高通IPQ9574/9554/9514 Wi-Fi 7芯片怎么选(附路由器型号对照表)
  • 连续流语言模型原理与高效文本生成实践
  • OpenCvSharp的Mat、System.Drawing的Bitmap和Image,到底该用哪个?一篇讲清区别与选用
  • 深度对比:Stellar文件修复工具包 vs. 手动修复,拯救损坏Office文档哪种更靠谱?
  • 从“分流器”到“电流检测电阻”:这个小元件的前世今生与选型实战
  • STM32玩转Nuttx:除了Makefile,你还需要搞定这些烧录工具链(OpenOCD/stm32flash详解)
  • 从WMS到瓦片服务:聊聊Web地图加载性能优化的‘前世今生’与选型建议
  • 2026录音转文字怎么做?免费工具手把手保姆级教程
  • 别再傻傻分不清!一文搞懂SDR(软件定义雷达)和SR(软件化雷达)的核心区别
  • RS485 HUB、中继器、分线器到底有啥区别?看完这篇别再买错了
  • 高通学习4-高通AR1平台(TODO)
  • yolov26改进 | Neck/颈部改进篇 | CVPR最新低照度图像增强模块HVI改进YOLOv26(有效涨点)
  • TO-39封装红外测温传感器怎么选?深度对比MLX90614与国产GD60914系列(含5° FOV进灰问题解决)
  • 不止于Vue:用200字节的mitt库,搞定React/原生JS项目中的事件管理
  • 从广播到对讲机:拆解生活中FM与PM调制的真实应用场景与硬件选型
  • 3毛钱的国产RS485芯片,真能省掉TVS和偏置电阻?实测CS48505S在工业板卡上的表现
  • 2026年论文党必备:盘点2026年标杆级的AI论文平台
  • PyQt5界面代码维护指南:.ui文件 vs 纯Python代码,哪种方式更适合你的项目?
  • 5个常见问题解决指南:Windows版Mesa3D图形驱动安装与故障排除
  • 从PyTorch转Rust?tch-rs、Candle、Burn、DFDX四大框架实战对比与选型指南
  • 终极指南:如何免费激活Adobe全家桶软件(2019-2023全版本)
  • PY32F002A vs PY32F003 vs PY32F030:手把手教你根据项目需求选对普冉M0+ MCU
  • AList项目易主后,我的私人云存储方案还安全吗?聊聊替代方案与数据安全实践
  • 工资信息管理系统毕业设计源码
  • 告别充电焦虑:一文看懂CCS、CHAdeMO和国标GB/T的充电枪与协议区别(2024版)
  • 校园健康驿站管理系统毕业设计
  • Java SpringBoot+Vue3+MyBatis WEB旅游推荐系统系统源码|前后端分离+MySQL数据库
  • Unlock-Music终极指南:3步解锁加密音乐,让音乐自由播放
  • AWQ vs GPTQ vs BitsAndBytes:给LLM‘瘦身’,选哪个?一张表讲清楚差异和选型