『 Linux 』信号概念与信号的产生

文章目录

    • 信号概念
    • 前台进程与后台进程
    • 信号的本质
    • 硬件理解
    • 信号的产生


信号概念

请添加图片描述

"信号"一词指用来传达信息或只是的各种形式的提示或标志;

在生活中常见的信号例如红绿灯,交通标志,短信通知等

在操作系统中,"信号"是一种用于异步通知进程发生特定事件的机制;

信号允许操作系统或其他进程向目标进程发送通知以便他们能够相应某些时间或条件;

  • 异步

    异步指在处理任务或操作时发起操作与处理结果的时间节点不一致;

    意味着操作的启动和完成不是在同一时间点发生,这种机制一般用于提高程序和效率的响应性;

    • 任务启动与完成的独立性

      在异步操作中任务的发起和完成是独立的;

      发起操作后程序可以继续执行其他任务而不需要等待操作完成;

      使得程序能够在等待期间进行其他有用的工作而提高效率;

    • 回调机制

      异步操作通常使用回调函数来处理操作完成后的结果;

      回调函数会在操作完成时被调用,处理结果或执行后续操作;

      这种机制允许程序继续运行其他任务直到操作完成时才处理结果;

在默认情况下,操作系统将为每个进程预设对所有信号的接收及处理方式;

默认的信号处理行为是由操作系统定义的,确保在没有消失设置信号处理程序的情况下能够合理响应各种信号;

使得即便目前未产生信号时进程也能知道在信号产生之后该作何处理;

  • 信号的异步处理特性

    当一个信号产生时并不会立即中断进程正在执行的关键代码,而是在适当的时候处理;

    表明在信号的产生到处理的过程中必须存在一个"时间窗口"用于进程保存已经识别到的信号,并在适合时间段对信号进行处理;

    这个机制通常通过信号的阻塞(屏蔽)和信号队列来实现;

一个信号由产生到被处理一般要经过三个步骤:

  • 信号的产生
  • 信号的保存
  • 信号的处理


前台进程与后台进程

请添加图片描述

  • 前台进程

    前台进程是直接与用户进行交互的进程;

    用户可以通过中断或者命令行界面与前台进程进行实时交互;

    • 交互性

      前台进程直接接收用户的输入,并将输出显示给用户;

      如在终端运行文本编辑器,浏览器等;

    • 控制终端

      前台进程与控制终端相关联,当用户在终端输入命令时这些命令会发送给前台进程;

    • 信号响应

      前台进程通常响应特点的信号,如按下Ctrl + c发送的SIGINT信号将终止前台进程;

  • 后台进程

    后台进程是在后台执行的进程,不直接与用户交互;

    这些进程通常执行长时间运行的任务,不需要用户的实时干预;

    • 非交互性

      后台进程不直接接受用户输入也不会将输出显示给用户;

      它们通常将输出重定向到文件或是日志;

    • 脱离终端

      后台进程不与控制终端直接关联,因此不会阻塞终端,允许用户继续在终端中执行其他操作;

    • 启动方式

      后台进程通常可以通过在运行命令后加上&符号启动,如:

      ./myprocess &
      
  • 前后台进程的转换

    将前台进程转换到后台:

    $ ./myprocess 
    I am a crazy Process , PID : 15708
    I am a crazy Process , PID : 15708
    ^Z
    [1]+  Stopped                 ./myprocess
    $ bg
    [1]+ ./myprocess &
    I am a crazy Process , PID : 15708
    $ I am a crazy Process , PID : 15708
    I am a crazy Process , PID : 15708
    • 暂停前台进程

      使用Ctrl + z将前台进程挂起;

    • 将挂起的进程放到后台

      bg命令;

    将后台进程转换到前台:

    $ ./myprocess &
    [1] 15720
    $ I am a crazy Process , PID : 15720
    I am a crazy Process , PID : 15720
    fI am a crazy Process , PID : 15720
    g I am a crazy Process , PID : 15720
    %I am a crazy Process , PID : 15720
    1
    ./myprocess
    I am a crazy Process , PID : 15720
    I am a crazy Process , PID : 15720
    ^C
    
    • 使用fg命令

      $ fg %1
      

      这个命令将后台进程编号为1(非进程PID)的进程调回前台;

Linux中,一次登录中一个终端一般会配上一个bash;

每一个登录只允许一个进程是前台进程,可以允许存在多个后台进程;

当未启动其他前台进程时bash将作为一个前台进程,负责与用户交互;

当用户从键盘中输入数据时将被前台进程获取;


信号的本质

请添加图片描述

使用命令kill -l可以查看当前操作系统中可用的信号;

$ kill -l1) SIGHUP	 2) SIGINT	 3) SIGQUIT	 4) SIGILL	 5) SIGTRAP6) SIGABRT	 7) SIGBUS	 8) SIGFPE	 9) SIGKILL	10) SIGUSR1
11) SIGSEGV	12) SIGUSR2	13) SIGPIPE	14) SIGALRM	15) SIGTERM
16) SIGSTKFLT	17) SIGCHLD	18) SIGCONT	19) SIGSTOP	20) SIGTSTP
21) SIGTTIN	22) SIGTTOU	23) SIGURG	24) SIGXCPU	25) SIGXFSZ
26) SIGVTALRM	27) SIGPROF	28) SIGWINCH	29) SIGIO	30) SIGPWR
31) SIGSYS	34) SIGRTMIN	35) SIGRTMIN+1	36) SIGRTMIN+2	37) SIGRTMIN+3
38) SIGRTMIN+4	39) SIGRTMIN+5	40) SIGRTMIN+6	41) SIGRTMIN+7	42) SIGRTMIN+8
43) SIGRTMIN+9	44) SIGRTMIN+10	45) SIGRTMIN+11	46) SIGRTMIN+12	47) SIGRTMIN+13
48) SIGRTMIN+14	49) SIGRTMIN+15	50) SIGRTMAX-14	51) SIGRTMAX-13	52) SIGRTMAX-12
53) SIGRTMAX-11	54) SIGRTMAX-10	55) SIGRTMAX-9	56) SIGRTMAX-8	57) SIGRTMAX-7
58) SIGRTMAX-6	59) SIGRTMAX-5	60) SIGRTMAX-4	61) SIGRTMAX-3	62) SIGRTMAX-2
63) SIGRTMAX-1	64) SIGRTMAX	

其中1 - 31号信号被称为 普通信号 ,34 - 64号信号被称为 实时信号 (不考虑实时信号);

信号列表中32号和33号信号不存在,原因为其并非标准的POSIX信号;

信号本质在内核中通过宏定义,通常可在<signal.h>,<asm/signal.h><linux/signal.h>等头文件中找到;

#define SIGHUP    1  /* Hangup (POSIX).  */
#define SIGINT    2  /* Interrupt (ANSI).  */
#define SIGQUIT   3  /* Quit (POSIX).  */
#define SIGILL    4  /* Illegal instruction (ANSI).  */
#define SIGABRT   6  /* Abort (ANSI).  */
#define SIGFPE    8  /* Floating-point exception (ANSI).  */
#define SIGKILL   9  /* Kill, unblockable (POSIX).  */
#define SIGSEGV   11 /* Segmentation violation (ANSI).  */
#define SIGPIPE   13 /* Broken pipe (POSIX).  */
#define SIGALRM   14 /* Alarm clock (POSIX).  */
#define SIGTERM   15 /* Termination (ANSI).  */
// ...

信号的处理方式一般分为三种:

  • 默认动作

    进程收到信号后将执行默认的动作,默认动作属于进程内置功能的一部分;

  • 忽略

    可调用系统调用接口signal()传递SIG_IGN参数来忽略信号(9号信号SIGKILL19号信号SIGSTOP无法被忽略);

    在进行忽略时不需要捕获信号,会直接进行忽略,这意味着捕获和忽略是完全不同的处理方式;

  • 自定义动作

    可调用系统调用接口signal()自定义对应的动作并设置信号的捕捉,使得进程在收到信号时作对应动作(9号信号SIGKILL19号信号SIGSTOP无法被捕捉);

处理方式只能三选一, 不能既…又… ;

本质上在使用Ctrl + C终止一个前台进程是向该前台进程发送了2号信号即SIGINT信号,可通过系统调用接口signal()使用自定义动作进行验证;

  • signal()

    NAMEsignal - ANSI C signal handlingSYNOPSIS#include <signal.h>typedef void (*sighandler_t)(int);sighandler_t signal(int signum, sighandler_t handler);RETURN VALUEsignal()  returns  the  previous  value  of  the signal handler, or SIG_ERR onerror.  In the event of an error, errno is set to indicate the cause.
    

    当接口调用成功时将返回先前信号处理程序的指针,指针的情况为如下:

    • 自定义处理程序

      将返回一个有效的函数指针指向先前设置的处理程序;

    • SIG_IGN

      表示信号先前被忽略;

    • SIG_DFL

      表示信号先前有默认处理程序;

    当调用失败时返回SIG_ERR表示一个错误并且errno被设置以指示错误原因;

    调用参数如下:

    • int signum

      传入一个参数代表需要操作的信号编号;

    • sighandler_t handler

      选项操作,传入一个函数指针,这个函数指针可以是用户自定义的函数指针(自定义操作),也可以是SIG_IGNSIG_DFL;

      当用户选择自定义操作时需要自行构造一个函数用于自定义动作,自定义动作所用函数必须传入一个参数来接收所捕获的信号编号;

在原有的myprocess.cc文件中进行修改,调用signal()系统调用接口设置信号捕获与自定义动作;

#include <signal.h>
#include <sys/types.h>
#include <unistd.h>#include <iostream>
using namespace std;// 自定义动作
void signal_handler(int signum) { cout << "get a signal : " << signum << endl; }int main() {signal(SIGINT, signal_handler);while (1) {cout << "I am a crazy Process , PID : " << getpid() << endl;sleep(1);}return 0;
}

该操作为两步:

  • 设置信号处理程序

    调用signal(SIGINT, signal_handler)设置信号处理,该操作是一个同步操作;

  • 异步信号处理

    当运行程序时用户按下Ctrl + C发送SIGINT信号将立即中断当前的执行并调用signal_handler函数然后恢复执行循环;

这意味着同一个信号的一种操作的捕获只需要设置一次,如本次操作需要设置捕获到信号2的行为为执行自定义动作;

对应结果为:

 ./myprocess 
I am a crazy Process , PID : 15850
I am a crazy Process , PID : 15850
^Cget a signal : 2
I am a crazy Process , PID : 15850
I am a crazy Process , PID : 15850
^Cget a signal : 2
I am a crazy Process , PID : 15850
I am a crazy Process , PID : 15850
^Cget a signal : 2
I am a crazy Process , PID : 15850
I am a crazy Process , PID : 15850
I am a crazy Process , PID : 15850
...

使用Ctrl + C将为前台进程发送信号2SIGINT,进程捕获到信号执行自定义动作打印对应信号编号;

  • 不可被忽略与捕获的信号

    在所有普通信号中9号信号和19号信号不可被捕获与忽略,其中9号信号SIGKILL19号信号SIGSTOP分别作用为:

    • SIGKILL

      强制终止进程,主要作用是立即停止进程的执行并将其从系统重移除;

    • SIGSTOP

      暂停进程;

      SIGSTOP信号会让进程进入停滞状态,直到接收到继续执行的信号,如SIGCONT;

    本质原因为防止不可杀死进程与无法暂停进程的恶意行为,可循环调用系统调用接口signal()进行验证;

    #include <signal.h>
    #include <sys/types.h>
    #include <unistd.h>#include <iostream>
    using namespace std;void signal_handler(int signum) { cout << "get a signal : " << signum << endl; }int main() {for (int i = 1; i <= 31; ++i) {signal(i, signal_handler);}while (1) {cout << "I am a crazy Process , PID : " << getpid() << endl;sleep(1);}return 0;
    }
    

    循环调用将131号信号都捕获并执行自定义动作;

    同时使用shell脚本:

    while :; do for i in $(seq 1 31); do  echo "Sending signal $i to process $TARGET_PID";kill -$i (pid);sleep 1;done; sleep 50;
    done
    

    来观察对应结果;

    其结果为:

    • bash命令行

      $ while :; do for i in $(seq 1 31); do echo "Sending signal $i to process $17342" ;kill -$i 17342; sleep 1; done; sleep 50; done
      Sending signal 1 to process 17342
      Sending signal 2 to process 17342
      Sending signal 3 to process 17342
      Sending signal 4 to process 17342
      Sending signal 5 to process 17342
      Sending signal 6 to process 17342
      Sending signal 7 to process 17342
      Sending signal 8 to process 17342
      Sending signal 9 to process 17342
      Sending signal 10 to process 17342
      -bash: kill: (17342) - No such process
      ^C
      $ while :; do for i in $(seq 10 31); do echo "Sending signal $i to process $17355" ;kill -$i 17355; sleep 1; done; sleep 50; done
      Sending signal 10 to process 17355
      Sending signal 11 to process 17355
      Sending signal 12 to process 17355
      Sending signal 13 to process 17355
      Sending signal 14 to process 17355
      Sending signal 15 to process 17355
      Sending signal 16 to process 17355
      Sending signal 17 to process 17355
      Sending signal 18 to process 17355
      Sending signal 19 to process 17355
      ^C
      $ kill -9 17355
      $ while :; do for i in $(seq 20 31); do echo "Sending signal $i to process $17367" ;kill -$i 17367; sleep 1; done; sleep 50; done
      Sending signal 20 to process 17367
      Sending signal 21 to process 17367
      Sending signal 22 to process 17367
      Sending signal 23 to process 17367
      Sending signal 24 to process 17367
      Sending signal 25 to process 17367
      Sending signal 26 to process 17367
      Sending signal 27 to process 17367
      Sending signal 28 to process 17367
      Sending signal 29 to process 17367
      Sending signal 30 to process 17367
      Sending signal 31 to process 17367
      ^C
      
    • myprocess所在窗口

      $ ./myprocess
      get a signal : 1
      I am a crazy Process , PID : 17342
      get a signal : 2
      I am a crazy Process , PID : 17342
      get a signal : 3
      I am a crazy Process , PID : 17342
      get a signal : 4
      I am a crazy Process , PID : 17342
      get a signal : 5
      I am a crazy Process , PID : 17342
      get a signal : 6
      I am a crazy Process , PID : 17342
      get a signal : 7
      I am a crazy Process , PID : 17342
      get a signal : 8
      I am a crazy Process , PID : 17342
      Killed  # 被9号信号杀死$ ./myprocess 
      I am a crazy Process , PID : 17355
      get a signal : 10
      I am a crazy Process , PID : 17355
      get a signal : 11
      I am a crazy Process , PID : 17355
      get a signal : 12
      I am a crazy Process , PID : 17355
      get a signal : 13
      I am a crazy Process , PID : 17355
      get a signal : 14
      I am a crazy Process , PID : 17355
      get a signal : 15
      I am a crazy Process , PID : 17355
      get a signal : 16
      I am a crazy Process , PID : 17355
      get a signal : 17
      I am a crazy Process , PID : 17355
      get a signal : 18
      I am a crazy Process , PID : 17355[1]+  Stopped                 ./myprocess # 被19号信号暂停$ ./myprocess
      I am a crazy Process , PID : 17367
      get a signal : 20
      I am a crazy Process , PID : 17367
      get a signal : 21
      I am a crazy Process , PID : 17367
      get a signal : 22
      I am a crazy Process , PID : 17367
      get a signal : 23
      I am a crazy Process , PID : 17367
      get a signal : 24
      I am a crazy Process , PID : 17367
      get a signal : 25
      I am a crazy Process , PID : 17367
      get a signal : 26
      I am a crazy Process , PID : 17367
      get a signal : 27
      I am a crazy Process , PID : 17367
      get a signal : 28
      I am a crazy Process , PID : 17367
      get a signal : 29
      I am a crazy Process , PID : 17367
      get a signal : 30
      I am a crazy Process , PID : 17367
      get a signal : 31
      

硬件理解

请添加图片描述

当用户按下键盘Ctrl + C时将默认转化为信号2,即SIGINT信号;

通常涉及到以下步骤:

  • 键盘输入

    键盘中每个按键都将有一个唯一的键码;

    当用户按下Ctrl + C组合键时,键盘硬件将会生成一个键码,键盘控制器将负责将按键的物理动作转化为键码;

  • 键盘硬件中断

    键盘控制器将检测到按键事件并向CPU发送一个硬件中断请求;

    在发送中断请求的同时,键盘控制器还会将键码放置在其缓冲区中;

    • 普通按键

      当用户按下普通按键(字母,数字等)时,操作系统将会将扫描码转化为ASCII码或其他字符编码并放入键盘缓冲区;

      如果按下普通按键时没有按下任何控制键,操作系统将会视其为普通按键进行输入;

    • 控制按键组合

      如果用户按下了一个普通按键同时按下了一个或多个控制按键(Ctrl,Alt,Shift等),操作系统会检查当前的控制按键状态标志并将按键组合视为特殊的控制输入;

      如按下Ctrl + C会被识别为一个特定的控制字符而不是单独的CtrlC;

  • CPU中断处理

    收到键盘中断请求后CPU将会暂停当前执行的任务并通过中断向量表查找对应的中断服务例程;

    中断向量表是一个包含中断服务例程地址的表,通常存储在内存的特定位置;

  • 中断服务例程

    键盘的中断服务例程是由操作系统提供的处理函数,负责从键盘控制器中读取键码并将其转化为ASCII码或其他合适的格式;

    这些键码通常会被放入一个键盘缓冲区一遍操作系统读取和处理;

  • 操作系统处理键盘输入并生成SIGINT信号

    操作系统内核将定期检查键盘缓冲区并处理其中的键码;

    当检测到Ctrl + C组合键时将是识别出这是一个产生SIGINT信号的控制字符并将该信号发送给当前的前台进程组;

  • 信号接收和处理信号

    进程接收到SIGINT信号将根据预先定义的信号处理机制处理信号;


信号的产生

请添加图片描述

信号的产生可以源于多种情况:

  • 用户输入

    • SIGINT

      用户通过键盘按键组合Ctrl + C产生,作用为中断当前前台进程;

    • SIGQUIT

      用户通过键盘按键组合Ctrl + \产生,作用为强制终止进程并生成核心转储文件;

    • SIGTSTP

      用户通过键盘按键组合Ctrl + Z产生,作用为暂停(停止)当前前台进程,将其放入后台;

  • kill -signo pid

    通过kill命令带信号编号及进程PID产生信号;

  • 系统调用接口

    • kill()

      可通过kill()函数调用产生信号;

      NAMEkill - send signal to a processSYNOPSIS#include <sys/types.h>#include <signal.h>int kill(pid_t pid, int sig);Feature Test Macro Requirements for glibc (see feature_test_macros(7)):kill(): _POSIX_C_SOURCE >= 1 || _XOPEN_SOURCE || _POSIX_SOURCERETURN VALUEOn  success (at least one signal was sent), zero is returned.  On error, -1 isreturned, and errno is set appropriately.

      当函数成功调用(至少发送一个信号)返回0,失败则返回-1并设置errno;

      其中参数pid_t pidint sig分别为进程的PID和信号的信号编号;

      可用该接口模拟实现一个kill命令;

      #include <signal.h>
      #include <sys/types.h>
      #include <unistd.h>#include <iostream>
      #include <string>
      using namespace std;// 运行用法为
      /*./mykill signum pid0       1      2一共3个 argc = 3
      */void Usage(string proc) {// 使用手册cout << "usage:\n\t" << proc << " signum pid" << endl << endl;
      }int main(int argc, char* argv[]) {if (argc != 3) {Usage(argv[0]);exit(3);}int signum = stoi(argv[1]);int pid = stoi(argv[2]);kill(pid, signum);return 0;
      }
      
    • raise()

      通过调用raise()接口向调用者发送一个信号;

      NAMEraise - send a signal to the callerSYNOPSIS#include <signal.h>int raise(int sig);DESCRIPTIONThe  raise()  function  sends a signal to the calling process or thread.  In asingle-threaded program it is equivalent tokill(getpid(), sig);RETURN VALUEraise() returns 0 on success, and nonzero for failure.

      函数调用成功时返回0,调用失败时返回!0;

      参数int sig表示需要发送的信号编号;

      验证:

      #include <signal.h>
      #include <sys/types.h>
      #include <unistd.h>#include <iostream>
      using namespace std;void signal_handler(int signum) { cout << "get a signal : " << signum << endl; }int main() {signal(SIGINT, signal_handler);int n = 4;while (n--) {cout << "I am a crazy Process , PID : " << getpid() << endl;sleep(1);}raise(2);return 0;
      }
      

      当进程执行4s后运行raise(2)向自己发送一个信号,并调用signal()设置捕获;

      $ ./myprocess 
      I am a crazy Process , PID : 18003
      I am a crazy Process , PID : 18003
      I am a crazy Process , PID : 18003
      I am a crazy Process , PID : 18003
      get a signal : 2
      

      结果为发送了信号2并被捕获;

      本质上raise()函数是kill(getpid(), sig)的封装;

    • abort()

      NAMEabort - cause abnormal process terminationSYNOPSIS#include <stdlib.h>void abort(void);DESCRIPTIONThe abort() first unblocks the SIGABRT signal, and then raises that signal forthe calling process.  This results in the abnormal termination of the  processunless  the  SIGABRT  signal  is caught and the signal handler does not return(see longjmp(3)).If the abort() function causes  process  termination,  all  open  streams  areclosed and flushed.If  the  SIGABRT  signal  is ignored, or caught by a handler that returns, theabort() function will still terminate the process.  It does this by  restoringthe  default  disposition for SIGABRT and then raising the signal for a secondtime.RETURN VALUEThe abort() function never returns.
      

      通过调用abort()函数向进程发送一个6号信号SIGABRT信号,并强行终止进程;

      #include <signal.h>
      #include <sys/types.h>
      #include <unistd.h>
      #include <cstdlib>
      #include <iostream>
      using namespace std;void signal_handler(int signum) { cout << "get a signal : " << signum << endl; }int main() {signal(SIGABRT, signal_handler);// 6号信号可被捕获与忽略int n = 0;while (++n) {cout << "I am a crazy Process , PID : " << getpid() << endl;sleep(1);if (n % 5 == 0) {abort();}}return 0;
      }
      

      自定义动作中并未涉及进程退出;

      5s过后将调用abort()向进程发送一个SIGABRT信号,并使得signal()捕获到该信号并执行自定义动作;

      $ ./myprocess 
      I am a crazy Process , PID : 18218
      I am a crazy Process , PID : 18218
      I am a crazy Process , PID : 18218
      I am a crazy Process , PID : 18218
      I am a crazy Process , PID : 18218
      get a signal : 6
      Aborted
      $ 
      

      运行结果表示信号被捕获但进程仍被终止;

      实际上abort()的封装过程中封装了_exit();

      /* 可能的abort()实现 (简化版本) */
      void abort(void) {// 发送 SIGABRT 信号raise(SIGABRT);// 如果 SIGABRT 信号被捕获,并且处理函数返回,则直接退出_exit(1); // 使用 _exit 直接退出,避免调用清理函数
      }
      

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

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

相关文章

DevExpress中文教程 - 如何在.NET MAUI应用中实现Material Design 3?

DevExpress .NET MAUI多平台应用UI组件库提供了用于Android和iOS移动开发的高性能UI组件&#xff0c;该组件库包括数据网格、图表、调度程序、数据编辑器、CollectionView和选项卡组件等。 获取DevExpress v24.1正式版下载 Material Design是一个由Google开发的跨平台指南系统…

MySQL_CRUD

目录 一、新增 (Create) 1.1 全列插入 1.2 指定列插入 二、查询 (Retrieve) 2.1 全列查询 2.2 指定列查询 2.3 查询字段为表达式 2.4 别名 2.5 去重&#xff1a;distinct 2.6 排序&#xff1a;order by 2.7 条件查询&#xff1a;where 2.8 分页查询&#xff1a;lim…

安装Ubuntu24.04服务器版本

Ubuntu系统安装 一.启动安装程序二.执行 Ubuntu Server 安装向导1.选择安装程序语言&#xff0c;通常选择「English」2.设置键盘布局&#xff0c;默认「English US」即可3.选择安装方式 三.配置网络1.按Tab键选择网络接口&#xff08;例如 ens160&#xff09;&#xff0c;然后按…

项目实战二 HIS项目

目标&#xff1a; 项目的操作流程&#xff1a; 开发体系 前端开发&#xff1a;负责页面的编写 HTML CSS JavaScript 后端开发&#xff1a;看不到 摸不着的功能 常用开发语言 PHP JAVA Python 框架 &#xff1a; 半成品 做好的功能模块 版本控制 Git 分布式版本控…

乐理基础知识

为了学习无源蜂鸣器播放音乐&#xff0c;我去学习了乐理知识&#xff0c;发现只要把握了音调和音值&#xff0c;也不算太难&#xff0c;我整理了笔记&#xff0c;现在分享出来 声音 声音是由物体振动产生的声波。 其主要特征如下&#xff1a; 1.音调指声音的高低&#xff0…

数据编织 VS 数据仓库 VS 数据湖

目录 1. 什么是数据编织?2. 数据编织的工作原理3. 代码示例4. 数据编织的优势5. 应用场景6. 数据编织 vs 数据仓库6.1 数据存储方式6.2 数据更新和实时性6.3 灵活性和可扩展性6.4 查询性能6.5 数据治理和一致性6.6 适用场景6.7 代码示例比较 7. 数据编织 vs 数据湖7.1 数据存储…

分享一个最近在进行前后端联调时改了2天的bug...

场景再现 我们这边前端端口是8080 后端端口是8121 我们在前端里在首页面写了一个任务 当进入网页三秒后 发起一个叫getLoginUser的请求 我们的getLoginUser是调用的这里 一个异步请求 这边我们前端调用后端的接口也已经写好 我们先把后端跑起来 访问前端页面 接收到了这个…

Air780EP-AT开发-HTTP应用指南

简介 关联文档和使用工具&#xff1a; AT固件获取AT指令手册 概述 4G模块支持HTTP和HTTPS协议&#xff0c; HTTP应用的基本流程如下&#xff1a; 1、激活PDP&#xff08;参考&#xff1a;http://oldask.openluat.com/article/937&#xff09;2、初始化HTTP服务3、设置HTTP会话…

Http 和 Https 的区别(图文详解)

在现代网络通信中&#xff0c;保护数据的安全性和用户的隐私是至关重要的。HTTP&#xff08;Hypertext Transfer Protocol&#xff09;和 HTTPS&#xff08;Hypertext Transfer Protocol Secure&#xff09;是两种常见的网络通信协议&#xff0c;但它们在数据保护方面的能力存在…

剧本杀小程序搭建,互联网下的游戏新体验,实现新增收!

近几年&#xff0c;桌游备受大众青睐&#xff0c;剧本杀行业更是瞬间曝火&#xff01;拥有强大社交体验与沉浸式游戏体验的剧本杀成为了众多年轻人的新宠&#xff0c;无论是外出游玩还是好友聚会&#xff0c;剧本杀游戏都成为了首选方式。 随着互联网的发展&#xff0c;线上小…

基于DPUSmartNic的云原生SDN解决方案

1. 方案背景与挑战 随着云计算&#xff0c;大数据和人工智能等技术的蓬勃发展&#xff0c;数据中心面临着前所未有的数据洪流和计算压力&#xff0c;这对SDN提出了更高的性能和效率要求。自云原生概念被提出以来&#xff0c;Kubernetes为云原生应用的落地提供了一个轻量级&am…

视频汇聚平台EasyCVR启动出现报错“cannot open shared object file”的原因排查与解决

安防视频监控EasyCVR安防监控视频系统采用先进的网络传输技术&#xff0c;支持高清视频的接入和传输&#xff0c;能够满足大规模、高并发的远程监控需求。EasyCVR平台支持多种视频流的外部分发&#xff0c;如RTMP、RTSP、HTTP-FLV、WebSocket-FLV、HLS、WebRTC、fmp4等&#xf…

物联网mqtt网关搭建背后的技术原理

前言 物联网是现在比较热门的软件领域&#xff0c;众多物联网厂商都有自己的物联网平台&#xff0c;而物联网平台其中一个核心的模块就是Mqtt网关。这篇文章的目的是手把手教大家写书写一个mqtt网关&#xff0c;后端存储支持Kafka/Pulsar&#xff0c;支持mqtt 连接、断链、发送…

【Java】中的List集合

目录 一、什么是List集合二、List的常用方法List的初始化元素操作1.添加元素2.删除元素3.修改元素4.查询元素 三、List集合的遍历1.for循环遍历2.增强for循环3.迭代器遍历 一、什么是List集合 List集合是最常用的一种数据结构之一。它具有动态扩容、元素添加、删除和查询等基础…

MongoDB教程(十八):MongoDB MapReduce

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝你生活愉快&#xff01; 文章目录 引言一、MapRed…

免费【2024】springboot 趵突泉景区的智慧导游小程序

博主介绍&#xff1a;✌CSDN新星计划导师、Java领域优质创作者、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行前辈交流✌ 技术范围&#xff1a;SpringBoot、Vue、SSM、HTML、Jsp、PHP、Nodejs、Python、爬虫、数据可视化…

[数据集][目标检测]躺坐站识别检测数据集VOC+YOLO格式9488张3类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;9488 标注数量(xml文件个数)&#xff1a;9488 标注数量(txt文件个数)&#xff1a;9488 标注…

bug诞生记——动态库加载错乱导致程序执行异常

大纲 背景问题发生问题猜测和分析过程是不是编译了本工程中的其他代码是不是有缓存是不是编译了非本工程的文件是不是调用了其他可执行文件查看CMakefiles分析源码检查正在运行程序的动态库 解决方案 这个案例发生在我研究ROS 2的测试Demo时发生的。 整体现象是&#xff1a;修改…

站在资本投资领域如何看待分布式光纤传感行业?

近年来&#xff0c;资本投资领域对于分布式光纤传感行业并不十分敏感。这主要是由于分布式光纤传感技术是一个专业且小众的领域&#xff0c;其生命周期相对较长&#xff0c;缺乏爆发性&#xff0c;与消费品或商业模式创新产业有所不同。此外&#xff0c;国内的投资环境也是影响…

Jmeter之count函数

counter函数 1、功能解释 count函数--计数器&#xff0c;每调用这个函数一次&#xff0c;它就会自动加1。它有两个参数&#xff0c;第一个参数是布尔型的&#xff0c;只能设置成 “TRUE”或者“FALSE”&#xff0c;如果是TRUE&#xff0c;那么每个用户有自己的计数器&#xf…