Java与Go:指针

在计算机内存中,每个变量都有一个唯一的地址,指针就是用来保存这个地址的变量。通过指针,我们可以间接地访问和修改存储在该地址处的数据。今天我们来聊一聊Java和Go指针,预告一下,我们需要借助C语言做一些小小的比较。

Java

在Java中,不存在直接的指针概念,而是通过引用来访问对象。Java中的对象是通过引用来操作的,这些引用本质上是对对象的引用,而不是指向内存地址的指针。Java的引用是一种高级抽象,它隐藏了底层内存管理的细节,开发者不需要关心对象的内存分配和释放。Java中的引用可以被认为是一种安全的指针,它提供了更高的抽象级别,可以减少内存管理错误的发生。其实大家第一门编程语言应该就是大一学习的C语言编程吧。当时学习指针的时候很懵,尤其是各种指针运算。后来学Java的对象以及引用再回来看就很好理解。

CStudent *student = (Student*)malloc(sizeof(Student));JavaStudent student = new Student();

他们的本质是一样的,都是指向堆空间的某一块地址。之后再回去写C语言,就越写越顺,每次创建变量之前都问自己一句,在栈里还是堆里? 然后决定要不要用指针。

话说回来,sun.misc.Unsafe类提供了一种机制来进行一些底层的、不安全的操作,包括直接操作内存和执行不受限制的任意指针算术运算等。

sun.misc.Unsafe类并不是 Java 标准 API 的一部分,因此它并不受到 Java 平台的正式支持,并且可能在未来的 Java 版本中被移除或更改。它主要被用来实现 Java 核心类库和一些 Java 虚拟机的实现。

通过sun.misc.Unsafe类,你可以直接进行一些底层的内存操作,比如:

  • 分配内存
  • 释放内存
  • 修改内存中的值
  • 进行指针算术运算等
    使用sun.misc.Unsafe类需要谨慎,因为它涉及到底层的内存操作,可能会导致不稳定性和不可预测的结果。此外,由于它不是 Java 标准 API 的一部分,因此在不同的 Java 实现中,它的行为可能会有所不同。

在 Java 9 中,sun.misc.Unsafe类的一些方法被标记为不安全,并且在一些场景下会抛出 java.lang.UnsupportedOperationException 异常,这是为了增强 Java 应用程序的安全性。

总的来说,除非你对 Java 的内存模型和底层运行机制非常了解,并且对使用 sun.misc.Unsafe 类的风险有所认识,并且确实需要进行底层的内存操作,否则不建议使用 sun.misc.Unsafe 类。Java部分就到此结束,毕竟不是重点。

Go 中的指针

Go语言支持指针,但是和C语言中的指针还是有些不同。在Go中,指针是一种数据类型,它指向了一个内存地址,允许你直接访问内存中的数据。与C语言不同的是,Go语言的指针是类型安全的,不允许进行指针运算,从而减少了一些常见的指针错误。

在Go中,你可以通过使用操作符来声明指针变量,使用&操作符来获取变量的地址,使用操作符来获取指针指向的值。

依稀记得当时学习C语言的时候老师引出的指针的第一个例子:将两个int数传入一个方法,希望这两个int互换数值。这里我用Go来写:

package mainimport "fmt"// 定义一个交换函数
func swap(x, y int) {var i int = xx = yy = i
}func main() {// 定义两个整数变量a, b := 5, 10fmt.Println("Before swapping:")fmt.Println("a =", a)fmt.Println("b =", b)// 调用交换函数,并接收返回值swap(a, b)fmt.Println("After swapping:")fmt.Println("a =", a)fmt.Println("b =", b)
}

结果显而易见,我们失败了。然后老师开始长篇大论的解释,然后我就睡着了。其实现在回来再想想很简单,x和y的变量在main的栈帧里面,调用swap方法无非就是创建新的栈帧,并把xy的数值复制一边传递出去。抓捕周树人和我鲁迅有什么关系?

那正确写法就是搞一个指向栈内存的指针。

package mainimport "fmt"// 定义一个交换函数
func swap(x, y *int) {temp := *x*x = *y*y = temp
}func main() {// 定义两个整数变量a, b := 5, 10fmt.Println("Before swapping:")fmt.Println("a =", a)fmt.Println("b =", b)// 调用交换函数swap(&a, &b)fmt.Println("After swapping:")fmt.Println("a =", a)fmt.Println("b =", b)
}

接下来我们再看看数组和指针碰到一起会发生什么

指针和数组

指针指向数组中的某一个元素
var arr [5]int
var ptr *int
ptr = &arr[0] // 将指针指向数组的第一个元素
指针指向数组整体
// 声明一个数组
var arr [5]int
// 声明一个指向数组的指针
var ptr *[5]int
// 将指针指向数组
ptr = &arr

这里要注意类型匹配,我们之前说过长度也是类型的一部分,[5]int和[7]int是不一样的。[5]int要配合*[5]int

眼神要好
var ptrArr *[3]int //指向一个数组
var ptrArr [3]*int //每个元素指向一个数
传递数组

之前我们说过,在方法间传递数组其实是复制整个数组传递过去,相当于寻觅另外一个地方盖一座一模一样的房子。如果我们传递指向数组的指针,函数将能够修改原始数组的值。

func modifyArray(arr *[5]int) {// 修改数组的值(*arr)[0] = 100
}func main() {var arr [5]intmodifyArray(&arr)fmt.Println(arr) // 输出 [100 0 0 0 0]
}

指针和切片

之前我们已经学过,切片其实就是对数组进行包装,内部通过指针对数组进行操作。那么当切片和指针一起使用时,可以实现更灵活和高效的数据操作,比如动态地管理内存和访问数组的部分元素(二级指针嘛,玩C的都懂)。以下是一些示例,演示了切片和指针的结合使用:

package mainimport "fmt"func main() {// 创建一个切片slice := []int{1, 2, 3, 4, 5}// 创建一个指向切片的指针var ptr *[]intptr = &slice// 修改切片的值(*ptr)[0] = 100(*ptr)[1] = 200// 打印修改后的切片fmt.Println(*ptr) // 输出: [100 200 3 4 5]
}

个人觉得切片指针反而比数组用起来更轻松。map也一样道理:

package mainimport "fmt"func modifyMap(m *map[string]int) {// 向 map 中添加新的键值对(*m)["d"] = 4(*m)["e"] = 5// 修改已有键的值(*m)["a"] = 100
}func main() {// 创建一个 mapmyMap := map[string]int{"a": 1,"b": 2,"c": 3,}// 创建指向 map 的指针ptr := &myMap// 修改 map 内容modifyMap(ptr)// 打印修改后的 mapfmt.Println(*ptr)
}

指针和结构体

在 Go 语言中,指向结构体的指针是一种非常常见的用法基本和C语言一样。结构体是一种用户自定义的复合数据类型,它可以包含多个不同类型的字段,而指向结构体的指针则允许我们直接访问结构体的字段,并且可以在函数之间共享结构体的实例,而不需要进行复制。以下是指向结构体的指针如何使用的详细说明:

创建结构体指针
// 定义结构体
type Person struct {Name stringAge  int
}func main() {// 创建结构体的指针var p *Personp = &Person{"John", 30}// 或者使用 new() 函数创建结构体的指针p = new(Person)p.Name = "Alice"p.Age = 25
}

可以理解这里的new就相当于(*Person)malloc(sizeof(Person));

操作结构体字段
访问结构体字段:
fmt.Println((*p).Name)  // 打印结构体字段 Name
fmt.Println(p.Age)       // 也可以直接使用 p.Age 访问
修改结构体字段:
p.Name = "Bob"  // 直接赋值修改结构体字段
p.Age = 40
传递结构体指针给函数:
func modifyPerson(p *Person) {if p != nil {p.Age = 50}
}modifyPerson(p) // 调用函数修改结构体的字段值

总的来说,指向结构体的指针在 Go 中是非常常见的用法,它允许我们在函数间共享结构体的实例,并且可以在需要时直接访问和修改结构体的字段。使用指向结构体的指针可以避免结构体的复制,提高程序的性能和效率。

总结

Java中的指针是被隐藏的,程序员无法直接操作内存地址,而Go中的指针是一等公民,允许直接操作内存地址。本文仅仅介绍了指针操作的冰山一角,之后我们会继续介绍指针在面向对象编程中的应用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/280502.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于cnn深度学习的yolov5+pyqt+分类+resnet+骨龄检测系统

往期热门博客项目回顾: 计算机视觉项目大集合 改进的yolo目标检测-测距测速 路径规划算法 图像去雨去雾目标检测测距项目 交通标志识别项目 yolo系列-重磅yolov9界面-最新的yolo 姿态识别-3d姿态识别 深度学习小白学习路线 YOLOv5与骨龄识别 YOLOv5&a…

容器中的大模型(三)| 利用大语言模型:容器化高效地部署 PDF 解析器实践...

作者:宋文欣,智领云科技联合创始人兼CTO 01 简介 大语言模型(LLMs)正逐渐成为人工智能领域的一颗璀璨明星,它们的强大之处在于能够理解和生成自然语言,为各种应用提供了无限可能。为了让这些模型更好地服务…

网络编程:数据库

一、作业 1> 创建一个工人信息库,包含工号(主键)、姓名、年龄、薪资。 2> 添加三条工人信息(可以完整信息,也可以非完整信息) 3> 修改某一个工人的薪资(确定的一个) 4> …

[C++]20:unorderedset和unorderedmap结构和封装。

unorderedset和unorderedmap结构和封装 一.哈希表&#xff1a;1.直接定址法&#xff1a;2.闭散列的开放定址法&#xff1a;1.基本结构&#xff1a;2.insert3.find4.erase5.补充&#xff1a;6.pair<k,v> k的数据类型&#xff1a; 3.开散列的拉链法/哈希桶&#xff1a;1.基…

【计算机网络】计算机网络概述

文章目录 一、计算机网络的概念二、 计算机网络的功能1. 数据通信2. 资源共享3. 分布式处理4. 提高可靠性5. 负载均衡 补充&#xff1a; 计算机的发展阶段小结三、计算机网络的组成1. 组成部分2. 工作方式3. 功能组成 四、 计算机网络的分类1. 按分布范围2. 按使用者3. 按交换技…

零拷贝原理+kafka中的零拷贝

零拷贝原理kafka中的零拷贝 kafka性能之零拷贝传统IO零拷贝mmp优化sendfile优化sendfile DMA scatter/gather优化Kafka是怎么使用零拷贝的 kafka性能之零拷贝 kafka中的零拷贝并不是说完全避免了上下文切换与cpu拷贝的次数, 而是减少这种拷贝次数 传统IO 传统的一次IO流程 rea…

学习开发小程序的起航日记

2024年3月16日 不知不觉中三月份还只剩了一半的光景&#xff0c;我想写的内容还很多没有写&#xff0c;或者更应该说&#xff0c;是想积累的还有很多。现在最应该去完善Java的内容&#xff0c;可还是想先等等。想等搞清楚小程序部分&#xff0c;想等积累完小程序的内容。 这几…

华为综合案例-普通WLAN全覆盖配置(2)

组网图 结果验证 在AC_1和AC_2上执行display ap all命令&#xff0c;检查当前AP的状态&#xff0c;显示以下信息表示AP上线成功。[AC_1] display ap all Total AP information: nor : normal [1] ExtraInfo : Extra information P : insufficient power supply ---…

冒泡排序的原理及其实现

✨✨✨学习的道路很枯燥&#xff0c;希望我们能并肩走下来! 目录 前言 一、冒泡排序的原理 二、代码实现 总结 前言 本篇详细介绍了冒泡排序的原理及其实现&#xff0c;让使用者对冒泡排序的原理及其实现有进一步认识&#xff0c;而不是仅仅停留在表面&#xff0c;更好的模…

xercesc库保存XML功能实现

目录 一 参考链接 二 运行结果 三 代码 一 参考链接 DOM Programming Guide (apache.org) Xerces-c DOM XML文件的构造_xerces-c domimplementation-CSDN博客 Xerces-c库的使用-CSDN博客 二 运行结果 三 代码 #if 1//参考链接&#xff1a; https://blog.csdn.net/RGBMa…

流畅的 Python 第二版(GPT 重译)(九)

第四部分&#xff1a;控制流 第十七章&#xff1a;迭代器、生成器和经典协程 当我在我的程序中看到模式时&#xff0c;我认为这是一个麻烦的迹象。程序的形状应该只反映它需要解决的问题。代码中的任何其他规律性对我来说都是一个迹象&#xff0c;至少对我来说&#xff0c;这表…

【数据可视化】Echarts中的其它图表

个人主页 &#xff1a; zxctscl 如有转载请先通知 文章目录 1. 前言2. 绘制散点图2.1 绘制基本散点图2.2 绘制两个序列的散点图2.3 绘制带涟漪特效的散点图 3. 绘制气泡图3.1 绘制标准气泡图3.2 绘制各国人均寿命与GDP气泡图3.3 绘制城市A、城市B、城市C三个城市空气污染指数气…

Tech Talks技术讲座中文培训-报名学习LPWAN、Matter、蓝牙和Wi-Fi最新开发技能!

Silicon Labs&#xff08;亦称“芯科科技”&#xff09;主办新一轮2024年“亚太区Tech Talks在线技术讲座”即将在5月9日至8月8日&#xff08;中文系列场次&#xff09;&#xff0c;以及4月24日至8月7日&#xff08;英文系列场次&#xff09;正式展开&#xff0c;现正热烈报名中…

uniapp使用Canvas给图片加水印把临时文件上传到服务器

生成的临时路径是没有完整的路径没办法上传到服务器 16:37:40.993 添加水印后的路径, _doc/uniapp_temp_1710923708347/canvas/17109238597881.png 16:37:41.041 添加水印后的完整路径, file://storage/emulated/0/Android/data/com.jingruan.zjd/apps/__UNI__BE4B000/doc/…

ES 常见面试题及答案

目录 es 写入数据流程 es 删除数据流程 es 读数据流程 es 部署的服务有哪些角色 es 的实现原理 es 和lucence 关系 如何提高写入效率 提高搜索效率 es doc value指的啥 分片指的啥&#xff0c;定义后可不可义再修改 深分页如何优化 对于聚合操作是如何优化的 元数据…

adobe animate 时间轴找不到编辑多个帧按钮

如题&#xff0c;找了半天&#xff0c;在时间轴上找不到编辑多个帧按钮,导致无法批量处理帧 然后搜索发现原来是有些版本被隐藏了&#xff0c;需要再设置一下 勾选上就好了

POI和EasyExcel区别和操作Excel

POI和EasyExcel操作Excel 常用场景 1、将用户信息导出为excel表格&#xff08;导出数据… &#xff09; 2、将Excel表中的信息录入到网站数据库&#xff08;文件数据上传… &#xff09; 开发中经常会设计到excel的处理&#xff0c;如导出Excel&#xff0c;导入Excel到数据库…

鸿蒙Harmony应用开发—ArkTS-转场动画(组件内隐式共享元素转场)

geometryTransition用于组件内隐式共享元素转场&#xff0c;在组件显示切换过程中提供平滑过渡效果。通用transition机制提供了opacity、scale等转场动效&#xff0c;geometryTransition通过id绑定in/out组件(in指入场组件、out指出场组件)&#xff0c;使得组件原本独立的trans…

Gateway新一代网关

Gateway新一代网关 1、概述 ​ Cloud全家桶中有个很重要的组件就是网关&#xff0c;在1.x版本中都是采用的Zuul网关&#xff1b; ​ 但在2.x版本中&#xff0c;zuul的升级一直跳票&#xff0c;SpringCloud最后自己研发了一个网关SpringCloud Gateway替代Zuul。 ​ 官网&…

手机运营商二要素检测:重塑信任基石,筑牢信息安全屏障

随着移动互联网的普及和数字经济的快速发展&#xff0c;用户信息安全的重要性日益凸显。运营商二要素检测作为一种强大的安全验证机制&#xff0c;以其精准匹配与实时验证的特性&#xff0c;为各类应用场景提供了一种可靠的身份识别解决方案&#xff0c;正在成为众多企业和服务…