等级头衔
等級: 晓枫资讯-列兵
在线时间 0 小时
积分成就
威望 0 点
贡献 52 次
主题 42 次
精华 0 篇
金钱 147 枚
积分 96 枚
、
注册时间 2023-10-4
最后登录 2025-9-8
make 函数的基本概念
语法与适用类型
函数的语法格式为
,其中
代表要创建的类型,必须是切片、映射或通道这三种引用类型之一,
是根据不同类型而定的参数。以下是三种类型使用
函数的基本形式:
切片 :make([]T, length, capacity) 复制代码 ,其中是切片元素的类型,是切片的初始长度,是切片的初始容量(可省略,默认与长度相同)。
映射 :make(map[K]V, initialCapacity) 复制代码 ,是键的类型,是值的类型,是映射的初始容量(可省略)。
通道 :,是通道中元素的类型,是通道的缓冲区大小(可省略,省略时为无缓冲通道)。
代码示例
package main
import "fmt"
func main() {
// 使用 make 创建切片
slice := make([]int, 3, 5)
fmt.Printf("切片长度: %d, 容量: %d, 内容: %v\n", len(slice), cap(slice), slice)
// 使用 make 创建映射
m := make(map[string]int)
m["apple"] = 1
m["banana"] = 2
fmt.Println("映射内容:", m)
// 使用 make 创建通道
ch := make(chan int, 2)
ch <- 10
ch <- 20
fmt.Println("从通道接收:", <-ch)
} 复制代码 在上述代码中,分别使用
函数创建了切片、映射和通道,并进行了简单的操作。
切片的 make 分配
长度与容量的区别
在使用
创建切片时,长度和容量是两个重要的概念。长度表示切片中当前元素的数量,而容量表示切片底层数组的大小。可以通过
函数获取切片的长度,通过
函数获取切片的容量。
package main
import "fmt"
func main() {
// 创建一个长度为 2,容量为 5 的切片
slice := make([]int, 2, 5)
fmt.Printf("初始长度: %d, 初始容量: %d\n", len(slice), cap(slice))
// 向切片追加元素
slice = append(slice, 1, 2, 3)
fmt.Printf("追加元素后长度: %d, 容量: %d, 内容: %v\n", len(slice), cap(slice), slice)
} 复制代码 在这个示例中,初始创建的切片长度为 2,容量为 5。当使用
函数追加元素时,如果长度超过了容量,Go 语言会自动重新分配更大的底层数组。
项目场景:数据处理
在数据处理项目中,我们可能需要动态地处理一批数据。使用
创建切片可以预先分配一定的容量,减少内存重新分配的次数,提高性能。
package main
import (
"fmt"
)
func processData() []int {
// 预先分配容量为 100 的切片
data := make([]int, 0, 100)
for i := 0; i < 100; i++ {
data = append(data, i)
}
return data
}
func main() {
result := processData()
fmt.Println("处理后的数据:", result)
} 复制代码 映射的 make 分配
初始容量的作用
在使用
创建映射时,指定初始容量可以提高映射的性能。如果预先知道映射可能存储的元素数量,指定合适的初始容量可以减少哈希表扩容的次数。
package main
import "fmt"
func main() {
// 创建一个初始容量为 10 的映射
m := make(map[string]int, 10)
for i := 0; i < 10; i++ {
key := fmt.Sprintf("key%d", i)
m[key] = i
}
fmt.Println("映射内容:", m)
} 复制代码 项目场景:缓存系统
在缓存系统中,映射可以用于存储缓存数据。使用
创建映射并指定合适的初始容量,可以提高缓存系统的性能。
package main
import (
"fmt"
)
type Cache struct {
data map[string]interface{}
}
func NewCache(capacity int) *Cache {
return &Cache{
data: make(map[string]interface{}, capacity),
}
}
func (c *Cache) Set(key string, value interface{}) {
c.data[key] = value
}
func (c *Cache) Get(key string) (interface{}, bool) {
val, exists := c.data[key]
return val, exists
}
func main() {
cache := NewCache(20)
cache.Set("item1", 100)
val, exists := cache.Get("item1")
if exists {
fmt.Println("缓存中获取的值:", val)
}
} 复制代码 通道的 make 分配
有缓冲通道与无缓冲通道
使用
创建通道时,可以指定缓冲区大小。如果不指定缓冲区大小,创建的是无缓冲通道,发送和接收操作会阻塞;如果指定了缓冲区大小,创建的是有缓冲通道,只有当缓冲区满时发送操作才会阻塞,只有当缓冲区为空时接收操作才会阻塞。
package main
import "fmt"
func main() {
// 创建无缓冲通道
ch1 := make(chan int)
go func() {
ch1 <- 10
fmt.Println("数据已发送到无缓冲通道")
}()
fmt.Println("从无缓冲通道接收:", <-ch1)
// 创建有缓冲通道
ch2 := make(chan int, 2)
ch2 <- 20
ch2 <- 30
fmt.Println("从有缓冲通道接收:", <-ch2)
} 复制代码 项目场景:并发任务处理
在并发任务处理中,通道可以用于协程之间的通信。使用
创建合适的通道可以协调不同协程的工作。
package main
import (
"fmt"
)
func worker(id int, jobs <-chan int, results chan<- int) {
for j := range jobs {
fmt.Printf("Worker %d 开始处理任务 %d\n", id, j)
results <- j * 2
}
}
func main() {
const numJobs = 5
jobs := make(chan int, numJobs)
results := make(chan int, numJobs)
// 启动 3 个工作协程
for w := 1; w <= 3; w++ {
go worker(w, jobs, results)
}
// 发送任务
for j := 1; j <= numJobs; j++ {
jobs <- j
}
close(jobs)
// 收集结果
for a := 1; a <= numJobs; a++ {
<-results
}
close(results)
} 复制代码 总结
make 函数在 Go 语言中是为切片、映射和通道进行内存分配和初始化的重要工具。通过合理使用 make 函数,可以根据不同的需求为这些引用类型分配合适的内存,提高程序的性能和效率。在实际项目中,无论是数据处理、缓存系统还是并发任务处理,make 函数都发挥着关键作用。开发者需要深入理解 make 函数的使用方法和不同类型的特点,根据具体的场景灵活运用。
以上就是Go语言使用make进行内存分配的代码示例的详细内容,更多关于Go make内存分配的资料请关注晓枫资讯其它相关文章!
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
1 、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
2 、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
3 、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:
进行删除处理。
4 、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
5 、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。