Go并发编程终极指南:深入内核与工程实践
Go并发编程终极指南:深入内核与工程实践
Go并发编程终极指南:深入内核与工程实践
- Go并发编程终极指南:深入内核与工程实践
- 一、Goroutine调度器深度解构
- 1.1 调度器演进史
- 1.2 调度器源码级解析
- 1.3 调度器可视化诊断
- 二、Channel底层实现揭秘
- 2.1 Channel内存结构
- 2.2 Channel操作代价分析
- 2.3 Channel高级模式
- 三、并发安全与内存模型
- 3.1 Go内存模型核心要点
- 3.2 Happens-Before规则
- 3.3 竞态检测实战
- 四、高级并发模式
- 4.1 无锁编程实践
- 4.2 工作窃取调度器
- 五、生产环境最佳实践
- 5.1 Goroutine生命周期管理
- 5.2 资源池化模式
- 六、性能调优指南
- 6.1 性能分析工具链
- 6.2 并发瓶颈识别
- 七、未来与展望
一、Goroutine调度器深度解构
Goroutine是Go语言的一个核心特性。它是一种轻量级的线程,允许开发者在程序中并发执行多个任务。Goroutine的调度由Go运行时自动管理,能够高效地利用系统资源进行并发编程。与传统的线程相比,Goroutine的创建和销毁开销更低,因此可以在Go程序中轻松地启动成千上万的Goroutine来处理并发任务。
1.1 调度器演进史
- Go 1.0:简单的M:N调度
- Go 1.1:引入P处理器概念
- Go 1.14:实现真正的抢占式调度
- Go 1.20:调度器延迟优化
调度器性能对比:
下面这段代码是一个Go语言的基准测试(Benchmark):
// 调度基准测试
func BenchmarkScheduler(b *testing.B) {for i := 0; i < b.N; i++ {go func() { atomic.AddInt64(&count, 1) }()}
}
不同Go版本下的百万Goroutine创建耗时对比:
版本 | 耗时 | 内存占用 |
---|---|---|
1.12 | 1.2s | 2.1GB |
1.18 | 0.8s | 1.7GB |
1.20 | 0.5s | 1.3GB |
当然,这些数据只是一个大致的基准,实际性能还会受到很多因素的影响,例如Go版本、硬件配置、操作系统等。
这里只是为了突显go并发的性能优势,不理解也可以继续下面的学习。
1.2 调度器源码级解析
下面是一个简化的Go调度器源码示例:
// runtime/proc.go 关键结构体
type g struct {stack stack // 执行栈sched gobuf // 调度上下文atomicstatus uint32 // 状态机
}type p struct {runqhead uint32 // 本地队列头runqtail uint32 // 本地队列尾runq [256]guintptr
}// 调度循环核心逻辑
func schedule() {_g_ := getg()top:pp := _g_.m.p.ptr()// 1. 检查本地队列if gp, inheritTime, tryWakeP := findRunnable(); gp != nil {execute(gp, inheritTime)}// 2. 全局队列检查if gp := globrunqget(pp, 0); gp != nil {execute(gp, false)}// 3. 网络轮询器检查if netpollinited() && atomic.Load(&netpollWaiters) > 0 {if list := netpoll(0); !list.empty() {gp := list.pop()injectglist(&list)return gp, false}}// 4. 工作窃取if gp := runqsteal(pp, p2, stealRunNextG); gp != nil {return gp}
}
1.3 调度器可视化诊断
使用调度器追踪工具:
GODEBUG=schedtrace=1000,scheddetail=1 go run main.go
输出示例:
SCHED 0ms: gomaxprocs=8 idleprocs=5 threads=5 ...P0: status=1 schedtick=0 syscalltick=0 m=3 runqsize=3 ...P1: status=3 schedtick=1 syscalltick=0 m=-1 runqsize=0 ...
二、Channel底层实现揭秘
2.1 Channel内存结构
// runtime/chan.go
type hchan struct {qcount uint // 队列中元素个数dataqsiz uint // 环形队列大小buf unsafe.Pointer // 指向环形队列elemsize uint16closed uint32sendx uint // 发送索引recvx uint // 接收索引recvq waitq // 接收者队列sendq waitq // 发送者队列lock mutex
}
2.2 Channel操作代价分析
操作复杂度对比:
操作 | 无缓冲 | 有缓冲 | 关闭后 |
---|---|---|---|
非阻塞发送 | O(1) | O(1) | panic |
阻塞发送 | O(n) | O(1) | panic |
非阻塞接收 | O(1) | O(1) | O(1) |
批量接收 | O(n) | O(n) | O(n) |
性能测试用例:
func BenchmarkUnbufferedChan(b *testing.B) {ch := make(chan int)go func() {for i := 0; i < b.N; i++ {<-ch}}()for i := 0; i < b.N; i++ {ch <- i}
}
- make(chan int) 创建的是无缓冲channel
- 发送和接收操作会阻塞,直到另一端准备好
- 启动一个goroutine专门接收数据(消费者)
- 主goroutine循环发送数据(生产者)
- 测试 b.N 次发送和接收操作的耗时
2.3 Channel高级模式
模式1:双通道管道
func processingPipeline() {rawData := make(chan *Data, 100)processed := make(chan *Result, 50)// 阶段1:数据清洗go func() {defer close(rawData)for data := range fetchData() {rawData <- sanitize(data)}}()// 阶段2:并行处理const workers = 4var wg sync.WaitGroupwg.Add(workers)for i := 0; i < workers; i++ {go func() {defer wg.Done()for data := range rawData {processed <- complexProcessing(data)}}()}// 阶段3:结果收集go func() {wg.Wait()close(processed)}()
}
模式2:零内存拷贝通道
type BigData struct {payload [1 << 20]byte // 1MB数据
}func zeroCopyExample() {ch := make(chan *BigData, 10)// 生产者go func() {for {data := pool.Get().(*BigData)generateData(data)ch <- data}}()// 消费者for data := range ch {process(data)pool.Put(data) // 放回sync.Pool}
}
三、并发安全与内存模型
3.1 Go内存模型核心要点
var a, b intfunc f() {a = 1b = 2
}func g() {print(b)print(a)
}// 可能的输出组合:
// 0 0
// 2 1
// 0 1
3.2 Happens-Before规则
同步原语保证:
- Channel发送 happens before 对应接收完成
- Mutex解锁 happens before 后续加锁
- Once.Do保证初始化函数只执行一次
3.3 竞态检测实战
func raceExample() {var counter intvar wg sync.WaitGroupwg.Add(1000)for i := 0; i < 1000; i++ {go func() {defer wg.Done()counter++ // 数据竞争!}()}wg.Wait()fmt.Println(counter)
}
运行检测:
go run -race main.go
四、高级并发模式
4.1 无锁编程实践
CAS实现计数器:
type AtomicCounter struct {value int64
}func (c *AtomicCounter) Increment() {for {old := atomic.LoadInt64(&c.value)new := old + 1if atomic.CompareAndSwapInt64(&c.value, old, new) {return}}
}
4.2 工作窃取调度器
type WorkStealingScheduler struct {queues []deque.Dequedone chan struct{}
}func (s *WorkStealingScheduler) Start() {for i := 0; i < runtime.GOMAXPROCS(0); i++ {go s.worker(i)}
}func (s *WorkStealingScheduler) worker(id int) {for {task, ok := s.queues[id].PopFront()if !ok {// 尝试窃取其他队列的任务for i := 0; i < len(s.queues); i++ {if i == id {continue}if task, ok := s.queues[i].PopBack(); ok {execute(task)break}}} else {execute(task)}}
}
五、生产环境最佳实践
5.1 Goroutine生命周期管理
func managedGoroutine(ctx context.Context) {defer func() {if r := recover(); r != nil {log.Printf("goroutine panic: %v", r)}}()for {select {case <-ctx.Done():log.Println("优雅退出")returncase task := <-taskChan:if err := handle(task); err != nil {metrics.RecordError(err)}}}
}
5.2 资源池化模式
type ConnPool struct {pool chan net.Connfactory func() (net.Conn, error)
}func (p *ConnPool) Get() (net.Conn, error) {select {case conn := <-p.pool:return conn, nildefault:return p.factory()}
}func (p *ConnPool) Put(conn net.Conn) {select {case p.pool <- conn:default:conn.Close() // 池已满,直接关闭}
}
六、性能调优指南
6.1 性能分析工具链
# CPU分析
go test -cpuprofile=cpu.out
go tool pprof -http=:8080 cpu.out# 内存分析
go test -memprofile=mem.out
go tool pprof -http=:8081 mem.out# 阻塞分析
go test -blockprofile=block.out
go tool pprof block.out
6.2 并发瓶颈识别
常见性能陷阱:
- 过度使用全局锁
- Channel误用导致频繁阻塞
- Goroutine泄漏
- 虚假共享(False Sharing)
优化案例:
// 优化前
var counter int
var mu sync.Mutexfunc increment() {mu.Lock()counter++mu.Unlock()
}// 优化后:分片计数器
const shards = 64
type Counter struct {shard [shards]struct {value int64pad [128 - 8]byte // 缓存行填充}
}func (c *Counter) Inc(idx int) {atomic.AddInt64(&c.shard[idx%shards].value, 1)
}
七、未来与展望
-
泛型对并发的影响
type Future[T any] struct {result chan T }func Async[T any](fn func() T) *Future[T] {f := &Future[T]{result: make(chan T, 1)}go func() {f.result <- fn()}()return f }
-
Wasm并发模型
-
分布式Goroutine研究
深度阅读推荐:
- Go调度器源码分析
- The Go Memory Model
- Concurrency in Go 英文原版
实战项目建议:
- 实现高性能Web Crawler
- 构建实时交易引擎
- 开发分布式任务调度系统
(注:文中所有性能数据均在Intel i9-13900K/Go 1.21环境下测试得出,实际结果可能因环境不同有所差异)
相关文章:
Go并发编程终极指南:深入内核与工程实践
Go并发编程终极指南:深入内核与工程实践 Go并发编程终极指南:深入内核与工程实践 Go并发编程终极指南:深入内核与工程实践一、Goroutine调度器深度解构1.1 调度器演进史1.2 调度器源码级解析1.3 调度器可视化诊断 二、Channel底层实现揭秘2.1…...
Neo4j操作数据库(Cypher语法)
Neo4j数据库操作语法 使用的数据库版本 (终端查询) >neo4j --version 2025.03.0批量上传数据 UNWIND [{name: Alice, age: 30},{name: Bob, age: 25} ] AS person CREATE (p:Person) SET p.name = person.name, p.age = person.age RETURN p;查询结点总数 MATCH (n) RETU…...
DHCP之中继 Relay-snooping及配置命令
随着网络规模的不断扩大,网络设备不断增多,企业内不同的用户可能分布在不同的网段,一台 DHCP 服务器在正常情况下无法满足多个网段的地址分配需求。如果还需要通过 DHCP 服务器分配 IP 地址,则需要跨网段发送 DHCP 报文 DHCP Rel…...
小迪安全110-tp框架,版本缺陷,不安全写法,路由访问,利用链
入口文件 前端页面显示文件 就是这串代码让我们看到前端的笑脸图 不用入口文件我们要访问这个文件就要按照开发手册的url访问模式 那就是index.php/index/index/index 对应的就是模块,控制器,操作,函数名 如果想要创建新模块,和操…...
Vanna:用检索增强生成(RAG)技术革新自然语言转SQL
引言:为什么我们需要更智能的SQL生成? 在数据驱动的业务环境中,SQL 仍然是数据分析的核心工具。然而,编写正确的 SQL 查询需要专业知识,而大型语言模型(LLM)直接生成的 SQL 往往存在**幻觉&…...
大语言模型应用和训练(人工智能)
RAG(Retrieval Augmented Generation,检索增强生成) 定义:是一种将外部知识检索与语言模型生成能力相结合的技术。在传统的大语言模型中,模型的知识是在预训练阶段学到的,可能存在知识过时或不完整的问题。…...
NLP高频面试题(三十五)——LLaMA / ChatGLM / BLOOM的区别
一、LLaMA 训练数据 LLaMA由Meta开发,拥有多个参数规模的版本:7B、13B、33B和65B。其中,较小的7B和13B版本采用了约1万亿tokens进行训练,而更大的33B和65B版本使用了约1.4万亿tokens进行训练。 模型结构特点 LLaMA采用与GPT类似的causal decoder-only Transformer结构,…...
【Python Cookbook】字符串和文本(五):递归下降分析器
目录 案例 目录 案例 字符串和文本(一)1.使用多个界定符分割字符串2.字符串开头或结尾匹配3.用 Shell 通配符匹配字符串4.字符串匹配和搜索5.字符串搜索和替换字符串和文本(三)11.删除字符串中不需要的字符12.审查清理文本字符串1…...
专为 零基础初学者 设计的最简前端学习路线,聚焦核心内容,避免过度扩展,帮你快速入门并建立信心!
第一阶段:HTML CSS(2-3周) 目标:能写出静态网页,理解盒子模型和布局。 HTML基础 常用标签:<div>, <p>, <img>, <a>, <ul>, <form> 语义化标签:<head…...
大模型-爬虫prompt
爬虫怎么写prompt 以下基于deepseek r1 总结: 以下是为大模型设计的结构化Prompt模板,用于生成专业级网络爬虫Python脚本。此Prompt包含技术约束、反检测策略和数据处理要求,可根据具体需求调整参数: 爬虫脚本生成Prompt模板1 …...
PyTorch深度实践:基于累积最大值的注意力机制设计与性能优化
引言:注意力机制的创新与挑战 在自然语言处理和序列建模中,注意力机制(Attention)是提升模型性能的关键技术。传统基于 softmax 的注意力机制虽然成熟,但在计算效率和长序列建模中存在局限。本文将介绍一种创新的注意…...
编程bug001:off by one (差一错误)
为什么看似简单的编码错误可能造成大灾难? Off-by-One Error(简称OBOE),即由于边界条件处理不当,导致循环、计数或索引时多算一次或少算一次的错误。这是非常常见的编程bug类型,尤其在处理数组、字符串或范…...
JavaScript 中常见的鼠标事件及应用
JavaScript 中常见的鼠标事件及应用 在 JavaScript 中,鼠标事件是用户与网页进行交互的重要方式,通过监听这些事件,开发者可以实现各种交互效果,如点击、悬停、拖动等。 在 JavaScript 中,鼠标事件类型多样࿰…...
使用Expo框架开发APP——详细教程
在移动应用开发日益普及的今天,跨平台开发工具越来越受到开发者青睐。Expo 是基于 React Native 的一整套工具和服务,它能够大幅降低原生开发的门槛,让开发者只需关注业务逻辑和界面实现,而不用纠结于复杂的原生配置。本文将从零开…...
深入探究 Hive 中的 MAP 类型:特点、创建与应用
摘要 在大数据处理领域,Hive 作为一个基于 Hadoop 的数据仓库基础设施,提供了方便的数据存储和分析功能。Hive 中的 MAP 类型是一种强大的数据类型,它允许用户以键值对的形式存储和操作数据。本文将深入探讨 Hive 中 MAP 类型的特点,详细介绍如何创建含有 MAP 类型字段的表…...
前端开发工厂模式的优缺点是什么?
一、什么是工厂模式? 工厂模式属于创建型设计模式,核心思想是将对象的实例化过程封装到特定方法或类中,让客户端不需要直接通过new关键字创建对象。 举个例子:就像奶茶店不需要顾客自己调配饮品,而是通过"点单-…...
框架PasteForm实际开发案例,换个口味显示数据,支持echarts,只需要标记几个特性即可在管理端显示(2)
PasteForm框架的主要思想就是对Dto进行标记特性,然后管理端的页面就会以不一样的UI呈现 使用PasteForm框架开发,让你免去开发管理端的烦恼,你只需要专注于业务端和用户端! 在管理端中,如果说表格是基本的显示方式,那么图表chart就是一个锦上添花的体现! 如果一个项目拥…...
QEMU学习之路(5)— 从0到1构建Linux系统镜像
QEMU学习之路(5)— 从0到1构建Linux系统镜像 一、前言 参考:从内核到可启动镜像:0到1构建你的极简Linux系统 二、linux源码获取 安装编译依赖 sudo apt install -y build-essential libncurses-dev flex bison libssl-dev li…...
AI Agent设计模式一:Chain
概念 :线性任务流设计 ✅ 优点:逻辑清晰易调试,适合线性处理流程❌ 缺点:缺乏动态分支能力 from typing import TypedDictfrom langgraph.graph import StateGraph, END# 定义后续用到的一些变量 class CustomState(TypedDict):p…...
实操(进程状态,R/S/D/T/t/X/Z)Linux
1 R 状态并不直接代表进程在运行,而是该进程在运行队列中进行排队,由操作系统在内存维护的队列 #include <stdio.h> #include <unistd.h>int main() {while(1){printf("我在运行吗\n");sleep(1);}return 0; }查看状态(…...
T-SQL语言的自动化运维
T-SQL语言的自动化运维 引言 在现代IT环境中,自动化运维成为了提高效率、降低成本、提升稳定性的重要手段。数据库作为系统的重要组成部分,运维工作往往需要耗费大量的人力物力。T-SQL(Transact-SQL)作为Microsoft SQL Server的…...
Day06 分割编译与中断处理
文章目录 1. 例程harib03c(c源文件分割并整理makefile文件)2. 例程harib03c(用于描述段的信息)3. 例程harib03d(初始化PIC)4. 例程harib03e(中断处理程序) 1. 例程harib03cÿ…...
数字化三维实训室:无穿戴动作捕捉技术如何赋能体育与舞蹈
在高校体育与舞蹈教学中,精准的动作训练至关重要。传统训练方式依赖教练的肉眼观察与手动记录,存在效率低下、误差较大的情况。尤其在快速连续动作或复杂肢体形态的捕捉中,人工判读易受主观经验限制,难以实现标准化评估。面对传统…...
6. RabbitMQ 死信队列的详细操作编写
6. RabbitMQ 死信队列的详细操作编写 文章目录 6. RabbitMQ 死信队列的详细操作编写1. 死信的概念2. 消息 TTL 过期(触发死信队列)3. 队列超过队列的最大长度(触发死信队列)4. 消息被拒(触发死信队列)5. 最后: 1. 死信的概念 先从概念上解释上搞清楚这个定义&#…...
AI浪潮下,新手短视频制作的破局之道
AI浪潮下,新手短视频制作的破局之道 引言:短视频新时代,AI 带来的机遇与挑战 在当下这个信息飞速流转的时代,短视频已然成为了人们生活中不可或缺的一部分。无论是在通勤路上、午休间隙,还是茶余饭后,打开…...
合肥SMT贴片制造工艺全解析
内容概要 作为电子制造领域的核心工艺,SMT(表面贴装技术)在合肥地区电子产业链中占据重要地位。本解析以合肥本地化生产场景为基础,系统梳理从焊膏印刷到成品检测的全流程工艺框架。具体而言,制造流程涵盖四大核心阶段…...
ctfshow VIP题目限免 协议头信息泄露
根据提示是协议头信息泄露,那就我们抓个包,抓包才能看到请求体响应体里的协议头啊,抓包之后在响应包里发现了 flag...
【国产工具链发展,生态链分析,TSMaster VS Zcanpro的技术对比】
黎明篇:国产汽车测试工具链的崛起、差距与未来 副标题: 从跟随到超越,中国技术如何重塑全球研发体系 一、国产工具链的崛起逻辑 政策驱动:信创战略与供应链安全需求 国家“十四五”规划明确提出支持关键领域技术自主化࿰…...
Linux线程同步与互斥:【线程互斥】【线程同步】【线程池】
目录 一.线程互斥 1.1相关概念 1.2互斥量 为什么会出现负数?? 互斥量的接口 问题: 1.3互斥量实现原理探究 1.4互斥量封装 二.线程同步 2.1条件变量 2.2同步概念与竞态条件 2.3接口 2.4生产者消费者模型 优点 2.5基于BlockingQueue的…...
网络安全基础知识总结
什么是网络安全 采取必要措施,来防范对网络的攻击,侵入,干扰,破坏和非法使用,以及防范一些意外事故,使得网络处于稳定可靠运行的状态,保障网络数据的完整性、保密性、可用性的能力(CIA)。 举例…...
请求被中止: 未能创建 SSL/TLS 安全通道。
需要安装vs2019社区办,下载VisualStudioSetup.exe后,报无法从"https://aka,ms/vs/16/release/channel"下载通道清单错误,接着打开%temp%目录下的最新日志,发现日志里报: [27d4:000f][2025-04-04T21:15:43] …...
FPGA学习(四)——状态机重写LED流水灯并仿真
FPGA学习(四)——状态机重写LED流水灯并仿真 目录 FPGA学习(四)——状态机重写LED流水灯并仿真一、状态机编程思想1、状态机要素2、状态迁移图3、状态机写法 二、LED流水灯仿真实现1、代码实现2、modesim仿真 三、实现效果1、仿真…...
spark 集群
hadoop客户端环境准备 找到资料包路径下的Windows依赖文件夹,拷贝hadoop-3.1.0到非中文路径(比如d:\hadoop-3.1.0) ① 打开环境变量 ② 在下方系统变量中新建HADOOP_HOME环境变量,值就是保存hadoop的目录。 效果如下: ③ 配置P…...
leetcode117 填充每个节点的下一个右侧节点指针2
LeetCode 116 和 117 都是关于填充二叉树节点的 next 指针的问题,但它们的区别在于 树的类型 不同,117与 116 题类似,但给定的树是 普通二叉树(不一定完全填充),即某些节点可能缺少左或右子节点。 树的结构…...
Java全栈面试宝典:线程安全机制与Spring Boot核心原理深度解析
目录 一、Java线程安全核心原理 🔥 问题1:线程安全的三要素与解决方案 线程安全风险模型 线程安全三要素 synchronized解决方案 🔥 问题2:synchronized底层实现全解析 对象内存布局 Mark Word结构(64位系统&…...
CCF GESP C++编程 三级认证真题 2025年3月
C 三级 2025 年 03 月 题号 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 答案 D B A A B A C A C D D D B D C 1 单选题 第 1 题 Base64 编码将每3字节的输入数据编码为 4 字节的输出数据。如果输入数据长度不是 3 的倍数,会用 号填充。在Base64编码中,如果…...
人工智能爬虫导致维基共享资源带宽需求激增 50%
2025 年 4 月 1 日,维基媒体基金会在博文中表示,自 2024 年 1 月以来,维基共享资源下载多媒体的带宽消耗激增 50%,这一变化趋势主要由用于 AI 训练数据集的网络爬虫导致。以下是具体分析1: 爬虫流量特征与数据存储模式…...
方案精读:华为数据治理之旅【全文阅读】
本文介绍了华为的数据管理工作,包括数据治理、数据质量建设、数据管理工作两阶段历程、数据管理组织和数据管理工作思考。华为以业务数字化为前提,以数据入湖为基础,重点建设数据中台,提高数据质量和管理能力,以支撑公司数字化转型。 重点内容: 1. 数据治理:华为进行数…...
Tradingview 策略分享 - SSL 混合和 CE 交易策略
交易策略|https://v.wkbrowser.com/s/e9DIvLGvYRo/|复制浏览器打开 各位交易员大家好 在本文中,我将分享一个简单的日内交易策略。我将 SSL 混合指标与 CE EXIT 相结合,以创建一个高效且有利可图的设置。此策略简单而强大&#x…...
华为OD机试真题——投篮大赛(2025A卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
2025Q1 A卷 100分 题型 本文涵盖详细解题思路、代码注释、讲解、复杂的分析以及测试用例; 并提供Java、python、JavaScript、C、C语言、GO六种语言的最佳实现方式! 华为OD机试A卷真题《投篮大赛》: 题目名称:投篮大赛 知识点&am…...
rcore day2
目前常见的操作系统内核都是基于 C 语言的,为何要推荐 Rust 语言? C 语言的指针灵活且易于使用,但不保证安全性,且缺x少有效的并发支持。这导致内存和并发漏洞成为当前基于 C 语言的主流操作系统的噩梦。Rust 语言具有与 C 一样的…...
【MATLAB定位例程】TDOA(到达时间差)的chan-tylor,三维环境,附完整代码
该代码实现了基于三维空间的动态目标TDOA定位,结合了Chan算法(解析解)与Taylor级数展开法(迭代优化)的双重优势。 文章目录 运行结果MATLAB代码代码讲解代码功能概述核心算法原理代码结构解析可视化与结果分析运行结果 定位示意图: 三轴状态曲线: 三轴误差曲线: MA…...
LLM面试题六
NLP方向CRF算法面试题 什么是CRF?CRF的主要思想是什么? 设X与Y是随机变量,P(Y | X)是给定条件X的条件下Y的条件概率分布,若随机变量Y构成一个由无向图G(V,E)表示的马尔科夫随机场。则称条件概率分布P(X | Y)为条件随机场。CRF的主要思想统计…...
FPGA(四)——状态机
FPGA(四)——状态机 文章目录 FPGA(四)——状态机一、状态机编程思想二、LED流水灯仿真实验三、实现效果四、CPLD和FPGA芯片主要技术区别五、hdlbitsFPGA——组合逻辑学习1、创建一个D触发器2、简单状态转换3、4位移位寄存器4、计数器1-125、边缘捕获寄存器 一、状态机编程思想…...
AI 浪潮下企业身份管理:特点凸显,安全挑战升级
“在 AI 时代的浪潮中,企业身份管理是安全之锚,精准把握权限边界,方能抵御身份安全的暗流。” 在人工智能迅猛发展的当下,企业身份管理呈现出诸多新特点,同时也面临着前所未有的身份安全挑战。要理解这些,我…...
OBS 录屏软件 for Mac 视频录制
OBS 录屏软件 for Mac 视频录制 文章目录 OBS 录屏软件 for Mac 视频录制一、介绍二、效果三、下载 一、介绍 Open Broadcaster Software for mac版,OBS 有多种功能并广泛使用在视频采集,直播等领域。而且该软件功能全面,专业强大࿰…...
从文本到多模态:如何将RAG扩展为支持图像+文本检索的增强生成系统?
目录 从文本到多模态:如何将RAG扩展为支持图像文本检索的增强生成系统? 一、为什么需要扩展到多模态? 二、多模态 RAG 系统的基本架构 三、关键技术点详解 (一)多模态嵌入(Embedding)技术 …...
AI助力高效PPT制作:从内容生成到设计优化
随着人工智能技术的不断发展,AI在各个领域的应用日益普及,尤其是在文档和演示文稿的创建过程中。PowerPoint(PPT)作为最常用的演示工具之一,借助AI的技术手段,可以极大地提高制作效率并提升最终呈现效果。在…...
调用kimi api
官网支持python,curl和node.js 因为服务器刚好有php环境,所以先用curl调个普通的语音沟通api <?php // 定义 API Key 和请求地址 define(MOONSHOT_API_KEY, sk-PXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXgk1); define(MOONSHOT_API_URL, https://…...
自定义注解导入自定义Bean
在Spring框架中,通过自定义注解实现容器启动时自动导入Bean,通常需要结合 Import 注解、ImportBeanDefinitionRegistrar 接口 或 Configuration 配置类。以下是具体实现步骤和示例: 1. 定义自定义注解 创建一个注解,用于标记需要…...