Golang

关注公众号 jb51net

关闭
首页 > 脚本专栏 > Golang > Go语言队列

Go语言队列的四种实现及使用场景

作者:Jayden_念旧

本文主要介绍了Go语言队列的四种实现及使用场景,包括切片、链表、带锁队列和Channel,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

队列(Queue)是一种先进先出(FIFO)的数据结构,队列是算法和并发编程中的基础数据结构,掌握其实现方式对Go开发者非常重要。接下来我将结合案例以及相关的资料,讲解队列的实现。

1. 使用切片(Slice)实现简单队列

下面用泛型切片(Slice)实现一个简单队列,首先声明队列类型,队列中的元素类型可以是任意的,所以类型约束为 any

type Queue[T any] []T

这里做一个延申,大家在查找资料的时候会看到如下的写法:

type Queue []interface{}

区别是:前者为Go 1.18+引入的泛型写法后者为Go 1.18之前的标准写法。

泛型写法特点:

标准写法特点:

小编更倾向泛型写法,当然这个因人而异。虽然标准写法可以存储混合类型,但是泛型写法可以是使用any解决。

队列的基本操作有四个方法:入队(Push)、出队(Pop)、查看队首元素(Peek)和获取队列大小(Size)

a.定义 Push 方法:

func (q *Queue[T]) Push(e T) {
  *q = append(*q, e)
}

b.定义 Pop 方法:

func (q *Queue[T]) Pop() (_ T) {
  if q.Size() > 0 {
    res := q.Peek()
    *q = (*q)[1:]
    return res
  }
  return
}

c.定义peek方法

func (q *Queue[T]) Peek() (_ T) {
  if q.Size() > 0 {
    return (*q)[0]
  }
  return
}

d.定义 Size 方法:

func (q *Queue[T]) Size() int {
  return len(*q)
}

尝试打印一下:

func main() {
	q := Queue[int]{}
	q.Push(1)
	q.Push(2)
	q.Push(3)
	fmt.Println(q)
	fmt.Println(q.Pop())
	fmt.Println(q.Peek())
	fmt.Println(q.Size())
}

输出结果

[1 2 3]
1
2
2

2. 使用链表实现高效队列

package main
 
import (
    "container/list"
    "fmt"
)
 
// 队列内部用了一个链表来存储元素
type Queue[T any] struct {
    list *list.List
}
 
func NewQueue[T any]() *Queue[T] {
    return &Queue[T]{list: list.New()}
}
 
func (q *Queue[T]) Enqueue(item T) {
    q.list.PushBack(item)
}
 
func (q *Queue[T]) Dequeue() (T, error) {
    if q.list.Len() == 0 {
        var zero T
        return zero, fmt.Errorf("queue is empty")
    }
    front := q.list.Front()
    q.list.Remove(front)
    return front.Value.(T), nil
}
 
func (q *Queue[T]) Peek() (T, error) {
    if q.list.Len() == 0 {
        var zero T
        return zero, fmt.Errorf("queue is empty")
    }
    return q.list.Front().Value.(T), nil
}
 
func (q *Queue) IsEmpty() bool {
    return q.list.Len() == 0
}
 
func (q *Queue) Size() int {
    return q.list.Len()
}
 
func main() {
    q := NewQueue[string]()
    q.Enqueue("a")
    q.Enqueue("b")
    q.Enqueue("c")
    
 
	if val, err := q.Dequeue(); err == nil {
		fmt.Println(val)
	}
 
	if peekVal, err := q.Peek(); err == nil {
		fmt.Println(peekVal)
	}
    fmt.Println(q.Size())    // 2
}

解读一下与上一个例子不同的地方:

1.创建新队列

func NewQueue[T any]() *Queue[T] {
    return &Queue[T]{list: list.New()}
}

2.入队和出队函数

3.主函数

	if peekVal, err := q.Peek(); err == nil {
		fmt.Println(peekVal)
	} else {
		fmt.Println("Error:", err)
	}

3.并发安全队列

首先我们先了解什么是并发环境

并发环境是指程序中多个执行流同时或交替运行的上下文,这些执行流可能共享资源并需要协调操作。在Go语言中,理解并发环境对编写正确高效的代码至关重要。

核心概念

  1. 并发 vs 并行

    • 并发:逻辑上的同时处理(单核CPU通过时间片切换)
    • 并行:物理上的同时执行(多核CPU真正同时运行)
  2. Go的并发模型

    • 基于Goroutine(轻量级线程)
    • 通过Channel通信(而不是共享内存)
    • 标准库提供sync包处理同步

并发环境的典型特征

  1. 资源共享

    • 多个Goroutine访问同一变量/数据结构
    • 示例:多个请求同时修改缓存
  2. 执行顺序不确定性

    • Goroutine调度顺序不可预测
    • 示例:两个Goroutine对同一变量先后写操作
  3. 竞态条件(Race Condition)

    • 程序行为依赖于不可控的执行时序
    • 示例:计数器未同步导致计数错误

接下来是并发安全队列

package main

import (
	"container/list"
	"fmt"
	"sync"
)

type ConcurrentQueue[T any] struct {
	list *list.List
	lock sync.Mutex
}

func NewQueue[T any]() *ConcurrentQueue[T] {
	return &ConcurrentQueue[T]{list: list.New()}
}

func (q *ConcurrentQueue[T]) Enqueue(item T) {
	q.lock.Lock()
	defer q.lock.Unlock()
	q.list.PushBack(item)
}

func (q *ConcurrentQueue[T]) Dequeue() (T, error) {
	q.lock.Lock()
	defer q.lock.Unlock()

	if q.list.Len() == 0 {
		var zero T
		return zero, fmt.Errorf("queue is empty")
	}

	front := q.list.Front()
	q.list.Remove(front)
	return front.Value.(T), nil
}

// 带类型安全的Peek方法
func (q *ConcurrentQueue[T]) Peek() (T, error) {
	q.lock.Lock()
	defer q.lock.Unlock()

	if q.list.Len() == 0 {
		var zero T
		return zero, fmt.Errorf("queue is empty")
	}
	return q.list.Front().Value.(T), nil
}

func main() {
	q := NewQueue[string]()
	q.Enqueue("first")
	q.Enqueue("second")

	if val, err := q.Dequeue(); err == nil {
		fmt.Println("Dequeue:", val) // 输出:Dequeue: first
	}

	if peekVal, err := q.Peek(); err == nil {
		fmt.Println("Peek:", peekVal) // 输出:Peek: second
	}
}

代码解读:

1.入队出队

2.查看队首元素

4. 使用channel实现队列

package main

import "fmt"

func main() {
    queue := make(chan int, 3) // 缓冲大小为3
    
    queue <- 1
    queue <- 2
    queue <- 3
    
    fmt.Println(<-queue) // 1
    fmt.Println(<-queue) // 2
}

四种方法的选择场景

  1. 对于简单场景,使用切片实现即可
  2. 需要频繁增删元素时,使用链表实现更高效
  3. 并发环境使用带锁的队列或channel
  4. channel适合生产者-消费者模式

到此这篇关于Go语言队列的四种实现及使用场景的文章就介绍到这了,更多相关Go语言队列内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文