go中的方法 func-----数据类型

news2024/10/6 10:30:38

本文是java学习者学go种产生的容易记混点的笔记,所以有其他编译语言的基础更好

go的方法有点像js

基础

func main() {
	fmt.Println("Starting")
	var p *string = new(string)
	*p = "hello world"
	demo := "demo"
	fmt.Println(*&demo) //这样既然也可以  &得到地址 也就是指针 在通过取值操作 得到数据
	fmt.Println(*p)
	for i := 0; i < 10; i++ {
		fmt.Println(i)
		if i == 5 {
			return //结束程序
		}
	}
	fmt.Println("Ending")

}

多参数 多返回值形式

func add(a, b int) int {
    return a + b
}
func add(a, b int) int,int {
    return a + b,b
}
可变参数

形参部分是slice 动态数组

func add( b ...int) int {
    return 1
}

之所以可以不用写;就是因为底层go做了优化 所以定义方法时候 {必须跟在方法名字之后

细节-方法的传值

方法的参数传递分为值传递和引用传递(go种的引用传递是通过地址 /指针等方式)
值传递:实际形式参数接收的是数据的拷贝
引用传递:传递的是数据的引用,在函数内修改这个参数,原来的数据也会改变
为了简单记得,只要记得下下面的类型都是引用传递,其他都是值传递既可

  • 切片
  • 指针/地址
  • 映射
  • 通道
  • 接口

基本就是复合类型加上指针接口 (函数虽然也是存储地址(函数体)的数据类型,但是作为参数也是把这个函数体地址作为值传递拷贝)所以想到达引用传递的效果就需要传递函数的地址(觉得头晕可见含后文)

在java种类传递的对象也是引用传递 但是在go中并不是,go种定义类是结构体的形式,然后创建实列对象有多种方法,只有使用new关键字会返回指针这个时候实列对象名字才是引用,其他方式实列的结构体的是数据本身

/*
*
只有下面的类型是引用传递 传递数据 修改数据会影响到原来的数据 其他的只可以通过传递地址指针实现方法修改元数据
理解引用传递
*/
package main

import "fmt"

// 1. 指针(Pointers)
func modifyValue(x *int) {
	*x = 42
}

// 2. 切片(Slices)
func modifySlice(s []int) {
	s[0] = 42
}

// 3. 映射(Maps)
func modifyMap(m map[string]int) {
	m["key"] = 42
}

// 4. 通道(Channels)
func send(ch chan int) {
	ch <- 42
}

// 5. 函数(Functions)
func callFunction(f func(int) int, x int) int {
	return f(x)
}

func square(n int) int {
	return n * n
}

// 6. 接口(Interfaces)
type Modifier interface {
	Modify()
}

type Data struct {
	Value int
}

func (d *Data) Modify() {
	d.Value = 42
}

func modifyInterface(m Modifier) {
	m.Modify()
}

func main() {
	// 测试指针
	a := 10
	fmt.Println("之前 (Pointer):", a)
	modifyValue(&a)
	fmt.Println("修改后 (Pointer):", a)

	// 测试切片
	b := []int{1, 2, 3}
	fmt.Println("之前 (Slice):", b)
	modifySlice(b)
	fmt.Println("修改后 (Slice):", b)

	// 测试映射
	c := map[string]int{"key": 1}
	fmt.Println("之前 (Map):", c)
	modifyMap(c)
	fmt.Println("修改后 (Map):", c)

	// 测试通道
	ch := make(chan int)
	go send(ch)
	fmt.Println("Received (Channel):", <-ch)



	// 测试接口
	d := &Data{Value: 10}
	fmt.Println("之前 (Interface):", d.Value)
	modifyInterface(d)
	fmt.Println("修改后 (Interface):", d.Value)
}

函数式编程

首先在go种,函数也可以作为参数以及返回值存在 为此就要介绍其他语言也有的匿名函数

匿名函数

func main() {
	a := 2
	//匿名函数的使用
	func(a *int) {
		container := []int{1, 2, 3, 4, 5}
		for _, value := range container {
			if value%2 == 0 {
				fmt.Println(value)
			}
		}
	}(&a) //即可调用
	b := func() int {
		return 1
	}()
	fmt.Println(b)
	f := func() int {
		fmt.Println("hello world")
		return 1
	}
	f()
	}
	

当然匿名参数存在的作用就是可以作为回调函数传参

值得注意的函数类型作为传递时候 申明格式就和go正常申明一样,但是申明go作为参数传递时候,这个时候的参数和返回值只需要申明类型即可


func main(){
r := opation(2, 3, func(a int, b int) int {
		return a*100 + b
	})
}
/*
*
主要用途是函数式编程 作用函数的参数和返回值类型可以省略,可以直接使用匿名函数
·回调函数 作为参数只需要申明返回值和参数类型即可
*/
func opation(a, b int, callback func(int, int) int) int {
	return callback(a, b)
}

闭包函数

首先解释闭包:

闭包(Closure)函数是一种函数,它可以访问其词法作用域(定义时的作用域)之外的变量。换句话说,闭包可以在其函数体内引用定义在函数外部的变量。这些变量称为自由变量。

/*
bitwiseIncrementer 是一个闭包函数,它返回一个函数,该函数每次调用时都会返回一个自增后的整数。
因为返回的函数引用了一个方法的变量 所以这个变量不会s被垃圾回收 所以这个函数可以一直使用
*/
func increment() func() int {
	a := 0
	f := func() int {
		a++
		return a
	}
	return f
}

main.go

	ff := increment()
	fmt.Println(ff)
	fmt.Println(&ff)
	v1 := ff()
	fmt.Println(v1)
	v2 := ff()
	fmt.Println(v2)
	又调用了一个新的闭包函数 创建了新的内存空间 所以这个i也是新生成额变量
	ff2 := increment()
	fmt.Println(ff2())

猜猜会输出什么
在这里插入图片描述
流程是 :第一个函数返回了一个函数,而这个函数是闭包函数,正常来说函数使用后就被gc垃圾回收,但是由于返回的这个函数引用了这个变量,所以不会被gc,所以多次调用,自增还是改变的这个数据,而有再次调用这个代码得到一个自增函数时候,这个代码块的内容又是重新建立在新内存空间的,其中引用的a 又是新开辟的内存 所以又是1

那么输出这俩个自增函数

fmt.Println(ff)
	fmt.Println(ff2)

发现这俩个变量 不一样
在这里插入图片描述

函数类型深入

细心的可以发现,感觉函数和指针一样,装的数据是地址,指针装的是变量地址,而函数装的就是代码快的地址
由此可以再次测试

	// 定义函数变量
	funcdemo1 := func() {
		fmt.Println("funcdemo1")
	}
	funcdemo2 := func() {
		fmt.Println("funcdemo2")
	}

	// 打印函数的指针地址
	fmt.Printf("函数代码块的地址 %p\n", funcdemo1)
	fmt.Printf("函数的地址 %p\n", &funcdemo1)

输出
在这里插入图片描述
可以证实,函数的值就是代码块的地址,而也可以对这个函数取地址
感觉和指针一样,指针也是装载地址的数据类型,也可以对指针取地址

	funcdemo1 := func() {
		fmt.Println("funcdemo1")
	}
	funcdemo2 := func() {
		fmt.Println("funcdemo2")
	}
	a := 12
	P := &a
		fmt.Println("对地址取指针的取值操作", *&a)
	fmt.Println("指针的数值", P)
	fmt.Println("指针的地址", &P)

	// 打印函数的指针地址
	fmt.Printf("函数代码块的地址 %p\n", funcdemo1)
	fmt.Printf("函数的地址 %p\n", &funcdemo1)

在这里插入图片描述
既然知道了这个底层,那么就可以实现俩个函数交换函数体

func main() {
	// 定义函数变量
	funcdemo1 := func() {
		fmt.Println("funcdemo1")
	}
	funcdemo2 := func() {
		fmt.Println("funcdemo2")
	}

	swapfunc(&funcdemo1, &funcdemo2)

	// 现在 funcdemo1 应该调用原来 funcdemo2 的内容
	funcdemo1() // 现在会调用 funcdemo2
}
func swapfunc(a, b *func()) {
	//调用传递的参数

	(*a)() //先对指针取值得到函数体然后执行
	fmt.Printf("之前的地址: a: %p, b: %p\n", *a, *b)
	*a, *b = *b, *a // 交换引用
	fmt.Printf("之后的地址: a: %p, b: %p\n", *a, *b)
}

确实是交换了函数体,这点和二重指针一样的,通过指针地址改变指针中装的地址,这里通过函数地址改变函数中装的函数体地址
在这里插入图片描述
而这样就不行

func swapfunc(a, b func()) {
	//调用传递的参数

	(a)()
	fmt.Printf("之前代码的地址: a: %p, b: %p\n", a, b)
	a, b = b, a // 交换引用
	fmt.Printf("之后的地址: a: %p, b: %p\n", a, b)
}

传递函数作为参数是,是值拷贝,传递的只是把形参作为副本,形参值是函数体的值,但是形参的地址不是传递的哪个函数地址,所以修改存储的函数体原来的函数不会改变,改变的只是形参
同理如果想要改变指针保存的地址,那么也是需要参数传递指针的地址

细节,如果是在最外部申明的函数,无法取地址
这样就会报错

func main() {
	//函数本身就是引用类型 无需地址符号即可提取地址
	fmt.Printf("函数的地址%T\n", funcdemo1)
	fmt.Printf("函数的地址%p\n", &funcdemo1)
	swapfunc(funcdemo1, funcdemo2)
	funcdemo1()

}
func funcdemo1() {
	fmt.Println("funcdemo1")
}
func funcdemo2() {
	fmt.Println("funcdemo2")
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

服务器部署与DDOS攻防

知识点&#xff1a;DHPC与Web服务部署&#xff0c;DHCPig攻防&#xff0c;SYN Flooding测试 DHCP&#xff1a; 能够自动为客户机获取IP等参数 DHCP是动态主机配置协议&#xff08;Dynamic Host Configuration Protocol&#xff09;的缩写&#xff0c;它是一种网络管理协议&am…

华三交换机的软件版本升级操作

升级操作很常见&#xff0c;掌握方法是关键 实验环境&#xff1a;1台华三S6520-EI交换机&#xff0c;版本从2432P03升级成2432P05。 整体思路&#xff1a; 1.先查验软件版本 2.官网下载对于设备型号的软件版本 3.配置交换机地址使得与电脑进行通信&#xff0c;使用TFTP/FTP工…

UFS协议—新手快速入门(五)【11-13】

目录 十一、逻辑单元&#xff08;Logical Unit, LU&#xff09; 1、概念 2、UFS逻辑单元的独立特性 3、Well known LU &#xff08;1&#xff09;Boot Logical Units&#xff08;BOOT LUs&#xff09; &#xff08;2&#xff09;RPMB&#xff08;Replay Protected Memory…

27. 高级特性(下)

目录 一、为了类型安全和抽象而使用 newtype 模式二、使用类型别名创建类型同义词2.1 使用type关键赋予现有类型一个别名2.2 减少重复2.3 与Result<T, E>结合使用2.4 从不返回的 never type 三、高级函数和闭包3.1 函数指针3.2 返回闭包 四、宏4.1 宏和函数的区别4.2 mac…

ElasticSearch8.X查询DSL语法案例进阶实战

什么是Query DSL Query DSL主要由两部分组成&#xff1a;查询和过滤。 查询部分&#xff1a;用于指定搜索条件和匹配规则。例如&#xff0c;可以使用match查询进行全文检索&#xff0c;term查询进行精确匹配&#xff0c;range查询进行范围匹配等。过滤部分&#xff1a;用于对查…

【八股系列】探索响应式布局的奥秘:关键技术与实战代码示例

&#x1f389; 博客主页&#xff1a;【剑九 六千里-CSDN博客】 &#x1f3a8; 上一篇文章&#xff1a;【为什么组件中的 data 必须是一个函数&#xff0c;然后 return 一个对象&#xff0c;而 new Vue 实例里&#xff0c;data 可以直接是一个对象&#xff1f;】 &#x1f3a0; …

【python】eval函数

1.eval函数的语法及用法 &#xff08;1&#xff09;语法&#xff1a;eval(expression) 参数说明&#xff1a; expression&#xff1a;必须为字符串表达式&#xff0c;可为算法&#xff0c;也可为input函数等。 说明&#xff1a;表达式必需是字符串&#xff0c;否则会报错&a…

(2024,稀疏高秩适配器(SHiRA),适配器快速切换和多适配器融合,稀疏掩码,稀疏高秩且正交的适配器,移动场景部署)

Sparse High Rank Adapters 公和众与号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0. 摘要 1. 介绍 2. 背景&#xff1a;LoRA 的边缘部署挑战 3. 提议的方法 3.1 稀疏高秩适配器&#xff0…

共享资源,共谋协同发展新机遇

树莓集团在资源共享与协同发展方面取得了显著成效。通过优化资源配置、构建产业服务平台、校企合作育人、推动产业转型升级以及主导产业园运营与建设等举措&#xff0c;树莓集团不仅促进了数字产业的快速发展&#xff0c;也为各数字产业园区和合作企业提供了更多的发展机会和合…

Compiled blocks

目录 一&#xff0c;编译块 Compiled Block 二&#xff0c;编译循环 三&#xff0c;调用编译块 Invoke 四&#xff0c;Tips and notes 一&#xff0c;编译块 Compiled Block 在几何体网络内&#xff0c;可将网络的一部分放入编译块&#xff08;compiled block&#xff09;内…

天润融通:AI赋能客户体验,推动企业收入和业绩增长

“客户体验已经成为全球企业差异化的关键。人工智能与数据分析等创新技术正在加速推动企业在客户体验计划中取得成功&#xff0c;以保持领先地位”。Customer Insights & Analysis 研究经理Craig Simpson说道。 客户体验 (CX&#xff0c;Customer Experience) 是客户在与企…

基于PHP+MySql的留言管理系统的设计与实现

功能概述 网页留言板管理系统&#xff0c;用户层面分为普通用户和管理员&#xff0c;并设权限&#xff08;即后台留言管理系统普通用户不能访问&#xff0c;别人的留言自己不可以修改删除&#xff0c;未登录不能使用留言功能&#xff09;&#xff0c;功能包括用户登录注册、留…

鸿蒙开发HarmonyOS NEXT(一)

最近总听见大家讨论鸿蒙&#xff0c;前端转型的好方向&#xff1f;先入门学习下 目前官方版本和文档持续更新中 一、开发环境 提示&#xff1a;要占用的空间比较多&#xff0c;建议安装在剩余空间多的盘 1、下载&#xff1a;官网最新工具 - 下载中心 - 华为开发者联盟 (huaw…

【Mac】王国保卫战:起源 for mac(塔防策略游戏)游戏介绍和安装教程

游戏介绍 《王国保卫战&#xff1a;起源》&#xff08;Kingdom: Origins&#xff09;是一款策略塔防游戏&#xff0c;其核心玩法融合了塔防、策略管理和资源管理元素。游戏的主要目标是在一个开放的像素化世界中建立和管理自己的王国&#xff0c;并抵御夜晚来袭的怪物入侵。 …

c语言学习记录(十)———函数

文章目录 前言一、函数的基本用法二、函数的参数传递1.基本方式2 数组在函数中的传参 前言 一个学习C语言的小白~ 有问题评论区或私信指出~ 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、函数的基本用法 函数是一个完成特定功能的代码模块&…

使用Dropout大幅优化PyTorch模型,实现图像识别

大家好&#xff0c;在机器学习模型中&#xff0c;如果模型的参数太多&#xff0c;而训练样本又太少&#xff0c;训练出来的模型很容易产生过拟合的现象。在训练神经网络时&#xff0c;过拟合具体表现在模型训练数据损失函数较小&#xff0c;预测准确率较高&#xff0c;但是在测…

gc.log中 CMS-concurrent-abortable-preclean

问题 在gc日志中看到 2024-06-26T16:16:07.5040800: 64690272.666: [CMS-concurrent-abortable-preclean-start]CMS: abort preclean due to time 2024-06-26T16:16:12.5530800: 64690277.716: [CMS-concurrent-abortable-preclean: 1.052/5.049 secs] [Times: user1.33 sys0…

点云处理实战 PCL求解点云表面曲率

目录 一、什么是曲率 二、曲率计算过程 三、pcl 求解点云局部曲率 四、思考?为何曲率计算会使用协方差矩阵? 五、推荐阅读 一、什么是曲率 曲率是几何学中用来描述曲线或曲面形状变化的一个量。它反映了曲线或曲面的弯曲程度。在不同的上下文中,曲率的定义和计算方式有…

C语言 指针——向函数传递字符串

目录 向函数传递字符串 计算实际字符个数 用字符数组编程实现字符串复制 用字符指针编程实现字符串复制 关于程序的效率的几点建议 向函数传递字符串 向函数传递字符串时  既可用 字符数组 作函数参数  也可用 字符指针 作函数参数 Simulating Call by reference &…

第二期书生·浦语大模型实战营优秀项目一览

书生浦语社区于 2023 年年底正式推出了书生浦语大模型实战营系列活动&#xff0c;至今已有两期五批次同学参加大模型学习、实战&#xff0c;线上课程累计学习超过 10 万人次。 实战营特设项目实践环节&#xff0c;提供 A100 算力支持&#xff0c;鼓励学员动手开发。第 2 期实战…