golang range for循环中如何正确的给goroutine传参

1.code example

 公共方法

func testDomain(ii string) {
	time.Sleep(time.Second * 4)
	fmt.Printf("pid: %d___point addr: %d___%s \n", GoID(), &ii, ii)
}
func GoID() int {
	var buf [64]byte
	n := runtime.Stack(buf[:], false)
	idField := strings.Fields(strings.TrimPrefix(string(buf[:n]), "goroutine "))[0]
	id, err := strconv.Atoi(idField)
	if err != nil {
		panic(fmt.Sprintf("cannot get goroutine id: %v", err))
	}
	return id
}

  

2. 错误示范

var a []string
for i := 1; i < 5; i++ {
a = append(a, fmt.Sprintf("%d", i))
}
	for _, i := range a {
		fmt.Printf("-----%s---\n", i)

		go func() {
			time.Sleep(time.Second * 4)
			testDomain(i)
		}()
	
	}

 打印发现i每次地址都是同一个

    协助每次先阻塞4秒

  4秒后 i的值是4, 这是协程中的方法testDomain开始工作,将i的值传给自己的形参

3. 正确示范

	for _, i := range a {
		fmt.Printf("-----%s---\n", i)
		go func(a string) {
			//time.Sleep(time.Second * 4)
			testDomain(a)
		}(i)
	}

 这种操作会先将i的值传递给形参a,i的变化不会对testDomain方法的执行产生影响

 

4. 完整代码

package main

import (
	"fmt"
	"runtime"
	"strconv"
	"strings"
	"time"
)

func main() {
	var a []string
	for i := 1; i < 5; i++ {
		a = append(a, fmt.Sprintf("%d", i))
	}
	for _, i := range a {
		fmt.Printf("-----%s---\n", i)
		go func(a string) {
			//time.Sleep(time.Second * 4)
			testDomain(a)
		}(i)

		go func() {
			time.Sleep(time.Second * 4)
			testDomain(i)
		}()
		fmt.Println(&i)
		time.Sleep(time.Second * 1)
	}

	time.Sleep(100 * time.Second)
}

func testDomain(ii string) {
	time.Sleep(time.Second * 4)
	fmt.Printf("pid: %d___point addr: %d___%s \n", GoID(), &ii, ii)
}

func GoID() int {
	var buf [64]byte
	n := runtime.Stack(buf[:], false)
	idField := strings.Fields(strings.TrimPrefix(string(buf[:n]), "goroutine "))[0]
	id, err := strconv.Atoi(idField)
	if err != nil {
		panic(fmt.Sprintf("cannot get goroutine id: %v", err))
	}
	return id
}

  

 

  

转载于:https://www.cnblogs.com/waken-captain/p/10496454.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值