前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >为什么会有atomic.LoadInt32

为什么会有atomic.LoadInt32

作者头像
LA0WAN9
发布于 2021-12-14 01:12:05
发布于 2021-12-14 01:12:05
1.1K00
代码可运行
举报
文章被收录于专栏:火丁笔记火丁笔记
运行总次数:0
代码可运行

前些天我们聊了 Golang 内存对齐的话题,后来我突然想到另一个问题:为什么会有 atomic.LoadInt32?可能你觉得思维太跳跃了,容我慢慢道来:首先,有 atomic.LoadInt64 很正常,因为对一个 int64 来说,它的大小是 8 个字节,如果是 32 位平台的话(字长 4 字节),CPU 一次最多操作 4 个字节,需要两次才能拿到全部数据,所以封装一个 atomic.LoadInt64 来实现原子操作;但是,对一个 int32 数据来说,它的大小是 4 字节,不管是 32 位平台(字长 4 字节),还是 64 位平台(字长 8 字节),CPU 应该都可以保证一次操作拿到数据,换句话说,如果读取一个 int32 数据,那么本身就应该是原子的,可是为什么会有 atomic.LoadInt32,这不是脱了裤子放屁么?

有病没病走两步,让我们写一段代码来验证一下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
package main

import "sync/atomic"

var v = int32(0)

func main() {
	var x int32
	x = v // main.go:9
	_ = x
	x = atomic.LoadInt32(&v) // main.go:11
	_ = x
}

通过「go tool compile」运行代码,拿到对应的汇编结果:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
shell> go tool compile -N -l -S main.go

0x0016 00022 (main.go:9)        MOVL    "".v(SB), AX
0x001c 00028 (main.go:9)        MOVL    AX, "".t+4(SP)
0x0020 00032 (main.go:11)       MOVL    "".v(SB), AX
0x0026 00038 (main.go:11)       MOVL    AX, "".t+4(SP)

不管是「x = v」还是「x = atomic.LoadInt32(&v)」,对应的汇编结果一摸一样,带着困惑,让我们继续看看是否能从 sync/atomic 的源代码中找到答案:

Golang 代码中只有函数声明,实际上是使用汇编实现的:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// doc.go
func LoadInt32(addr *int32) (val int32)

// asm.s
TEXT ·LoadInt32(SB),NOSPLIT,$0
	JMP runtime∕internal∕atomic·Load(SB)

顺着路径,跳转到 runtime/internal/atomic,会发现每个平台都有独立的 Load 实现:

在 amd64 平台,Load 是用 Golang 实现的,等价于直接读取:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
func Load(ptr *uint32) uint32 {
	return *ptr
}

在 arm64 平台,Load 是用汇编实现的,并不是简单的一次操作:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
TEXT ·Load(SB),NOSPLIT,$0-12
	MOVD	ptr+0(FP), R0
	LDARW	(R0), R0
	MOVW	R0, ret+8(FP)
	RET

如上可见,atomic.LoadInt32 之所以存在,是因为某些平台存在特殊性,所以我们需要封装一个统一的操作,如此更有利于我们写出平台无关的代码。

本文仅讨论了 atomic 的原子性,实际上它还保证了可见性有序性,有兴趣的朋友可以搜索内存屏障相关内容,这是一个很复杂的主题,我就不献丑了,推荐阅读:Golang Memory Model

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021-10-08,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
只了解View的事件分发是不够的,来看下输入系统对事件的处理
在上一篇文章中,我们学习了IMS的诞生(创建),IMS创建后还会进行启动,这篇文章我们来学习IMS的启动过程和输入事件的处理。
用户1269200
2018/12/18
7520
只了解View的事件分发是不够的,来看下输入系统对事件的处理
Android输入系统的事件传递流程和IMS的诞生
很多同学可能会认为输入系统是不是和View的事件分发有些关联,确实是有些关联,只不过View事件分发只能算是输入系统事件传递的一部分。这个系列讲的输入系统主要是我们不常接触的,但还是需要去了解的那部分。
用户1269200
2018/12/05
1.9K0
Android输入系统的事件传递流程和IMS的诞生
[Android][Framework] 无障碍快捷方式相关代码
问题:无障碍快捷方式(Accessibility Shortcut)打开不生效。
wOw
2018/09/18
2K0
[Android][Framework] 无障碍快捷方式相关代码
Android GPS学习 (一) :GPS 启动流程
packages/apps/Settings/src/com/android/settings/location/LocationSwitchBarController.java
用户7557625
2020/07/15
2.5K0
Android9.0 蓝牙开启流程
/packages/apps/Settings/src/com/android/settings/bluetooth/BluetoothSwitchPreferenceController.java
用户7557625
2020/07/15
2.2K0
Android IME输入法启动&显示&隐藏流程梳理以及常见问题&调试技巧小结
主要指输入法框架的InputMethodManager, 每个app都一个实例, 用来和输入法控制端交互。运行在需要使用输入法的进程
wizzie
2023/05/12
9.8K1
Android IME输入法启动&显示&隐藏流程梳理以及常见问题&调试技巧小结
Android框架简介--启动过程--大体流程
Android设备的启动必须经历3个阶段,即Boot Loader,Linux Kernel和Android系统服务。严格来讲Android系统实际上是运行于Linux 内核上的一系列用户进程,并不算一个严格意义上的操作系统。一般面试问及启动流程都是从init进程开始
小蚂蚁与大象
2022/04/02
8700
Android框架简介--启动过程--大体流程
Android UsageStatsService(应用使用统计服务)的学习与调研
一. 简介 UsageStatsService是一个系统服务,其主要通过AMS等,来监测并记录各个应用的使用数据,如上次调用com.android.settings的时间等。 UsageStatsSe
宋凯伦
2018/01/04
3.1K0
Android系统启动流程(三)解析SyetemServer进程启动过程
前言 上一篇我们学习了Zygote进程,并且知道Zygote进程启动了SyetemServer进程,那么这一篇我们就来学习Android7.0版本的SyetemServer进程的启动过程。 1.Zygote启动SyetemServer进程 在上一篇文章中我们讲到在ZygoteInit.java的startSystemServer函数中启动了SyetemServer进程,如下所示。 frameworks/base/core/java/com/android/internal/os/ZygoteInit.jav
用户1269200
2018/02/01
1.3K0
Android系统启动流程(三)解析SyetemServer进程启动过程
WifiDisplay开启流程
我们先来看一下WifiDisplay的代码,先从UI看起。 WifiDisplay是在Wifi P2P的基础上发展而来的,他的功能实现也离不开P2P。所以在WifiDisplaySettings的onCreate中,就先初始化了WifiP2P。 /packages/apps/Settings/src/com/android/settings/wfd/WifiDisplaySettings.java
用户7557625
2020/07/15
1.9K0
按下电源键后竟然发生了这一幕 —— Android 系统启动流程分析
作为一名 Android 程序员,你有没有想过:那么复杂的 Android 系统,它是怎样运行起来的,我们的 App 又是怎样被 Android 系统加载后呈现在屏幕上的呢?Android 系统的启动是一个比较复杂的过程,涉及到了一些我们没有接触过的知识,本文基于 Android Nougat 最新的代码上讲述 Android 系统的启动流程。
程序亦非猿
2019/08/16
8310
按下电源键后竟然发生了这一幕 —— Android 系统启动流程分析
Android 11 WiFi开启流程
从刚接触WiFi时跟过wifi的开启流程,当时还是android9。到了Android11代码架构有了不小的改动,在这里重新梳理一遍,便于在工作中更快速的跟踪代码。
用户7557625
2021/07/19
5.3K0
【Android 事件分发】事件分发源码分析 ( 驱动层通过中断传递事件 | WindowManagerService 向 View 层传递事件 )
【Android 事件分发】事件分发源码分析 ( 驱动层通过中断传递事件 | WindowManagerService 向 View 层传递事件 )
韩曙亮
2023/03/29
1.2K0
【Android 事件分发】事件分发源码分析 ( 驱动层通过中断传递事件 | WindowManagerService 向 View 层传递事件 )
Android中Input事件初始化、接收以及分发
至此 , InputManager完成初始化. 接下来就等待/dev/input中添加设备文件.
None_Ling
2020/09/24
2.1K0
Android 当点击飞行模式都发生了什么?
今天中午午休时,我把手机开飞行模式了,能看到 WiFi 、蜂窝数据和蓝牙都关闭了,心想,这时候还能收到短信吗?顺着好奇心,我们不妨来研究一下源码,看看点击飞行模式都发生了什么?
吴小龙同學
2019/12/11
1.7K0
Android | 输入系统(IMS)
一般情况下很多同学对于点击事件的认识都只存在于从 Activity 开始的,然后从 Window 中进行分发,甚至有些人可能也只知道 onTouchEvent 和 dispatchTouchEvetn 这几个方法,对于 View 层的了解都不属性。
345
2023/03/08
1.6K0
Android | 输入系统(IMS)
Android9.0Wifi热点开启流程梳理
Android9.0中对热点做了较大改动,将热点很大程度从Wifi中剥离出来了。 下面我们看一下热点是怎么开启的。
用户7557625
2020/07/15
4.6K0
Android DropBox模块笔记
frameworks/base/services/java/com/android/server/SystemServer.java - 函数run() —–> 调用函数startOtherServices() —–> 调用类SystemServiceManager的函数startService(),注册到SystemService列表
wizzie
2022/11/21
3.2K0
Android DropBox模块笔记
Android Framework学习(三)之SyetemServer进程启动解析
从上篇博客中,我们知道了Zygote进程启动了SyetemServer进程,本篇博客我们就一起来学习SyetemServer进程。
老马的编程之旅
2022/06/22
4290
Android系统启动之SystemServer
SystemServer是由Zygote启动的. 源码路径frameworks/base/services/java/com/android/server/SystemServer.java
李小白是一只喵
2020/04/23
2.4K0
Android系统启动之SystemServer
推荐阅读
相关推荐
只了解View的事件分发是不够的,来看下输入系统对事件的处理
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验