返回顶部
首页 > 资讯 > 精选 >使用新的atomic.Pointer类型实现无锁无界队列
  • 593
分享到

使用新的atomic.Pointer类型实现无锁无界队列

2024-02-09 11:02:25 593人浏览 泡泡鱼
摘要

PHP小编草莓今天要为大家介绍一种新的技术——使用新的atomic.Pointer类型实现无锁无界队列。在并发编程中,队列是一种常见的数据结构,但传统的队列实现通常需要使用锁来保证线程

PHP小编草莓今天要为大家介绍一种新的技术——使用新的atomic.Pointer类型实现无无界队列。在并发编程中,队列是一种常见的数据结构,但传统的队列实现通常需要使用锁来保证线程安全,这会带来性能的损耗。而新的atomic.Pointer类型则提供了一种无锁的解决方案,可以实现高效的并发队列操作。下面我们将详细介绍这种新的实现方式,以及它的优势和使用方法。

问题内容

我正在尝试实现 michael 和 scott 的这个非阻塞队列。

我正在尝试使用 Go 1.19 中引入的新的atomic.pointer 类型,但我的应用程序中出现了数据争用。

这是我的实现:

package queue

import (
    "errors"
    "sync/atomic"
)

// LockfreeQueue represents a FIFO structure with operations to enqueue
// and dequeue generic values.
// Reference: https://www.cs.rochester.edu/research/synchronization/pseudocode/queues.html
type LockFreeQueue[T any] struct {
    head atomic.Pointer[node[T]]
    tail atomic.Pointer[node[T]]
}

// node represents a node in the queue
type node[T any] struct {
    value T
    next  atomic.Pointer[node[T]]
}

// newNode creates and initializes a node
func newNode[T any](v T) *node[T] {
    return &node[T]{value: v}
}

// NewQueue creates and initializes a LockFreeQueue
func NewLockFreeQueue[T any]() *LockFreeQueue[T] {
    var head atomic.Pointer[node[T]]
    var tail atomic.Pointer[node[T]]
    n := &node[T]{}
    head.Store(n)
    tail.Store(n)
    return &LockFreeQueue[T]{
        head: head,
        tail: tail,
    }
}

// Enqueue adds a series of Request to the queue
func (q *LockFreeQueue[T]) Enqueue(v T) {
    n := newNode(v)
    for {
        tail := q.tail.Load()
        next := tail.next.Load()
        if tail == q.tail.Load() {
            if next == nil {
                if tail.next.CompareAndSwap(next, n) {
                    q.tail.CompareAndSwap(tail, n)
                    return
                }
            } else {
                q.tail.CompareAndSwap(tail, next)
            }
        }
    }
}

// Dequeue removes a Request from the queue
func (q *LockFreeQueue[T]) Dequeue() (T, error) {
    for {
        head := q.head.Load()
        tail := q.tail.Load()
        next := head.next.Load()
        if head == q.head.Load() {
            if head == tail {
                if next == nil {
                    return head.value, errors.New("queue is empty")
                }
                q.tail.CompareAndSwap(tail, next)
            } else {
                v := next.value
                if q.head.CompareAndSwap(head, next) {
                    return v, nil
                }
            }
        }
    }
}

// Check if the queue is empty.
func (q *LockFreeQueue[T]) IsEmpty() bool {
        return q.head.Load() == q.tail.Load()
}

我在这里发现了一个不同的实现,它可以在我的应用程序中工作而无需数据竞争,但我似乎无法弄清楚两者之间到底有什么不同。

感谢任何帮助或反馈!

解决方法

事实证明,更改一些内容可以解决问题。

第一个变化:

var n = node[t]{}
head.store(&n)
tail.store(&n)

第二个更改是更改 dequeue() 返回签名。

最终文件如下所示:

package queue

import (
    "sync/atomic"
)

// LockfreeQueue represents a FIFO structure with operations to enqueue
// and dequeue generic values.
// Reference: Https://www.cs.rochester.edu/research/synchronization/pseudocode/queues.html
type LockFreeQueue[T any] struct {
    head atomic.Pointer[node[T]]
    tail atomic.Pointer[node[T]]
}

// node represents a node in the queue
type node[T any] struct {
    value T
    next  atomic.Pointer[node[T]]
}

// newNode creates and initializes a node
func newNode[T any](v T) *node[T] {
    return &node[T]{value: v}
}

// NewQueue creates and initializes a LockFreeQueue
func NewLockFreeQueue[T any]() *LockFreeQueue[T] {
    var head atomic.Pointer[node[T]]
    var tail atomic.Pointer[node[T]]
    var n = node[T]{}
    head.Store(&n)
    tail.Store(&n)
    return &LockFreeQueue[T]{
        head: head,
        tail: tail,
    }
}

// Enqueue adds a series of Request to the queue
func (q *LockFreeQueue[T]) Enqueue(v T) {
    n := newNode(v)
    for {
        tail := q.tail.Load()
        next := tail.next.Load()
        if tail == q.tail.Load() {
            if next == nil {
                if tail.next.CompareAndSwap(next, n) {
                    q.tail.CompareAndSwap(tail, n)
                    return
                }
            } else {
                q.tail.CompareAndSwap(tail, next)
            }
        }
    }
}

// Dequeue removes a Request from the queue
func (q *LockFreeQueue[T]) Dequeue() T {
    var t T
    for {
        head := q.head.Load()
        tail := q.tail.Load()
        next := head.next.Load()
        if head == q.head.Load() {
            if head == tail {
                if next == nil {
                    return t
                }
                q.tail.CompareAndSwap(tail, next)
            } else {
                v := next.value
                if q.head.CompareAndSwap(head, next) {
                    return v
                }
            }
        }
    }
}

// Check if the queue is empty.
func (q *LockFreeQueue[T]) IsEmpty() bool {
    return q.head.Load() == q.tail.Load()
}

以上就是使用新的atomic.Pointer类型实现无锁无界队列的详细内容,更多请关注编程网其它相关文章!

--结束END--

本文标题: 使用新的atomic.Pointer类型实现无锁无界队列

本文链接: https://lsjlt.com/news/562475.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 使用新的atomic.Pointer类型实现无锁无界队列
    php小编草莓今天要为大家介绍一种新的技术——使用新的atomic.Pointer类型实现无锁无界队列。在并发编程中,队列是一种常见的数据结构,但传统的队列实现通常需要使用锁来保证线程...
    99+
    2024-02-09
  • C++11如何实现无锁队列
    无锁操作的本质依赖的原子操作,C++11提供了atomic的原子操作支持 atomic compare_exchange_weak / compare_exchange_stron...
    99+
    2024-04-02
  • 使用Java无界队列的线程池会怎么样
    本篇内容主要讲解“使用Java无界队列的线程池会怎么样”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“使用Java无界队列的线程池会怎么样”吧!(1)背景引入今天跟大家聊一个互联网大厂的Java面...
    99+
    2023-06-04
  • Vue3+Vite使用双token实现无感刷新
    目录前言一、token 登录鉴权二、何为双 token双 token 验证流程注意事项三、服务端代码1. 搭建koa2服务器2. 双token3. 路由4. 应用中间件四、前端代码1...
    99+
    2023-05-17
    Vue3 无感刷新 Vue3 双token无感刷新
  • 探索多用户操作系统的神奇世界:实现团队无缝协作
    文件共享和访问控制 多用户操作系统集成文件共享功能,使团队成员可以轻松访问和编辑共同项目文件。访问控制机制允许管理员授予不同用户不同的权限级别,确保文件安全和协作的可控性。成员可以同时编辑文档,实时查看更改,从而加快协作流程。 通信和协作...
    99+
    2024-03-14
    多用户操作系统
  • C#中使用CAS实现无锁算法的示例详解
    目录CAS 的基本概念C# 中如何使用 CAS算法示例示例1:计数器示例2:队列总结CAS 的基本概念 CAS(Compare-and-Swap)是一种多线程并发编程中常用的原子操作...
    99+
    2023-05-16
    C#使用CAS实现无锁算法 C# CAS实现无锁算法 C# CAS 无锁算法 C# 无锁算法
  • C#使用泛型队列Queue实现生产消费模式
    如果把生产消费想像成自动流水生产线的话,生产就是流水线的物料,消费就是某种设备对物料进行加工的行为,流水线就是队列。 现在,要写一个体现生产消费模式的泛型帮助类,比如叫Produce...
    99+
    2022-11-13
    C# 泛型队列 Queue 实现生产消费模式
  • .NETCore中RabbitMQ使用死信队列的实现
    在.NET Core中,可以使用RabbitMQ.Client库来实现与RabbitMQ的交互。 RabbitMQ死信队列(Dead Letter Queue)是一种用于存储和处理无...
    99+
    2023-05-14
    .NET Core RabbitMQ死信队列 .NET Core 死信队列
  • 使用PHP实现消息队列的开发
    随着现代互联网应用对高并发、高吞吐量和高可靠性的要求越来越高,消息队列作为一种异步解耦系统架构方式越来越被应用在互联网领域的各个方面。其原理是先将消息发送到消息队列中,等待异步消费,从而达到解耦的目的,提高系统的可扩展性与可维护性。在目前市...
    99+
    2023-05-25
    PHP 消息队列 开发
  • 使用javascript怎么实现页面无刷新更新数据
    这篇文章给大家介绍使用javascript怎么实现页面无刷新更新数据,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。 首先在服务器上建立一个CheckUser.asp文件,用来检测用户是否存在,根据用户是否存在...
    99+
    2023-06-10
  • 教你用python实现一个无界面的小型图书管理系统
    目录一、需求了解二、环境准备三、代码实现一、需求了解 功能模块 图书信息 二、环境准备 安装mysql数据库 参考文章: MySQL数据库压缩版本安装与配置 MySQL msi版...
    99+
    2024-04-02
  • 怎么使用css3实现一个类在线直播的队列动画
    这篇文章给大家分享的是有关怎么使用css3实现一个类在线直播的队列动画的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。之前在群里有个朋友问了这样一个问题, 就是如何在 小程序 中实现类似 直播平台 的用户上线时的 ...
    99+
    2023-06-08
  • Go错误:无法在没有实例化的情况下使用泛型类型
    php小编苹果今天要和大家分享的是关于Go语言中的一个错误:无法在没有实例化的情况下使用泛型类型。在Go语言中,泛型是一种非常强大的特性,可以让我们编写更加通用和灵活的代码。然而,有时...
    99+
    2024-02-09
    go语言
  • 使用AJAX怎么实现无刷新分页功能
    使用AJAX怎么实现无刷新分页功能,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。  首先讲一下原理:分页有两个要点:1.有多少页,2.每页有多...
    99+
    2024-04-02
  • React如何使用refresh_token实现无感刷新页面
    目录步骤如下:具体实现步骤如下: 1-token过期根据refresh_token获取新的token 重新获取数据 2-创建一个新的axios实例 【使用request防止再次进入请...
    99+
    2024-04-02
  • 使用Ajax怎么实现一个无刷新分页
    本篇文章为大家展示了使用Ajax怎么实现一个无刷新分页,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。<!doctype html><html lang=&quo...
    99+
    2023-06-08
  • 如何使用AJAX实现无刷新数据分页
    这篇文章将为大家详细讲解有关如何使用AJAX实现无刷新数据分页,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。一、实现过程注意:一下的内容都是在服务器内使用的。首先要在服务器的路径下建立几个文件,比如,pa...
    99+
    2023-06-08
  • React怎么使用refresh_token实现无感刷新页面
    这篇文章主要介绍了React怎么使用refresh_token实现无感刷新页面的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇React怎么使用refresh_token实现无感刷新页面文章都会有所收获,下面我们...
    99+
    2023-06-30
  • 为 KrakenD 实现插件时无效的节点类型恐慌
    php小编百草今天为大家介绍的是关于KrakenD插件开发中的一个常见问题:"为 KrakenD 实现插件时无效的节点类型恐慌"。KrakenD是一个快速、高性能的API网关,它提供了...
    99+
    2024-02-09
  • 怎么使用php递归实现无限级分类
    使用PHP递归实现无限级分类的步骤如下:1. 创建一个数组或从数据库中获取分类数据,包含id和parent_id字段,表示分类的唯一...
    99+
    2023-09-29
    php
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作