Golang笔记——协程同步

news2025/1/18 4:06:33

大家好,这里是Good Note,关注 公主号:Goodnote,专栏文章私信限时Free。本文详细介绍Golang的协程同步的实现和应用场景。

在这里插入图片描述

文章目录

      • 协程同步是什么?
      • 为什么需要协程同步?
      • 常见的协程同步机制
        • 互斥锁(Mutex)
        • 读写锁(RWMutex)
        • 等待组(WaitGroup)
        • 通道(Channel)
        • 原子操作(Atomic Operations)
      • 典型的协程同步场景
      • 对比:`channel` 和 `sync.Mutex`, `sync.WaitGroup`
      • 互斥锁控制并发顺序
        • 使用互斥锁实现Goroutine 控制顺序
        • 解释
    • 历史文章
      • MySQL数据库
      • Redis
      • Golang

协程同步是什么?

协程同步(Goroutine synchronization)是指在多个 Goroutine 并发执行时,通过某些机制来协调它们之间的执行顺序、共享数据访问和资源管理,从而避免数据竞争(race condition)、死锁、资源冲突等问题。同步的目的是确保在并发程序中,多个 Goroutine 可以正确地共享数据、按预期的顺序执行任务,并且避免因为并发操作导致的不可预测行为。

为什么需要协程同步?

在 Go 语言中,Goroutine 是一种轻量级的线程,多个 Goroutine 可能并发执行并共享资源。在并发执行时,如果没有适当的同步机制,多个 Goroutine 可能会同时访问共享资源,导致数据不一致、程序崩溃或其他并发问题。因此,需要同步机制来确保:

  1. 共享资源的正确访问:避免多个 Goroutine 同时修改同一资源(如变量、数据结构等),从而导致数据竞态。
  2. 执行顺序的控制:确保 Goroutine 在特定顺序下执行,满足某些逻辑条件(如等待某些任务完成)。
  3. 任务完成的等待:在某些场景下,需要等待多个并发任务完成后再继续后续操作。

常见的协程同步机制

互斥锁(Mutex)

互斥锁(sync.Mutex)用于保护共享资源,确保同一时刻只有一个 Goroutine 能访问临界区(共享资源)。在锁的保护下,其他 Goroutine 必须等待,直到当前 Goroutine 完成对资源的操作并释放锁。

  • 优点

    • 简单易用,适合保护临界区。
    • 避免多个 Goroutine 同时读写共享资源时的数据竞态。
  • 缺点

    • 锁的粒度较粗,不适合高并发场景。
    • 可能会导致死锁,尤其是当锁的使用不当时。
读写锁(RWMutex)

sync.RWMutex 是一种更细粒度的锁,它允许多个 Goroutine 同时读取共享资源,但写操作时会阻止所有其他的读取和写入操作。适用于读多写少的场景。

  • 优点

    • 适合读多写少的场景,可以允许多个 Goroutine 同时读取共享数据。
    • 减少了锁竞争,提高了并发性能。
  • 缺点

    • 写操作仍然是独占的,不适用于频繁写操作的场景。
等待组(WaitGroup)

sync.WaitGroup 用于等待一组 Goroutine 完成任务。它提供了 AddDoneWait 方法,用来协调多个 Goroutine 的执行顺序。

  • 优点

    • 非常适合等待多个并发任务的完成。
    • 通过 Add 增加等待的任务数,通过 Done 表示任务完成,Wait 阻塞当前 Goroutine 直到所有任务完成。
  • 缺点

    • 只能用于同步“任务完成”,不能用于同步临界区的访问。

WaitGroup源码如下:


// A WaitGroup waits for a collection of goroutines to finish.
// The main goroutine calls Add to set the number of
// goroutines to wait for. Then each of the goroutines
// runs and calls Done when finished. At the same time,
// Wait can be used to block until all goroutines have finished.
// A WaitGroup must not be copied after first use.
type WaitGroup struct {
   noCopy noCopy
   // 64-bit value: high 32 bits are counter, low 32 bits are waiter count.
   // 64-bit atomic operations require 64-bit alignment, but 32-bit
   // compilers do not ensure it. So we allocate 12 bytes and then use
   // the aligned 8 bytes in them as state, and the other 4 as storage
   // for the sema.
   state1 [3]uint32
}
  • 一个WaitGroup等待多个goroutine执行完成,main的goroutine可以调用Add()方法设置需要等待的goroutine数量,之后每一个goroutine在运行结束时调用Done(),在这段时间内,我们可以使用Wait()阻塞main的goroutine直到所有的goroutine都执行完成。
  • WaitGroup不能进行复制操作【struct里面有noCopy类型,禁止做值拷贝,只能通过指针来传递】,在函数中使用 WaitGroup 时,需要传递它的指针,即 *sync.WaitGroup
通道(Channel)

Channel 是 Go 的核心特性之一,它不仅用于 Goroutine 间通信,还能通过阻塞机制隐式地实现同步。通过发送和接收操作,Channel 可以协调多个 Goroutine 的执行,确保它们按照特定顺序进行。

  • 优点

    • 通过 Channel 传递数据本身就会进行同步,使用非常灵活。
    • 可以避免使用显式的锁(如 sync.Mutex)来控制并发。
  • 缺点

    • 不适合所有场景,特别是需要复杂同步时,可能需要更多的设计。
    • 需要注意死锁和缓冲区的大小等问题。
原子操作(Atomic Operations)

sync/atomic 包提供了一些原子操作,用于在多个 Goroutine 之间同步访问单个变量。这些操作不需要使用锁,适用于简单的计数器、标志位等场景。

  • 优点

    • 对单一变量的原子操作非常高效。
    • 适用于计数器、标志位等简单的同步操作。
  • 缺点

    • 只适用于单个变量,不适合复杂的数据结构。
    • 操作较为低级,可能需要更多的代码来管理并发逻辑。

典型的协程同步场景

  1. 保护共享数据:当多个 Goroutine 需要读写共享数据时,可以使用 MutexRWMutex 来保护数据的访问。
package main

import (
   "fmt"
   "sync"
   "time"
)

func main() {
   var mu sync.Mutex
   var counter int
   for i := 0; i < 1000; i++ {
   	go func() {
   		mu.Lock()
   		counter++
   		mu.Unlock()
   	}()
   }
   time.Sleep(time.Second * 3)
   fmt.Println(counter)
}

说明:time.Sleep可以使用WaitGroup进行替代。

  1. 等待多个任务完成:使用 WaitGroup 来等待多个并发任务完成后再继续执行后续操作。
package main

import (
	"fmt"
	"sync"
)

func main() {
	var wg sync.WaitGroup
	for i := 0; i < 5; i++ {
		wg.Add(1)
		go func(i int) {
			defer wg.Done()
			fmt.Println(i)
		}(i)
	}
	wg.Wait()
}

说明:time.Sleep可以使用WaitGroup进行替代。

  1. 协调 Goroutine 执行顺序:使用 Channel 来确保 Goroutine 按照特定顺序执行。
package main

import (
	"fmt"
	"time"
)

func main() {
	// 创建多个无缓冲的 Channel,用来控制 Goroutine 的顺序
	step1 := make(chan struct{})
	step2 := make(chan struct{})
	step3 := make(chan struct{})

	// 定义第一个 Goroutine
	go func() {
		fmt.Println("Goroutine 1: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 1: Done")
		// 通知 Goroutine 2 可以开始
		close(step1)
	}()

	// 定义第二个 Goroutine
	go func() {
		// 等待 Goroutine 1 完成
		<-step1
		fmt.Println("Goroutine 2: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 2: Done")
		// 通知 Goroutine 3 可以开始
		close(step2)
	}()

	// 定义第三个 Goroutine
	go func() {
		// 等待 Goroutine 2 完成
		<-step2
		fmt.Println("Goroutine 3: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 3: Done")
		// 通知主线程结束
		close(step3)
	}()

	// 等待 Goroutine 3 完成
	<-step3
	fmt.Println("All Goroutines Finished!")
}
package main

import (
	"fmt"
	"time"
)

func main() {
	// 创建多个无缓冲的 Channel,用来控制 Goroutine 的顺序
	step1 := make(chan struct{})
	step2 := make(chan struct{})
	step3 := make(chan struct{})

	// 定义第一个 Goroutine
	go func() {
		fmt.Println("Goroutine 1: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 1: Done")
		// 通知 Goroutine 2 可以开始
		step1 <- struct{}{}
	}()

	// 定义第二个 Goroutine
	go func() {
		// 等待 Goroutine 1 完成
		<-step1
		fmt.Println("Goroutine 2: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 2: Done")
		// 通知 Goroutine 3 可以开始
		step2 <- struct{}{}
	}()

	// 定义第三个 Goroutine
	go func() {
		// 等待 Goroutine 2 完成
		<-step2
		fmt.Println("Goroutine 3: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 3: Done")
		// 通知主线程结束
		step3 <- struct{}{}
	}()

	// 等待 Goroutine 3 完成
	<-step3
	fmt.Println("All Goroutines Finished!")
}

这两个代码的主要区别在于如何实现 Goroutine 之间的同步信号传递:一个使用 close() 关闭通道,另一个使用 <-chan 来发送信号。

  1. 使用 close 通道

    • close(stepX) 通知接收方,通道不再发送任何数据。这意味着接收方在收到数据后,可以认为没有更多的工作需要处理。
    • 这种方式适用于需要明确表示“结束”或“没有更多数据”的场景。
  2. 使用 <-stepX 信号传递

    • stepX <- struct{}{} 用来传递一个信号,通常是通过发送一个空的结构体。接收方通过 <-stepX 等待信号,表示前一个 Goroutine 已经完成,可以继续执行。
    • 这种方式更常见用于同步 Goroutine 之间的顺序执行,它不表示“结束”,只是简单的通知和同步。

在 Go 中,无缓冲通道chan struct{})通常用于同步信号传递。代码中使用的 step1, step2, 和 step3 是用于控制 Goroutine 执行顺序的信号通道。对于这种情形,关闭通道并不是必要的,因为:

  1. 无缓冲通道的用途:在你的代码中,通道是用于 Goroutine 之间的同步,而不是用来传输数据或产生多次通信。

  2. 通道关闭的场景:关闭通道通常用于发送者完成发送所有数据并且没有更多数据要发送时,或者用于接收方识别通道的结束。在这个例子中,主线程只需要等待第三个Goroutine 发出的信号,而不需要读取或等待更多的值。因此,不关闭通道不会导致问题

  3. 等待通道信号的场景:主线程通过 <-step3 来等待所有 Goroutine 完成。当最后一个 Goroutine 通过 step3 <- struct{}{} 完成时,主线程就能结束。无缓冲通道不需要关闭,也不会导致死锁或资源泄漏。

总结:

  • 通常在有缓冲的通道多个接收者的情况下,关闭通道的意义更大,因为接收者可能需要知道什么时候没有更多的数据,或者什么时候发送者不再发送数据。这种channel如果未关闭,可能导致它们在垃圾回收机制中未被及时回收。
  • 在当前的场景下(无缓冲通道、每个通道仅用于同步信号),没有关闭通道也不会影响程序的正确性。Go 的垃圾回收机制会自动处理那些不再使用的对象和数据结构,包括通道。所以即使没有显式关闭通道,程序结束时,未关闭的通道也会被垃圾回收。

当然,也可以在使用 <-chan 来发送信号后强制关闭通道,如下:

package main

import (
	"fmt"
	"time"
)

func main() {
	// 创建多个无缓冲的 Channel,用来控制 Goroutine 的顺序
	step1 := make(chan struct{})
	step2 := make(chan struct{})
	step3 := make(chan struct{})

	// 定义第一个 Goroutine
	go func() {
		fmt.Println("Goroutine 1: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 1: Done")
		// 通知 Goroutine 2 可以开始
		step1 <- struct{}{}
		// 关闭 step1 通道,表示没有更多信号
		close(step1)
	}()

	// 定义第二个 Goroutine
	go func() {
		// 等待 Goroutine 1 完成
		<-step1
		fmt.Println("Goroutine 2: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 2: Done")
		// 通知 Goroutine 3 可以开始
		step2 <- struct{}{}
		// 关闭 step2 通道,表示没有更多信号
		close(step2)
	}()

	// 定义第三个 Goroutine
	go func() {
		// 等待 Goroutine 2 完成
		<-step2
		fmt.Println("Goroutine 3: Start")
		time.Sleep(1 * time.Second) // 模拟工作
		fmt.Println("Goroutine 3: Done")
		// 通知主线程结束
		step3 <- struct{}{}
		// 关闭 step3 通道,表示没有更多信号
		close(step3)
	}()

	// 等待 Goroutine 3 完成
	<-step3
	fmt.Println("All Goroutines Finished!")
}

channel同步执行Goroutine请参考:【todo】

  1. 原子操作:使用 atomic 来实现对单个变量的原子操作,如计数器的增加,但是是无序的。
package main

import (
	"fmt"
	"sync"
	"sync/atomic"
)

func main() {
	var count int32
	var wg sync.WaitGroup

	// 创建10个 Goroutine 来增加计数器
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go func(i int) {
			defer wg.Done()
			// 使用 atomic 对 count 进行原子增加
			atomic.AddInt32(&count, 1)
			// 打印当前的数字
			fmt.Println(i)
		}(i)
	}

	// 等待所有 Goroutine 完成
	wg.Wait()

	// 输出最终的 count 值
	fmt.Println("Final count:", count)
}

对比:channelsync.Mutex, sync.WaitGroup

  • channel:更灵活,能够传递数据并同步控制执行顺序。通常用于需要显式控制执行顺序的场景,比如一个任务完成后通知另一个任务。
  • sync.Mutexsync.RWMutex:主要用于同步对共享资源的访问,无法直接控制 Goroutine 的执行顺序。
  • sync.WaitGroup:用于等待多个 Goroutine 完成,可以确保所有 Goroutine 都完成后再执行下一步,但它不控制 Goroutine 的执行顺序。

使用channel可以控制执行顺序,当然也可只使用 sync.Mutexsync.RWMutexsync.WaitGroup 来控制 Goroutine 的执行顺序,只不过没有channel那么优雅,参考下节。

互斥锁控制并发顺序

如果单独使用 sync.Mutexsync.RWMutexsync.WaitGroup 来实现 Goroutine 顺序打印 0 到 9,需要巧妙地利用 sync.Mutexsync.RWMutex 来确保 Goroutine 按顺序执行。

使用互斥锁实现Goroutine 控制顺序

我们可以通过 sync.Mutex 来实现一个基本的锁机制,确保每次只有一个 Goroutine 在执行,并按顺序打印数字。

package main

import (
	"fmt"
	"sync"
)

func main() {
	var mu sync.Mutex
	// 	var mu sync.RWMutex
	var wg sync.WaitGroup
	counter := 0

	// 使用 Mutex 来控制顺序打印
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go func(i int) {
			defer wg.Done()

			// 控制顺序打印
			mu.Lock()
			// 这里使用 counter 来确保按顺序执行
			for counter != i {
				mu.Unlock()
				mu.Lock()
			}

			// 打印当前数字
			fmt.Println(i)
			counter++
			mu.Unlock()
		}(i)
	}

	wg.Wait() // 等待所有 Goroutine 执行完毕
}
解释
  • 我们使用 sync.Mutex 来保护共享变量 counter,确保每个 Goroutine 在它轮到执行时才会打印。
  • counter 用于跟踪已经执行的顺序,mu.Lock()mu.Unlock() 确保只有一个 Goroutine 可以进入临界区。
  • for counter != i 的检查保证每个 Goroutine 在它的数字到达时才开始执行。

历史文章

MySQL数据库

MySQL数据库

Redis

Redis数据库笔记合集

Golang

  1. Golang笔记——语言基础知识
  2. Golang笔记——切片与数组
  3. Golang笔记——hashmap
  4. Golang笔记——rune和byte
  5. Golang笔记——channel
  6. Golang笔记——Interface类型
  7. Golang笔记——数组、Slice、Map、Channel的并发安全性

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

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

相关文章

Visual Studio Community 2022(VS2022)安装方法

废话不多说直接上图&#xff1a; 直接上步骤&#xff1a; 1&#xff0c;首先可以下载安装一个Visual Studio安装器&#xff0c;叫做Visual Studio installer。这个安装文件很小&#xff0c;很快就安装完成了。 2&#xff0c;打开Visual Studio installer 小软件 3&#xff0c…

目标检测新视野 | YOLO、SSD与Faster R-CNN三大目标检测模型深度对比分析

目录 引言 YOLO系列 网络结构 多尺度检测 损失函数 关键特性 SSD 锚框设计 损失函数 关键特性 Faster R-CNN 区域建议网络&#xff08;RPN&#xff09; 两阶段检测器 损失函数 差异分析 共同特点 基于深度学习 目标框预测 损失函数优化 支持多类别检测 应…

mac intel芯片下载安卓模拟器

一、调研 目前主流两个模拟器&#xff1a; 雷神模拟器 不支持macosmumu模拟器pro版 不支持macos intel芯片 搜索到mumu的Q&A中有 “Intel芯片Mac如何安装MuMu&#xff1f;” q&a&#x1f517;&#xff1a;https://mumu.163.com/mac/faq/install-on-intel-mac.html 提…

发送dubbo接口

史上最强&#xff0c;Jmeter接口测试-dubbo接口实战&#xff08;超级详细&#xff09;_jmeter调用dubbo接口-CSDN博客 干货分享&#xff1a;Dubbo接口及测试总结~ 谁说dubbo接口只能Java调用&#xff0c;我用Python也能轻松搞定 telnet xxx.xxx.xxx.xxx 端口号 再回车显示dub…

Leetcode 91. 解码方法 动态规划

原题链接&#xff1a;Leetcode 91. 解码方法 自己写的代码&#xff1a; class Solution { public:int numDecodings(string s) {int ns.size();vector<int> dp(n,1);if(s[n-1]0) dp[n-1]0;for(int in-2;i>0;i--){if(s[i]!0){string ts.substr(i,2);int tmpatoi(t.c…

SpringBoot源码解析(七):应用上下文结构体系

SpringBoot源码系列文章 SpringBoot源码解析(一)&#xff1a;SpringApplication构造方法 SpringBoot源码解析(二)&#xff1a;引导上下文DefaultBootstrapContext SpringBoot源码解析(三)&#xff1a;启动开始阶段 SpringBoot源码解析(四)&#xff1a;解析应用参数args Sp…

SCSSA-BiLSTM基于改进麻雀搜索算法优化双向长短期记忆网络多特征分类预测Matlab实现

SCSSA-BiLSTM基于改进麻雀搜索算法优化双向长短期记忆网络多特征分类预测Matlab实现 目录 SCSSA-BiLSTM基于改进麻雀搜索算法优化双向长短期记忆网络多特征分类预测Matlab实现分类效果基本描述程序设计参考资料 分类效果 基本描述 SCSSA-BiLSTM基于改进麻雀搜索算法优化双向长…

XML在线格式化 - 加菲工具

XML在线格式化 打开网站 加菲工具 选择“XML 在线格式化” 输入XML&#xff0c;点击左上角的“格式化”按钮 得到格式化后的结果

树莓派5--系统问题汇总

前言&#xff1a; 该文章是我在使用树莓派5时所遇到的问题以及解决方案&#xff0c;希望对遇到相同问题的能够有所帮助。我的树莓派系统版本为&#xff1a;Pi-OS-ROS_2024_09_29 注意&#xff1a;如果没有什么需求千万不要更新树莓派中任何软件或者系统&#xff0c;除非你真的…

C#学习笔记 --- 基础补充

1.operator 运算符重载&#xff1a;使自定义类可以当做操作数一样进行使用。规则自己定。 2.partial 分部类&#xff1a; 同名方法写在不同位置&#xff0c;可以当成一个类使用。 3.索引器&#xff1a;使自定义类可以像数组一样通过索引值 访问到对应的数据。 4.params 数…

【2024年华为OD机试】 (C卷,100分)- 免单统计(Java JS PythonC/C++)

一、问题描述 题目描述 华为商城举办了一个促销活动&#xff0c;如果某顾客是某一秒内最早时刻下单的顾客&#xff08;可能是多个人&#xff09;&#xff0c;则可以获取免单。 请你编程计算有多少顾客可以获取免单。 输入描述 输入为 n 行数据&#xff0c;每一行表示一位顾…

python中数据可视化库(Matplotlib)

python中数据可视化库&#xff08;Matplotlib&#xff09; 安装 Matplotlib基本使用绘图类型示例散点图 (Scatter Plot)柱状图 (Bar Chart)饼图 (Pie Chart)直方图 (Histogram) 自定义图表样式多面板图表 (Subplots)3D 图表 Matplotlib 是 Python 中一个非常流行的绘图库&#…

某国际大型超市电商销售数据分析和可视化

完整源码项目包获取→点击文章末尾名片&#xff01; 本作品将从人、货、场三个维度&#xff0c;即客户维度、产品维度、区域维度&#xff08;补充时间维度与其他维度&#xff09;对某国际大型超市的销售情况进行数据分析和可视化报告展示&#xff0c;从而为该超市在弄清用户消费…

DETR论文阅读

1. 动机 传统的目标检测任务需要大量的人工先验知识&#xff0c;例如预定义的先验anchor&#xff0c;NMS后处理策略等。这些人工先验知识引入了很多人为因素&#xff0c;且较难处理。如果能够端到端到直接生成目标检测结果&#xff0c;将会使问题变得很优雅。 2. 主要贡献 提…

工业视觉2-相机选型

工业视觉2-相机选型 一、按芯片类型二、按传感器结构特征三、按扫描方式四、按分辨率大小五、按输出信号六、按输出色彩接口类型 这张图片对工业相机的分类方式进行了总结&#xff0c;具体如下&#xff1a; 一、按芯片类型 CCD相机&#xff1a;采用电荷耦合器件&#xff08;CC…

《机器学习》——TF-IDF(关键词提取)

文章目录 TF-IDF简介TF-IDF应用场景TF-IDF模型模型参数主要参数 TF-IDF实例实例步骤导入数据和模块处理数据处理文章开头和分卷处理将各卷内容存储到数据帧jieba分词和去停用词处理 计算 TF-IDF 并找出核心关键词 TF-IDF简介 TF - IDF&#xff08;Term Frequency - Inverse Do…

LabVIEW与WPS文件格式的兼容性

LabVIEW 本身并不原生支持将文件直接保存为 WPS 格式&#xff08;如 WPS 文档或表格&#xff09;。然而&#xff0c;可以通过几种间接的方式实现这一目标&#xff0c;确保您能将 LabVIEW 中的数据或报告转换为 WPS 可兼容的格式。以下是几种常见的解决方案&#xff1a; ​ 导出…

CV 图像处理基础笔记大全(超全版哦~)!!!

一、图像的数字化表示 像素 数字图像由众多像素组成&#xff0c;是图像的基本构成单位。在灰度图像中&#xff0c;一个像素用一个数值表示其亮度&#xff0c;通常 8 位存储&#xff0c;取值范围 0 - 255&#xff0c;0 为纯黑&#xff0c;255 为纯白。例如&#xff0c;一幅简单的…

【JavaScript】比较运算符的运用、定义函数、if(){}...esle{} 语句

比较运算符 !><> < 自定义函数&#xff1a; function 函数名&#xff08;&#xff09;{ } 判断语句&#xff1a; if(判断){ }else if(判断){ 。。。。。。 }else{ } 代码示例&#xff1a; <!DOCTYPE html> <html> <head><meta charset&quo…

centos 7 Mysql服务

将此服务器配置为 MySQL 服务器&#xff0c;创建数据库为 hubeidatabase&#xff0c;将登录的root密码设置为Qwer1234。在库中创建表为 mytable&#xff0c;在表中创建 2 个用户&#xff0c;分别为&#xff08;xiaoming&#xff0c;2010-4-1&#xff0c;女&#xff0c;male&…