go语言切片作为函数参数的研究

slice作为函数参数是值传递

golang中的切片slice底层通过数组实现,slice类似一个结构体,其中一个字段保存的是底层数组的地址,还有长度(len) 和 容量(cap)两个字段。

结构体作为函数参数时是值拷贝,同理,实际上slice作为函数参数时也是值拷贝,在函数中对slice的修改是通过slice中保存的地址对底层数组进行修改,所以函数外的silce看起来被改变了。

当需要对slice做插入和删除时,由于需要更改长度字段,值拷贝就不行了,需要传slice本身在内存中的地址。

以删除为例,说明这种情况。

不通过函数,直接删除slice中的一个元素

package main

import "fmt"

func main() {
    si := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}

    fmt.Printf("%v  len %d \n", si, len(si))

    si = append(si[:3],si[4:]...)

    fmt.Printf("%v  len %d \n", si, len(si))
}
// ---------输出结果-------------------
[1 2 3 4 5 6 7 8 9]  len 9
[1 2 3 5 6 7 8 9]  len 8
// --------成功删除了4  长度减1---------

通过函数,参数是slice变量

func main() {
    si := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}

    fmt.Printf("%v  len %d \n", si, len(si))

    test1(si)

    fmt.Printf("%v  len %d \n", si, len(si))
}

func test1(si []int) {
    si = append(si[:3], si[4:]...)
}

// -------------输出结果-------------------
[1 2 3 4 5 6 7 8 9]  len 9
[1 2 3 5 6 7 8 9 9]  len 9
// -----4虽然被删除了,但是长度不变,最后面还多了个9,可见不是想要的结果

通过函数,参数是slice变量的地址

func main() {
    si := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}

    fmt.Printf("%v  len %d \n", si, len(si))

    test(&si)

    fmt.Printf("%v  len %d \n", si, len(si))
}

func test(si *[]int) {
    (*si) = append((*si)[:3], (*si)[4:]...)
}

// ---------输出结果-------------------
[1 2 3 4 5 6 7 8 9]  len 9
[1 2 3 5 6 7 8 9]  len 8
// --------成功删除了4  长度减1---------

原文地址:https://www.cnblogs.com/endurance9/p/10347336.html

时间: 2024-10-28 19:23:19

go语言切片作为函数参数的研究的相关文章

c语言数组做函数参数退化为指针

我的系统是MAC OS 64位.根据自己的系统位数不同是有差异的.以下是我学习过程中遇到的问题 大侠略过... 有时候我们想通过一个函数对数组的元素进行操作,在sizelen函数中想想通过sizeof获得数组的长度. 想法是好的,不要以为你对C语言很了解了.其实数组在做函数参数是只是传了,一个指针.也就是数组 的首地址. 从结果可以看出,C编译器确实是这么做得.我的系统是64位,所以int *是占8个字节. 以上只是我学习过程中遇到的问题.大侠略过...

C语言 数组做函数参数退化为指针的技术推演

//数组做函数参数退化为指针的技术推演 #include<stdio.h> #include<stdlib.h> #include<string.h> //一维数组做函数参数退化为指针的技术推演 void printfA(char * strarr[3]); //计算机中,数组都是线性存储,二维数组元素也是一个个的排列的 //例如: 1,2,3,4,5,6,7,8,9 像这组数据 我们可以认为是一维数组 int a[9]={1,2,3,4,5,6,7,8,9}; //也

C/C++语言中的函数参数传参三种对比

学了很长时间C/C++有时指针方面还是有点乱. 希望大神发现如果下面有不对的地方请指出.我发现之所以我乱就是因为中文表述不准确的问题,比如 ,地址值和地址 #include <iostream> #include <string> using namespace std; void swap1(string* str1,string* str2){// 1.对象指针作为函数参数 //影响实参 cout<<&str1<<" "<

C语言 数组做函数参数不传数组个数的遍历方法

//数组做函数参数不传数组个数的遍历方法 #include<stdio.h> #include<stdlib.h> #include<string.h> void PrintfAK(char **pin){ int i = 0; //关键点:pin[i]!=NULL为终止条件 for (i = 0; pin[i]!=NULL; i++) { printf("%s\n", pin[i]); } } void main(){ //赋值数组最后一个元素是0

C语言中结构体参数变量的传递

c语言  结构体  函数  参数  传递 一.前言 本文中的程序实现对员工信息结构体字段赋值并打印出来的功能.该结构体的定义如下: // 员工信息结构体 typedef struct {        INT8       szEmployeeName[100];     // 员工姓名        UINT16  iEmployeeAge;                    // 员工年龄        UINT32  iEmployeeNo;                      

关于c语言不定参数的研究

一. 学习过程 编写程序如下: 编译连接并用debug加载,观察main函数的内容: Showchar函数的内容: 观察发现,main函数要传递两个参数‘a’和2,在汇编代码中是先将2赋给ax,再将ax入栈,然后将a赋给ax,将ax入栈.在showchar函数中,程序将sp赋给bp,再从bp+4处取出形参a赋给al,再将al中的值赋给b800:690h,然后再从bp+6处取出形参b赋给al,再将al中的值赋给b800:691h.可见main函数给showchar传递参数是把要传递的值赋给ax,再

C语言函数参数压栈顺序为何是从右到左?(从左向右的话,碰到printf的会陷入死循环)

上学期学习了汇编语言,并在操作系统实验中使用了汇编+C语言混合编程,中间也了解了一些C语言与汇编语言的对应关系. 由于汇编语言是底层的编程语言,各种函数参数都要直接控制栈进行存取,在混合编程中,要用汇编来调用C函数,当然就要知道参数的压栈情况了. 当知道C函数的参数压栈顺序是从右到左时,我觉得很奇怪,因为大多数情况下,人们的习惯是从左到右的,难不成设计者学咱们中国古代写字从右到左的习惯不成? 当时只是记下了这个规则而已,并没有去探究这其中的缘由,后来在实验中自己用汇编实现了printf和scan

C语言函数参数为空和void

C语言中函数参数的为空和用void标注他们的效果一样吗? #include <stdio.h>  2 //int add();  3 int add(void);  4 int main(void)  5 {  6     int i;  7     i = add(5 ,6);  8   9     printf("%d\n", i); 10     return 0; 11 } 12 int add(a ,b) 13 { 14     return a + b ; 15

C语言中函数参数为什么是由右往左入栈的?

先通过一个小程序来看一看: #include void foo(int x, int y, int z) { printf("x = %d at [%X]n", x, &x); printf("y = %d at [%X]n", y, &y); printf("z = %d at [%X]n", z, &z); } int main(int argc, char *argv[]) { foo(100, 200, 300);