FPGA-DDS信号发生器
FPGA-DDS信号发生器
DDS基本原理
FPGA实现的DDS(直接数字频率合成)波形生成器是一种高效、灵活的数字信号生成技术,广泛应用于通信、雷达和测试设备中。其核心原理是通过数字计算生成特定频率的波形。
DDS通过相位累加、查找表(LUT)和数模转换(DAC) 三个步骤生成波形:
- 相位累加器: 以固定时钟频率 F c l k F_{clk} Fclk运行,每个时钟周期将当前相位值增加一个频率控制字M。相位值决定了波形的瞬时相位;
- 查找表(LUT): 存储预先计算的波形样本(如正弦波的离散幅度值)。相位累加器的输出作为地址索引,从LUT中读取对应幅值;
- 数模转换(DAC): 将数字幅值转换为模拟信号(需外部DAC)。若仅需数字输出(如仿真或数字处理),可省略DAC;
- 其输出频率公式为: F o u t = M 2 N F c l k F_{out}=\frac{M}{2^N}F_{clk} Fout=2NMFclk,其中N为累加器的位数。
ACM9767 DAC模块
ACM9767是一个高速双通道DAC模块,每个通道数据分辨率为14位,输出电压为±5V,且转换速率高达125MHz,非常适合信号发生器、数字调制通信等应用。
其引脚如下图所示,每个通道有14个数据引脚,1个时钟输入(CLK),一个写入使能(WRT),其中写入使能与CLK保持一致即可(根据芯片手册描述,CLK信号的上升沿必须提前或恰好和 WRT 信号的上升沿一起出现)。
总体设计
每个通道分别通过三个按键控制其生成信号的波形,频率和相位。其中频率的可选范围在1Hz-10MHz,每次按动按钮频率提高十倍。波形分为正弦波、三角波和方波。
DDS相关设计分三层。最底层为基本原理层,用三个ROM保存不同波形的信息,根据输入的频率字和相位字,输出一定频率的14位的Data。中间层模块中预设好能够选择的频率和相位,根据输入的频率选择字和相位选择字,得到对应的频率字和相位字,再调用最底层模块得到相应的Data输出。顶层模块例化两个中间层模块作为通道A和B,并例化6个按键,对两个通道的波形、频率和相位进行控制。
ACM9767模块
ACM9767模块为最底层模块,实现DDS信号发生器最底层的原理。主要实现的功能如图所示。其中D/A转换器的功能由外接硬件模块实现。
波形数据表ROM
首先生成对应波形的LUT,LUT的深度为4096,也就是由4096个数据构成完整的波形,数据的位宽为14位(跟ACM9767的14位分辨率对应),波形用厂商提供的软件生成。如下图所示,生成正弦波形的数据。
然后创建ROM IP核,将LUT存储到ROM中。
具体步骤为,打开IP Catalog,然后搜索rom,双击Block Memory …
然后选择存储器类型为单口ROM,并将IP核名字改成rom_sin
配置数据位宽为14,深度为4096,使能配置为一直使能即可
最后载入波形数据即可
按照以上方法生成方波和三角波的波形和ROM
ACM9767.v
需要注意这里相位累加器(Freq_ACC)的位数设为32位,相位控制器的位数为12(因为数据的深度为 4096 = 2 12 4096=2^{12} 4096=212),则输出波形的频率为:
F o u t = M 2 32 ∗ F c l k F_{out}=\frac{M}{2^{32}}*F_{clk} Fout=232M∗Fclk
- Note: 实际中使用时, F c l k F_{clk} Fclk通过锁相环产生125MHz频率的时钟信号,但仿真文件中始终信号波形为50MHz
module ACM9767( Clk, Reset_n,Model_Sel[1:0], // 选择时钟Fword[31:0], // 频率控制字Pword[11:0], // 相位控制字Data[13:0] // 输出数据);input Clk;input Reset_n;input [1:0] Model_Sel;input [31:0] Fword;input [11:0] Pword;output reg [13:0] Data;// 频率控制字同步寄存器reg [31:0] Fword_r;always@(posedge Clk or negedge Reset_n)if(!Reset_n)Fword_r <= 0;elseFword_r <= Fword;// 相位控制字同步寄存器reg [11: 0] Pword_r;always@(posedge Clk or negedge Reset_n)if(!Reset_n)Pword_r <= 0;elsePword_r <= Pword;// 相位累加器reg [31:0] Freq_ACC;always@(posedge Clk or negedge Reset_n)if(!Reset_n)Freq_ACC <= 0;elseFreq_ACC <= Fword_r + Freq_ACC;// 波形数据表地址wire [11:0] Rom_Addr;assign Rom_Addr = Freq_ACC[31:20] + Pword_r;// 三角波wire [13: 0] Data_Tri;rom_triangular rom_triangular_ins(.clka(Clk),.addra(Rom_Addr),.douta(Data_Tri));// 正弦波wire [13: 0] Data_Sin;rom_sin rom_sin_ins(.clka(Clk),.addra(Rom_Addr),.douta(Data_Sin));// 方波wire [13:0] Data_Squ;rom_square rom_square_ins(.clka(Clk),.addra(Rom_Addr),.douta(Data_Squ));// 输出波形选择always@(posedge Clk or negedge Reset_n)if(!Reset_n)Data <= 0;else begincase(Model_Sel)0: Data <= Data_Tri;1: Data <= Data_Sin;2: Data <= Data_Squ;3: Data <= 0;endcaseend
endmodule
ACM9767_tb.v
`timescale 1ns / 1nsmodule ACM9767_tb();reg Clk;reg Reset_n;reg [1:0] Model_Sel;reg [31:0] Fword;reg [11:0] Pword;wire [13:0] Data;ACM9767 ACM9767_ins(.Clk(Clk), .Reset_n(Reset_n),.Model_Sel(Model_Sel), // 选择时钟.Fword(Fword), // 频率控制字.Pword(Pword), // 相位控制字.Data(Data) // 输出数据);initial Clk = 1;always #10 Clk = !Clk;initial beginReset_n = 0;#201;Reset_n = 1;Model_Sel = 0;Fword = 85899; // 1khz 1msPword = 0;#20000000;$stop;end
endmodule
DDS_ACM9767模块
DDS_ACM9767为中间模块,主要功能为根据计算出的频率字和相位字,预设频率和相位,并能通过频率选择字、相位选择字进行选择。
- Note: 该模块仿真中设定的 F c l k F_{clk} Fclk为125MHz
DDS_ACM9767.v
// 时钟为125MHz
module DDS_ACM9767(Clk,Reset_n,Model_Sel[1:0],Fword_Sel[2:0],Pword_Sel[2:0],Data[13:0]);input Clk;input Reset_n;input [1:0] Model_Sel;input [2:0] Fword_Sel;input [2:0] Pword_Sel;output wire [13:0] Data;// Fwordreg [31:0] Fword;always@(posedge Clk or negedge Reset_n)if(!Reset_n)Fword <= 0;else begincase(Fword_Sel)0: Fword <= 34; // 1Hz 1/(125MHz)/2^{32}1: Fword <= 344; // 10Hz2: Fword <= 3436; // 100Hz3: Fword <= 34360; // 1KHz4: Fword <= 343597; // 10KHz5: Fword <= 3435974; // 100KHZ6: Fword <= 34359738; // 1MHz7: Fword <= 343597384; // 10MHzdefault: Fword <= 0;endcaseend// Pwordreg [11:0] Pword;always@(posedge Clk or negedge Reset_n)if(!Reset_n)Pword <= 0;else begincase(Pword_Sel)0: Pword <= 0; // 01: Pword <= 341; // 30 4096 / 122: Pword <= 683; // 603: Pword <= 1024; // 904: Pword <= 1707; // 1505: Pword <= 2048; // 1806: Pword <= 3072; // 2707: Pword <= 3641; // 320default: Pword <= 0;endcaseendACM9767 ACM9767_Ins(.Clk(Clk), .Reset_n(Reset_n),.Model_Sel(Model_Sel), // 波形选择.Fword(Fword), // 频率控制字.Pword(Pword), // 相位控制字.Data(Data) // 输出数据);
endmodule
DDS_ACM9767_tb.v
`timescale 1ns / 1nsmodule DDS_ACM9767_tb();reg Clk;reg Reset_n;reg [1:0] Model_Sel_A;reg [2:0] Fword_Sel_A;reg [2:0] Pword_Sel_A;wire [13: 0] DataA;reg [1:0] Model_Sel_B;reg [2:0] Fword_Sel_B;reg [2:0] Pword_Sel_B;wire [13: 0] DataB;DDS_ACM9767 DDS_ACM9767_Ins_A(.Clk(Clk),.Reset_n(Reset_n),.Model_Sel(Model_Sel_A),.Fword_Sel(Fword_Sel_A),.Pword_Sel(Pword_Sel_A),.Data(DataA));DDS_ACM9767 DDS_ACM9767_Ins_B(.Clk(Clk),.Reset_n(Reset_n),.Model_Sel(Model_Sel_B),.Fword_Sel(Fword_Sel_B),.Pword_Sel(Pword_Sel_B),.Data(DataB));initial Clk = 1;always #4 Clk = !Clk; //125MHzinitial beginReset_n = 0;#201;Reset_n = 1;Model_Sel_A = 0;Fword_Sel_A = 6; // 1MHzPword_Sel_A = 0;Model_Sel_B = 0;Fword_Sel_B = 6; // 1MHzPword_Sel_B = 3; // 90#2000000;Reset_n = 0;#201;Reset_n = 1;Model_Sel_A = 1;Fword_Sel_A = 5; // 1MHzPword_Sel_A = 0;#2000000;$stop;end
endmodule
DDS_Module模块
该模块为顶层模块,例化两个DDS_ACM9767模块,并例化六个按键用于控制波形、频率和相位。由于晶振为50MHz有源晶振,要想使用125MHz的时钟信号来驱动ACM9767运行,需要通过锁相环(PLL)对基频信号进行倍频和分频,锁相环的配置可通过时钟管理单元IP核进行。
- Note: 用按键触发信号,作为
DDS_ACM9767
模块的复位信号,当按下按键改变设置时,相应参数也重新复原。
时钟管理单元IP核
由于开发板上晶振为50MHz,需要使用时钟管理单元,生成125MHz信号。
在IP Catlalog中找到Clocking Wizard
输入信号频率设为50MHz
输出信号频率设为125MHz
复位类型选择低电平触发(根据开发板硬件设计)
DDS_Module.v
module DSS_Module(Clk,Reset_n,ClkA,SkeyA,FkeyA,PkeyA,WrtA,ClkB,SkeyB,FkeyB,PkeyB,WrtB,DataA[13:0],DataB[13:0]);input Clk;input Reset_n;input SkeyA;input FkeyA;input PkeyA;input SkeyB;input FkeyB;input PkeyB;output wire ClkA;output wire ClkB;output wire WrtA;output wire WrtB;output wire [13:0] DataA;output wire [13:0] DataB;// clockwire Clk125MHz;wire locked;// DDS_ACM9767reg [1:0] Model_Sel_A;reg [2:0] Fword_Sel_A;reg [2:0] Pword_Sel_A;reg [1:0] Model_Sel_B;reg [2:0] Fword_Sel_B;reg [2:0] Pword_Sel_B;// keywire Press_Flag_SkeyA;wire Release_Flag_SkeyA;wire filter_state_SkeyA;wire Press_Flag_FkeyA;wire Release_Flag_FkeyA;wire filter_state_FkeyA;wire Press_Flag_PkeyA;wire Release_Flag_PkeyA;wire filter_state_PkeyA;wire Press_Flag_SkeyB;wire Release_Flag_SkeyB;wire [1:0] filter_state_SkeyB;wire Press_Flag_FkeyB;wire Release_Flag_FkeyB;wire [1:0] filter_state_FkeyB;wire Press_Flag_PkeyB;wire Release_Flag_PkeyB;wire [1:0] filter_state_PkeyB;wire Reset_A;wire Reset_B;assign Reset_A = Release_Flag_SkeyA | Release_Flag_FkeyA | Release_Flag_PkeyA;assign Reset_B = Release_Flag_SkeyB | Release_Flag_FkeyB | Release_Flag_PkeyB;assign ClkA = Clk125MHz;assign ClkB = Clk125MHz;assign WrtA = ClkA;assign WrtB = ClkB;// 时钟管理单元生成125MHz时钟clk_wiz_0 clk_125_ins(.Clk125MHz(Clk125MHz), // output Clk125MHz.resetn(Reset_n), // input rese .locked(locked), // output locked// Clock in ports.clk_in1(Clk)); // 例化DDS_ACM9767DDS_ACM9767 DDS_ACM9767_Ins_A(.Clk(Clk125MHz),.Reset_n(!Reset_A),
// .Reset_n(Reset_n),.Model_Sel(Model_Sel_A),.Fword_Sel(Fword_Sel_A),.Pword_Sel(Pword_Sel_A),.Data(DataA));DDS_ACM9767 DDS_ACM9767_Ins_B(.Clk(Clk125MHz),.Reset_n(!Reset_B),
// .Reset_n(Reset_n),.Model_Sel(Model_Sel_B),.Fword_Sel(Fword_Sel_B),.Pword_Sel(Pword_Sel_B),.Data(DataB));// 例化Key_Filterkey_filter key_filter_SkeyA(.Clk(Clk125MHz),.Reset_n(Reset_n),.signal(SkeyA),.Press_Flag(Press_Flag_SkeyA),.Release_Flag(Release_Flag_SkeyA),.filter_state(filter_state_SkeyA),.result());key_filter key_filter_FkeyA(.Clk(Clk125MHz),.Reset_n(Reset_n),.signal(FkeyA),.Press_Flag(Press_Flag_FkeyA),.Release_Flag(Release_Flag_FkeyA),.filter_state(filter_state_FkeyA),.result());key_filter key_filter_PkeyA(.Clk(Clk125MHz),.Reset_n(Reset_n),.signal(PkeyA),.Press_Flag(Press_Flag_PkeyA),.Release_Flag(Release_Flag_PkeyA),.filter_state(filter_state_PkeyA),.result());key_filter key_filter_SkeyB(.Clk(Clk125MHz),.Reset_n(Reset_n),.signal(SkeyB),.Press_Flag(Press_Flag_SkeyB),.Release_Flag(Release_Flag_SkeyB),.filter_state(filter_state_SkeyB),.result());key_filter key_filter_FkeyB(.Clk(Clk125MHz),.Reset_n(Reset_n),.signal(FkeyB),.Press_Flag(Press_Flag_FkeyB),.Release_Flag(Release_Flag_FkeyB),.filter_state(filter_state_FkeyB),.result());key_filter key_filter_PkeyB(.Clk(Clk125MHz),.Reset_n(Reset_n),.signal(PkeyB),.Press_Flag(Press_Flag_PkeyB),.Release_Flag(Release_Flag_PkeyB),.filter_state(filter_state_PkeyB),.result());// Model_Sel_Aalways@(posedge Clk125MHz or negedge Reset_n)if(!Reset_n)Model_Sel_A <= 0;else if(Release_Flag_SkeyA && (filter_state_SkeyA == 0))Model_Sel_A <= Model_Sel_A + 1;// Fword_Sel_Aalways@(posedge Clk125MHz or negedge Reset_n)if(!Reset_n)Fword_Sel_A <= 3; // 1khzelse if(Release_Flag_FkeyA && (filter_state_FkeyA == 0))Fword_Sel_A <= Fword_Sel_A + 1;// Pword_Sel_Aalways@(posedge Clk125MHz or negedge Reset_n)if(!Reset_n)Pword_Sel_A<= 0;else if(Release_Flag_PkeyA && (filter_state_PkeyA == 0))Pword_Sel_A <=Pword_Sel_A + 1;// Model_Sel_Balways@(posedge Clk125MHz or negedge Reset_n)if(!Reset_n)Model_Sel_B <= 1;else if(Release_Flag_SkeyB && (filter_state_SkeyB == 0))Model_Sel_B <= Model_Sel_B + 1;// Fword_Sel_Balways@(posedge Clk125MHz or negedge Reset_n)if(!Reset_n)Fword_Sel_B <= 3;else if(Release_Flag_FkeyB && (filter_state_FkeyB == 0))Fword_Sel_B <= Fword_Sel_B + 1;// Pword_Sel_Balways@(posedge Clk125MHz or negedge Reset_n)if(!Reset_n)Pword_Sel_B<= 0;else if(Release_Flag_PkeyB && (filter_state_PkeyB == 0))Pword_Sel_B <= Pword_Sel_B + 1;
endmodule
DDS_Module_tb.v
- Note: 由于信号较多,软件仿真的速度很慢,不利于调试,仅供参考,推荐使用ILA硬件调试。(ILA硬件调试方法会在后面小节具体介绍)
`timescale 1ns / 1ns
// Not in use
module DDS_Module_tb();reg Clk;reg Reset_n;reg SkeyA;reg FkeyA;reg PkeyA;reg SkeyB;reg FkeyB;reg PkeyB;wire ClkA;wire ClkB;wire [13:0] DataA;wire [13:0] DataB;DSS_Module DSS_Module_ins(.Clk(Clk),.Reset_n(Reset_n),.ClkA(ClkA),.SkeyA(SkeyA),.FkeyA(FkeyA),.PkeyA(PkeyA),.WrtA(),.ClkB(ClkB),.SkeyB(SkeyB),.FkeyB(FkeyB),.PkeyB(PkeyB),.WrtB(),.DataA(DataA),.DataB(DataB));initial Clk = 1;always #10 Clk = !Clk;reg [31: 0] rand;initial beginReset_n = 0;#201;Reset_n = 1;press_skeyA(0);press_pkeyA(0);press_fkeyA(0);$stop;endtask press_skeyA;input [3:0] seed;beginSkeyA = 1;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand SkeyA = ~SkeyA;endSkeyA = 0;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand SkeyA = ~SkeyA;endSkeyA = 1;#20000000;#20000000;endendtasktask press_fkeyA;input [3:0] seed;beginFkeyA = 1;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand FkeyA = ~FkeyA;endFkeyA = 0;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand FkeyA = ~FkeyA;endFkeyA = 1;#20000000;#20000000;endendtasktask press_pkeyA;input [3:0] seed;beginPkeyA = 1;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand PkeyA = ~PkeyA;endPkeyA = 0;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand PkeyA = ~PkeyA;endPkeyA = 1;#20000000;#20000000;endendtask
endmodule
key_filter模块
按键模块,进行20ms的消抖处理,若时钟频率改变,则需要修改对于计数器的最大计数值,这里支持的是125MHz的频率。
key_filter.v
// 按键消抖 等待按键按下(下降沿) --->消抖, 若20ms没出现上升沿 ---> 按键已按下---?等待释放 ---> 20ms? --->已释放
module key_filter(Clk,Reset_n,signal,Press_Flag,Release_Flag,filter_state,result);input Clk;input Reset_n;input signal;output reg Press_Flag;output reg Release_Flag;output reg [1:0] filter_state;output reg result;// 多级寄存器缓解亚稳态现象reg [1: 0] sync_reg;always@(posedge Clk or negedge Reset_n)if(!Reset_n)sync_reg <= 0;elsesync_reg <= {sync_reg[0], signal};// 边缘检测器reg [1:0] edge_reg; always@(posedge Clk or negedge Reset_n)if(!Reset_n)edge_reg <= 0;elseedge_reg <= {edge_reg[0], sync_reg[1]};wire pos_edge;assign pos_edge = (edge_reg == 2'b01);wire neg_edge;assign neg_edge = (edge_reg == 2'b10);reg [23:0] cnt;// filter_statealways@(posedge Clk or negedge Reset_n)if(!Reset_n)filter_state <= 0;else if(filter_state == 0)beginif(neg_edge)filter_state <= 1;endelse if(filter_state == 1)beginif(pos_edge)filter_state <= 0; // 回到等待状态else if(cnt == 2500000 - 1)// 20msfilter_state <= 2;endelse if(filter_state == 2)beginif(pos_edge) // 出现上升沿filter_state <= 3; // 进入释放消抖endelse if(filter_state == 3)beginif(neg_edge)filter_state <= 2;else if(cnt == 2500000 - 1)filter_state <= 0;end// Press_Flagalways@(posedge Clk or negedge Reset_n)if(!Reset_n)Press_Flag <= 0;else if(filter_state == 1 && (cnt == 2500000 - 1))Press_Flag <= 1;else if(filter_state == 2)Press_Flag <= 0;// Release_Flagalways@(posedge Clk or negedge Reset_n)if(!Reset_n)Release_Flag <= 0;else if(filter_state == 0)Release_Flag <= 0;else if(filter_state == 3 && (cnt == 2500000 - 1)) Release_Flag <= 1;// resultalways@(posedge Clk or negedge Reset_n)if(!Reset_n)result <= 1;else if(filter_state == 1 && (cnt == 2500000 - 1))result <= 0;else if(filter_state == 3 && (cnt == 2500000 - 1))result <= 1;// cntalways@(posedge Clk or negedge Reset_n)if(!Reset_n)cnt <= 0;else if(filter_state == 1)begincnt <= cnt + 1;if(pos_edge)cnt <= 0;else if(cnt == 2500000 - 1)cnt <= 0;endelse if(filter_state == 3)begincnt <= cnt + 1;if(neg_edge)cnt <= 0; else if(cnt == 2500000 - 1)cnt <= 0;end
endmodule
key_filter_tb.v
`timescale 1ns / 1ns
module key_filter_tb();reg Clk;reg Reset_n;reg signal;wire Press_Flag;wire Release_Flag;wire [1:0] filter_state;wire result;key_filter key_filter_ins(.Clk(Clk),.Reset_n(Reset_n),.signal(signal),.Press_Flag(Press_Flag),.Release_Flag(Release_Flag),.filter_state(filter_state),.result(result));initial Clk = 1;always #4 Clk = !Clk;reg [31: 0] rand;initial beginReset_n = 0;#201;Reset_n = 1;#20;press_key(0);$stop;endtask press_key;input [3:0] seed;beginsignal = 1;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand signal = ~signal;endsignal = 0;#20000000;repeat(5)beginrand = {$random} % 10000000; // 10ms以内的抖动#rand signal = ~signal;endsignal = 1;#20000000;#20000000;endendtask
endmodule
ILA硬件调试方法
ILA的使用方式有很多,这里就讲一种我比较喜欢的方式。
首先,运行综合,后打开综合设计器。
然后在netlist里面,可以选择感兴趣的信号设置debug,被选中的信号左边会有个小爬虫
右上角选择Debug界面
在Debug界面为没有分配ILA IP核的信号,创建或分配一个探针
进入到ILA IP核向导界面
跟着向导走,直到来到以下界面,该界面可以配置需要debug信号探针的类型,Data
、Trigger
Data&Trigger
,触发类型信号,设置触发条件后,当达到触发条件时,ILA会对信号进行采样。这里我们只想看看按键按下后,对应的选择字位有没有变化,因此设置成Data
类型的探针就好,当按下开发板按键后,手动触发信号采样。
- PkeyA_IBUF: 是为了展示使用ILA过程。
然后选择合适的深度,就是一次采样中,采样点的个数
设置完成后,正常流程生产比特流文件即可。然后当连接到硬件时,Vivado会将比特流文件和调试文件一起烧录到开发板中。
然后可以看到ILA的界面,点击触发按钮手动触发一次采样。
可以看到采样到的值
按下开发板上的按键,再重新手动触发采样,看看值是否有改变,若改变,则说明按键部分的功能实现无误。
ILA还能设置触发条件进行采样,在有些应用中只能通过该方法进行调试,具体方法自行了解。
实验现象
看看效果
FPGA-DDS
相关硬件
小梅哥FPGA ACX720
FPGA芯片型号: XC7A35TFGG484ABX2l
DMA: ACM9767
示波器: 正点原子
相关文章:
FPGA-DDS信号发生器
FPGA-DDS信号发生器 DDS基本原理 FPGA实现的DDS(直接数字频率合成)波形生成器是一种高效、灵活的数字信号生成技术,广泛应用于通信、雷达和测试设备中。其核心原理是通过数字计算生成特定频率的波形。 DDS通过相位累加、查找表(LUT)…...
二进制、高位低位、位移操作与进制转换全解
二进制、高位低位、位移操作与进制转换全解 在计算机科学中,理解高位与低位、左移与右移、进制转换与位运算非常重要。这篇文章用清晰直观的方式梳理这些基本概念。 高位与低位 低位:二进制中靠右的位,权值较小(例如 (2^0, 2^1…...
docker存储
注意:数据卷挂载(卷映射):Docker会自动创建数据卷,并将容器运行所需的文件复制到数据卷中。 目录挂载:如果宿主机上没有对应的目录,容器会因为缺少运行所需的文件而出错。 1.目录挂载 指令&am…...
回归预测 | Matlab实现DBO-LightGBM蜣螂算法优化轻量级梯度提升机多输入单输出回归预测,作者:机器学习之心
回归预测 | Matlab实现DBO-LightGBM蜣螂算法优化轻量级梯度提升机多输入单输出回归预测,作者:机器学习之心 目录 回归预测 | Matlab实现DBO-LightGBM蜣螂算法优化轻量级梯度提升机多输入单输出回归预测,作者:机器学习之心预测效果…...
[ 问题解决 ] sqlite3.ProgrammingError: SQLite objects created in a thread can ...
目录 为什么会出现这个问题? 解决方法一:每个请求新建自己的连接(推荐) 解决方法二:允许 SQLite 跨线程使用连接(不推荐) 小结 当你在 python 中使用 Flask 里面调用了数据库的操作的时候&a…...
AI智能体开发新范式:多智能体协作与自进化系统的构建之道
一、从单Agent到多Agent:为什么“群体智能”是必然? 复杂任务的分而治之案例: 电商大促活动的全自动运营商品Agent:实时调价(根据库存/竞品)用户Agent:生成千人千面推荐风控Agent:检…...
js补环境工具使用技巧、补环境实例、重点环境检测点详解
什么是补环境,模拟浏览器环境让浏览器js运行,为什么需要补环境,因为浏览器和本地nodejs环境有差异,网站开发者为了检测用户是否是本地环境运行 主要补的环境Document,Window,Navigator,Location,Element 这是内置原始类型&#…...
TF_LOG 配置及级别详解
以下是Terraform中TF_LOG配置及级别的详解: 配置方法 设置日志级别 通过设置TF_LOG环境变量来启用Terraform的日志功能,并指定日志级别。可以将该变量设置为以下值之一:TRACE、DEBUG、INFO、WARN、ERROR。其中,TRACE级别最为详…...
vue3使其另一台服务器上的x.html,实现x.html调用中的函数,并向其传递数据。
vue3例子 <template><div><iframeload"loadIFreamSite"id"loadIframeSite":src"iframeSrc1"frameborder"0"scrolling"no"allowtransparency"true"style"width: 100%"></iframe&g…...
英语五大基本句型
文章目录 一、主谓二、主谓宾三、主系表什么是什么什么怎么样系动词感官动词 一、主谓 构成:动作的发出者 动作 例句:I run.(我跑步。) 二、主谓宾 构成:动作的发出者 动作 动作的接受者 构成:主语&a…...
什么是 DDoS 攻击?高防 IP 如何有效防护?2025全面解析与方案推荐
一、DDoS 攻击:互联网时代的 “数字核武器” 1. DDoS 攻击的本质与原理 ** 分布式拒绝服务攻击(DDoS)** 通过操控海量僵尸设备,向目标服务器发送洪水般请求,耗尽带宽、连接或计算资源,导致合法用户无法访…...
论文速报《Enhancing Autonomous Driving Systems...:LLM-MPC混合架构增强自动驾驶》
论文链接:https://arxiv.org/pdf/2504.11514 代码链接:https://github.com/ForzaETH/LLMxRobot 0. 简介 自动驾驶领域的传统方法多依赖于数据驱动模型,通过大量标注数据训练实现路径规划和控制。然而,现实世界中道路临时施工、突…...
Nacos 3.0 上线 MCP Registry,支持 MCP 服务注册到发现全流程管理
Nacos 3.0 正式版本发布啦!升级 MCP Registry,围绕着 MCP(Model Context Protocol) 服务管理,MCP 多种类型注册,包含 MCP Server 注册、编排、动态调试和管理,并且提供 Nacos-MCP-Router 可以进…...
一文解析大语言模型量化技术
目录 一、为什么需要量化技术 1、数据规模 2、32位浮点数(FP32) 3、16位浮点数(FP16) 4、Bfloat16(BF16) 5.INT8(8位整数)和INT4(4位整数) 总结&#…...
使用python实现自动化拉取压缩包并处理流程
使用python实现自动化拉取压缩包并处理流程 实现成果展示使用说明 实现成果展示 使用说明 执行./run.sh 脚本中的内容主要功能是: 1、从远程服务器上下拉制定时间更新的数据 2、将数据中的zip拷贝到指定文件夹内 3、解压后删除所有除了lcm之外的文件 4、新建一个ou…...
解构编程语言的基因密码:论数据类型如何被语言系统定义与重塑
摘要 本文从理论与实践层面系统探讨编程语言中数据类型的定义、实现与演化。通过静态与动态类型系统的差异分析,结合案例、流程图和表格,全面呈现主流语言数据类型设计特点及其对内存管理、错误防范与性能优化的影响。文章旨在为语言设计者和开发者提供…...
GRPO vs SFT:强化学习提升大模型多模态推理泛化能力的原因研究
GRPO vs SFT:强化学习提升大模型多模态推理泛化能力的原因研究 作者:吴宇斌 原文地址:https://zhuanlan.zhihu.com/p/1892362859628963761 训练目标与优化方式差异对比 监督微调(SFT)的目标: SFT使用带标注…...
从千兆到40G:飞速(FS)助力制造企业构建高可靠智能生产网络
案例亮点 部署S5850-24S2Q交换机,启用MLAG跨设备链路聚合,构建高性能冗余架构,消除单点故障风险,将网络可用性提升至99.99%,保障生产系统与全球业务连续性。采用40G光模块与US Conec MTP连接头多模跳线实现数据中心间…...
WHAT - 《成为技术领导者》思考题(第三章)
文章目录 涉及内容理解问题管理想法的交流保证质量 思考题思路和示例框架1. 观察一个你认为是领导者的人,列出他的行为,分类,并思考自己未采用的行为2. 观察一个不太像领导者的人,列出错过的简单机会,并反思3. 让别人注…...
Go 语言入门:(一) 环境安装
一、前言 这里不同于其他人的 Go 语言入门,环境安装我向来注重配置,比如依赖包、缓存的默认目录。因为前期不弄好,后面要整理又影响这影响那的,所以就干脆写成文章,方便后期捡起。 二、安装 1. 安装包 https://go.…...
GTC2025全球流量大会:领驭科技以AI云端之力,助力中国企业出海破浪前行
在全球化与数字化浪潮下,AI技术正成为中国企业出海的重要驱动力。一方面,AI通过语言处理、数据分析等能力显著提升出海企业的运营效率与市场适应性,尤其在东南亚等新兴市场展现出"高性价比场景适配"的竞争优势;另一方面…...
013几何数学——算法备赛
几何数学 平面切分 蓝桥杯2020年省赛题 问题描述 平面上有N条直线,其中第i条直线为yAxB.请计算这些直线将平面分成了几个部分? 输入 第一行输入一个N,接下来N行输入两个整数代表Ai和Bi。 1<N<10^5. 思路分析 初始时一条直线将…...
VUE3:封装一个评论回复组件
之前用React封装的评论回复组件,里面有三个主要部分:CommentComponent作为主组件,CommentItem处理单个评论项,CommentInput负责输入框。现在需要将这些转换为Vue3的组件。 Vue3和React在状态管理上有所不同,Vue3使用r…...
DELL R740服务器闪黄灯不开机故障案例
1:DELL R740服务器 2:东莞长安客户工厂晚上十一二点电路跳闸多次,导致R740 ERP服务器无法开机。 3:故障现象为:主机能正常通电,开机按钮无通电迹象,正常情况会闪绿灯慢闪,通电一会后…...
记录一下QA(from deepseek)
Q1:__init__.py文件 在 Python 中,当你在一个目录下创建 __init__.py 文件时,这个目录会被视为一个 包(Package)。包的存在使得 Python 能够通过点号(.)层级式地组织模块(.py 文件)&…...
码蹄集——进制输出、求最大公约数、最小公倍数
进制乱炖 本题考查输出的进制转换,可以直接使用c里的format格式输出 #include<iostream> #include<algorithm> #include<string> using namespace std;int main() {int x;cin>>x;printf("%d %o %x %u\n",x,x,x,x);//十进制 八进…...
从技术走向管理:带来哪些角色转变与挑战
文章目录 一、从技术到管理1、从技术转到管理的优劣势(1)优势(2)劣势 2、刚转岗容易犯的几个问题3、最大的变化:不再是一个人单打独斗4、警惕:一开始不要把“人”过早的介入到“事”5、如何完成角色的转变&…...
C语言-指针(一)
目录 指针 内存 概念 指针变量 取地址操作符(&) 操作符“ * ” 指针变量的大小 注意 指针类型的意义 作用 void * 指针 const修饰指针变量 const放在*前 const放在*后 双重const修饰 指针的运算 1.指针 - 整数 2.指针 - 指针 3.指…...
Python面试问题
一、Python 基础 1. Python 的特点 动态类型:变量无需声明类型。解释型语言:逐行解释执行。支持多种编程范式(面向对象、函数式、过程式)。 2. 列表(List)与元组(Tuple)的区别 特…...
RAG工程-基于LangChain 实现 Advanced RAG(预检索优化)
Advanced RAG 概述 Advanced RAG 被誉为 RAG 的第二范式,它是在 Naive RAG 基础上发展起来的检索增强生成架构,旨在解决 Naive RAG 存在的一些问题,如召回率低、组装 prompt 时的冗余和重复以及灵活性不足等。它重点聚焦在检索增强࿰…...
【时时三省】(C语言基础)循环结构程序设计习题1
山不在高,有仙则名。水不在深,有龙则灵。 ----CSDN 时时三省 习题1 输入两个正整数m和n,求其最大公约数和最小公倍数。 解题思路: 求两个正整数 m 和 n 的最大公约数通常使用辗转相除法(欧几里得算法ÿ…...
[密码学实战]SDF之设备管理类函数(一)
[密码学实战]SDF之设备管理类函数(一) 一、标准解读:GM/T 0018-2023核心要求 1.1 SDF接口定位 安全边界:硬件密码设备与应用系统间的标准交互层功能范畴: #mermaid-svg-s3JXUdtH4erONmq9 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16p…...
CDGP|如何建立高效的数据治理团队?
近年来,数据治理行业迅速发展,越来越多的企业开始重视并投入大量资源来建立和完善数据治理体系。数据治理体系不仅能够帮助企业更好地管理和利用数据资源,提升数据质量和数据价值,还能够为企业带来竞争优势和可持续发展能力。 然…...
如何评价 DeepSeek 的 DeepSeek-V3 模型?
DeepSeek-V3 是由杭州 DeepSeek 公司于 2024 年 12 月 26 日发布的一款开源大语言模型,其性能和创新技术在国内外引起了广泛关注。从多个方面来看,DeepSeek-V3 的表现令人印象深刻,具体评价如下: 性能卓越 DeepSeek-V3 拥有 6710 …...
【基础篇】prometheus命令行参数详解
文章目录 本篇内容讲解命令行参数详解 本篇内容讲解 prometheus高频修改命令行参数详解 命令行参数详解 在页面的/页面上能看到所有的命令行参数,如图所示: 使用shell命令查看 # ./prometheus --help usage: prometheus [<flags>]The Promethe…...
SpringBoot实现接口防刷的5种高效方案详解
目录 前言:接口防刷的重要性 方案一:基于注解的访问频率限制 实现原理 核心代码实现 使用示例 优缺点分析 方案二:令牌桶算法实现限流 算法原理 核心实现 配置使用 适用场景分析 方案三:分布式限流(Redis …...
DeepSearch复现篇:QwQ-32B ToolCall功能初探,以Agentic RAG为例
DeepSearch复现篇:QwQ-32B ToolCall功能初探,以Agentic RAG为例 作者:CyPaul Space 原文地址:https://zhuanlan.zhihu.com/p/30289363967 全文阅读约3分钟~ 背景 今天看到 论文:Search-R1: Training LLMs to Reason …...
项目实战-贪吃蛇大作战【补档】
这其实算是一个补档,因为这个项目是我在大一完成的,但是当时没有存档的习惯,今天翻以前代码的时候翻到了,于是乎补个档,以此怀念和志同道合的网友一起做项目的日子 ₍ᐢ ›̥̥̥ ༝ ‹̥̥̥ ᐢ₎♡ 这里面我主要负责…...
power bi获取局域网内共享文件
power bi获取局域网内共享文件 需求: 数据源并不一定都是在本地,有可能在云端,也有可能在其他服务器,今天分享如果数据源在另外一台服务器,如何获取数据源的方法。 明确需求:需要通过PowerBI获取局域网中的…...
100%提升信号完整性:阻抗匹配在高速SerDes中的实践与影响
一个高速信号SerDes通道(例如PCIe、112G/224G-PAM4)包含了这些片段: 传输线连通孔(PTH or B/B via)连接器高速Cable锡球(Ball and Bump) 我们会希望所有的片段都可以有一致的阻抗,…...
第六章:Tool and LLM Integration
Chapter 6: Tool and LLM Integration 从执行流到工具集成:如何让AI“调用真实世界的技能”? 在上一章的执行流框架中,我们已经能让多个代理协作完成复杂任务。但你是否想过:如果用户要求“查询实时天气”或“打开网页搜索”&…...
prompt提示词编写技巧
为什么学习prompt编写 目的:通过prompt的编写,提升LLM输出相关性、准确性和多样性,并对模型输出的格式进行限制,满足我们的业务需求。 学过提示词工程的人:像“专业导演”,通过精准指令控制 AI 输出&#…...
Nginx配置SSL详解
文章目录 Nginx配置SSL详解1. SSL/TLS 基础知识2. 准备工作3. 获取SSL证书4. Nginx SSL配置步骤4.1 基础配置4.2 配置说明 5. 常见配置示例5.1 双向认证配置5.2 多域名SSL配置 6. 安全优化建议7. 故障排查总结参考资源下载验证的完整实例 Nginx配置SSL详解 1. SSL/TLS 基础知识…...
网络安全之红队LLM的大模型自动化越狱
前言 大型语言模型(LLMs)已成为现代机器学习的重要支柱,广泛应用于各个领域。通过对大规模数据的训练,这些模型掌握了多样化的技能,展现出强大的生成与理解能力。然而,由于训练数据中难以完全剔除有毒内容&…...
【技术笔记】通过Cadence Allegro创建一个PCB封装(以SOT23为例)
【技术笔记】通过Cadence Allegro创建一个PCB封装(以SOT23为例) 一、焊盘创建二、PCB封装设计三、丝印位号及标识添加 更多内容见专栏:【硬件设计遇到了不少问题】、【Cadence从原理图到PCB设计】 一、焊盘创建 首先要找到元器件的相关手册&…...
新环境注册为Jupyter 内核
1. 确认环境是否已注册为内核 在终端运行以下命令,查看所有已注册的内核: jupyter kernelspec list2. 为自定义环境注册内核 步骤 1:激活目标虚拟环境 conda activate your_env_name # 替换为你的环境名步骤 2:安装…...
[Spring] Seata详解
🌸个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 🏵️热门专栏: 🧊 Java基本语法(97平均质量分)https://blog.csdn.net/2301_80050796/category_12615970.html?spm1001.2014.3001.5482 🍕 Collection与…...
使用JDK的数据校验和Spring的自定义注解校验前端传递参数的两种方法
第一种:JDK的数据校验注解 PostMapping("/test")public String test(QueryParam param, RequestHeader(value "App_key") String App_key,RequestHeader(value "App_secret") String App_secret) throws IOException {param.setApp…...
JS错误处理的新方案 (不使用try-catch)
错误处理一直是JavaScript开发者需要认真对待的问题,传统的try-catch语法虽然简单直观,但在异步代码中使用时存在诸多限制。 try-catch的局限性 传统try-catch模式在现代JavaScript开发中面临的问题: 1. 异步错误捕获的缺陷 try-catch无法…...
前端实现商品放大镜效果(Vue3完整实现)
前端实现商品放大镜效果(Vue3完整实现) 前言 在电商类项目中,商品图片的细节展示至关重要。放大镜效果能显著提升用户体验,允许用户在不跳转页面的情况下查看高清细节。本文将基于Vue3实现一个高性能的放大镜组件,完整…...