当前位置: 首页 > news >正文

【Linux】深入理解进程信号机制:信号的产生、捕获与阻塞

52bc67966cad45eda96494d9b411954d.png

🎬 个人主页:谁在夜里看海.

📖 个人专栏:《C++系列》《Linux系列》《算法系列》

⛰️ 时间不语,却回答了所有问题


目录

📚前言

📚一、信号的本质

📖1.异步通信

📖2.信号列表

📖3.信号发送的本质

📚二、信号的产生

📖1.终端按键

🔖core dump

📖2.系统调用

🔖kill()

🔖raise()

🔖abort()

🔖alarm()

📚三、信号的捕获

📖1.signal信号处理

🔖函数原型

🔖示例

📖2.sigaction信号处理

🔖函数原型

🔖示例 

📚四、信号的阻塞与未决

📖1.阻塞的本质

🔖未决信号

🔖阻塞信号

⚠️阻塞≠忽略

📖2.阻塞的应用

🔖函数原型

🔖示例

📚五、可重入函数

📖1.定义

🔖特性

🔖示例

📖2.volatile关键字

🔖-O1优化 


📚前言

上一章我们谈论了进程间的通信机制,由于不同进程之间有时需要相互协作,所以引入了进程间通信的概念,使得不同的进程可以共享同一份资源,完成数据的传输。进程间通信时是用户层面的通信,为什么这么说呢,因为在这个过程中,用户空间的进程通过特定的通信机制来传递数据或信息,而这些操作并不涉及操作系统的内核执行逻辑,通信的内容也是由用户自定义的。

然而系统层面也需要通信,比如一个进程执行异常时,操作系统必须对异常进程进行处理,那么有哪些处理方式呢?

✅①:直接“杀死”该进程,将进程从CPU运行队列中移除,但是存在隐患,比如文件还没关闭,资源还未释放,有内存泄漏的风险;

✅②:告知进程退出,让进程主动退出,这样可以保证资源被释放。

系统告知进程退出,不就是系统与进程间的通信嘛,这一类通信不需要用户层的参与,在操作系统底层完成,并且通信的内容也不由用户定义,而是由系统定义,这种通信方式就叫做信号

📚一、信号的本质

📖1.异步通信

在用户层的通信中,比如管道通信,我们可以明确通信何时发生,数据何时进行交换吗?答案是可以,管道通信是借助一个管道文件完成的,进程需要对管道文件进行打开和关闭操作,而这些操作在执行流中是确定的,也就是说,用户规定了进程在何时发送数据,何时接收数据,我们称这种通信为同步通信

然而在系统层的通信中,我们可以明确通信何时发生吗,就比如程序执行异常时,需要发送信号告知程序终止,我们能确定程序何时异常吗,并不能确定(如果能确定的话,我们就完全可以避免异常的发生了,就不会引发异常了),我们称这种通信为异步通信

📖2.信号列表

❓为什么操作系统需要与进程进行通信呢?

✅通常,进程和操作系统之间的通信发生在一些特殊情况下,而不是无缘无故的通信。这些特殊情况往往是有限的,比如异常、外部中断、资源不足等。针对这些情况,操作系统有相应的处理方法,例如遇到异常时,操作系统会终止进程;当外部中断发生时,操作系统会中断进程的执行并进行相应处理。因此,操作系统只需将所有可能出现的特殊情况及其对应的处理办法保存在一个集合中,当进程遇到某种情况时,操作系统就发送相应的信号进行处理,这个集合被称为信号列表。

通过kill -l命令可以查看系统定义的信号列表:

我们可以看到,每个信号都有一个编号以及宏定义名称,这些宏定义可以signal.h中找到。其中编号32号以下的信号是标准信号,它们用于处理进程中的常见情况,如进程终止、异常、退出等;32号以上的信号是是实时信号,通常由用户定义并用于更精细的进程控制。

📖3.信号发送的本质

在操作系统中,每一个信号都有一个对应的编号,这些编号的范围是1~64,所以在信号传输的过程中,我们并不需要传输具体的信号宏名称,只需要一个位图即可实现,比如传输SIGINT信号时,只需要将位图中第二位置为1,即可完成信号的传输。因此,每一个进程都会维护一个用于表示信号集的位图,每一位对应一个信号的状态。

理解了信号的本质,我们来谈谈信号的操作,即信号是如何产生的,进程对信号又是如何处理的

📚二、信号的产生

📖1.终端按键

其实,我们早就接触过信号了,平常我们写代码时,偶尔会遇到程序陷入死循环或卡住的情况,这个时候我们按住ctrl+c就可以退出进程,结束死循环,其实,按下ctrl+c的过程就是向进程发送一个SIGINT信号,用于通知进程中断当前操作,这就是第一种信号产生的方式:终端按键产生。

不止ctrl+c可以产生信号,其他按键也可以产生相应的信号,例如ctrl+\,用于发送SIGQUIT信号,该信号可以强制终止进程,这与SIGINT类似,但不同的是,SIGQUIT可以生成核心转储文件(core dump)以便于调试:

🔖core dump

Core dump(核心转储)是操作系统在进程崩溃时将该进程的内存内容保存到一个文件中的机制,能够为开发者提供有关崩溃时程序状态的详细信息。

⚠️触发core dump时,系统会向磁盘写入一个核心转储文件,用于保存程序崩溃的完整快照,但是并不是所有程序崩溃的场景都需要一个快照,因为这样会导致磁盘空间被大量占用,当磁盘空间被大量占用时,可能会引发比程序崩溃更严重的问题,所以默认情况下,禁用core dump文件

然而可以通过ulimit命令接触限制:

① 使用 ulimit -c 命令查看当前系统的核心转储限制。如果输出为 0,则表示禁用了核心转储。

② 通过 ulimit -c unlimited 命令可以取消核心转储大小限制,允许生成核心转储文件。

📖2.系统调用

🔖kill()

通常情况下,我们是通过系统调用 kill 来实现信号的发送的,例如:

kill -SIGINT (进程pid)
# 或者
kill -2 (进程pid)

这种方式等价与按键ctrl+c,都是向指定进程发送SIGINT信号。

除了在终端输入kill指令,我们还可以在函数内部调用kill()函数:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>int main() {pid_t pid = getpid();  // 获取当前进程IDprintf("Sending SIGINT signal to process %d\n", pid);// 向当前进程发送 SIGINT 信号if (kill(pid, SIGINT) == -1) {perror("Error sending signal");}// 除了向当前进行发送,还可以向其他进程发送 SIGINT 信号,只要知道它的pidreturn 0;
}

❓为什么是kill?

kill是“杀死”的意思,为什么发送信号的指令要取名为“杀死”呢?

✅这大概是因为,在信号被设计之初,其作用主要就是“杀死”一个进程,所以当时直接将信号发送取名为kill,到了后面,信号的种类越来越多,但是kill这个指令名称被保留了下来,沿用至今。但是不要因为指令叫做kill就认为它只能“杀死”进程,实际上它用于发送多种不同类型的信号。

除了kill()之外,其他一些与信号相关的系统调用和库函数也能触发信号的发送或控制信号的行为。 

🔖raise()

raise()函数用于向当前进程发送信号(自己给自己发信号):

#include <signal.h>
int raise(int sig);

sig表示要发送的信号。

🔖abort()

abort()函数也可以向当前进程发送信号,但与raise不同的是,它只能发送SIGABRT信号,因此函数内部不包含任何参数:

#include <stdio.h>
#include <stdlib.h>int main() {printf("Before abort\n");abort();  // 终止进程并生成核心转储printf("This line will never be executed.\n");return 0;
}
🔖alarm()

alarm()函数的作用是在指定时间之后发送一个SIGALRM信号:

#include <unistd.h>unsigned int alarm(unsigned int seconds);

参数

        seconds:设置定时器的时间,单位为秒,当 alarm() 被调用时,系统会在 seconds 秒后向当前进程发送 SIGALRM 信号。

返回值

        如果成功设置定时器,它返回剩余的时间。如果定时器已经存在,返回值是该定时器剩余的时间。

示例

#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>int main()
{alarm(5);while(1){cout<<"I am a process..."<<endl;sleep(1);}return 0;}

📚三、信号的捕获

进程对于信号的处理有以下三种方式:

忽略信号:进程可以选择忽略某个信号,即在接收该信号时,不执行任何操作;

处理信号:进程可以指定一个信号处理函数来处理指定信号;

默认处理:如果没有指定默认的信号处理函数,操作系统会采用该信号的默认行为。

📖1.signal信号处理

为了捕捉并处理信号,我们通常调用signal()函数,通过该函数,我们可以自定义信号处理行为。

🔖函数原型
#include <signal.h>typedef void (*sighandler_t)(int);sighandler_t signal(int signum, sighandler_t handler);

signum是我们要捕获的信号编号;

对于handler参数,我们有两种选项:

采用关键字:SIG_IGN,表示当前信号被忽略;SIG_DFL,表示采用默认处理方式;

传入自定义信号处理函数,函数内部带有一个int型参数,表示信号编号

🔖示例
#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>void sigcb(int sig)
{if(sig == 2)cout<<"catch a sig: "<<sig<<endl;
}int main()
{signal(2,sigcb); // 调用自定义的信号处理函数cout<<"This is a process,pid is: "<<getpid()<<endl;while(1);return 0;
}

此时通过ctrl+c发生SIGINT信号时程序执行自定义行为:

此时我们将signal的第二个参数修改成SIG_IGN,表示忽略当前信号:

#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>int main()
{signal(2,SIG_IGN);cout<<"This is a process,pid is: "<<getpid()<<endl;while(1);return 0;
}

此时输入ctrl+c,SIGINT信号被忽略,不执行任何行为:

📖2.sigaction信号处理

sigaction函数不仅可以指定信号处理程序,还可以设置额外的选项来控制信号处理的细节。

🔖函数原型
#include <signal.h>int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

参数:

① signum: 指定要操作的信号编号;

② act:指向一个struct sigaction 类型的结构体,用于定义信号处理行为,可设置为NULL;

③ oldact:保存之前的信号处理行为(结构体),可设置为NULL。

struct sigaction:

该结构体用于定义信号处理行为:

struct sigaction {void     (*sa_handler)(int);     // 信号处理函数,或以下宏之一:// SIG_DFL:使用默认行为// SIG_IGN:忽略信号void     (*sa_sigaction)(int, siginfo_t *, void *); // 备用信号处理函数sigset_t sa_mask;                // 在处理此信号时需要阻塞的其他信号集合int      sa_flags;               // 信号处理的标志void     (*sa_restorer)(void);   // 已废弃
};

结构体成员:

① sa_handler:指定信号处理函数(简单处理);

② sa_sigaction:用于处理更复杂信号信息(需要设置 SA_SIGINFO 标志);

③ sa_mask:在处理信号时需要阻塞的其他信号;

④ sa_flags:控制信号处理行为的选项。常用值:

        SA_RESTART:自动重启被信号中断的系统调用

        SA_SIGINFO:启用 sa_sigaction,传递更多信号信息

        SA_NOCLDWAIT:阻止僵尸进程的生成(用于 SIGCHLD

        SA_NODEFER:在处理信号时不自动阻塞该信号

🔖示例 

设置对SIGINT信号的处理行为:

#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>void sigcb(int sig)
{cout<<"catch a sig: "<<sig<<endl;
}int main()
{struct sigaction sa;sa.sa_handler = sigcb;sigemptyset(&sa.sa_mask);sa.sa_flags = 0;sigaction(2,&sa,nullptr);while(1);return 0;
}

 

📚四、信号的阻塞与未决

然而,对于进程来说,并不是所有信号都需要立即响应的,因为在信号到达时,进程需要立即停止当前执行流,去执行信号处理操作。例如,一个进程的部分执行内容是对一个大文件进行拷贝写入,如果在写入的途中中断了,就会导致文件不完整甚至损坏,所以进程希望写入操作执行完毕后再响应信号,此时就需要信号阻塞操作。

📖1.阻塞的本质

我们前面提到,信号发送的本质是将目标进程的信号集(位图)中相应位置的标志位置为1,实际上,进程要维护两个这样信号集,一个为阻塞信号集,一个为未决信号集

🔖未决信号

当系统向进程发送信号时,修改的就是未决信号集中的指定标志位,在未决信号是指已经由系统发送但进程因为某些原因还未处理的信号,一个信号要想被进程处理的第一个条件就是在未决信号集(Pending)中标志位为1;

🔖阻塞信号

阻塞信号(Block)就是进程希望延迟执行的信号(例如在文件写入完毕之后再执行SIGINT信号,此时就需要先将SIGINT信号阻塞,写入完毕后再取消阻塞),未决信号想要被进程执行还需要判断是否阻塞,即在阻塞信号集中的标志位是否为1,为1则阻塞:

所以一个信号想要被执行需要满足:

未决位为1,表示信号已产生;

阻塞位为0,表示信号不被阻塞

③ 其他条件,例如进程空闲,没有在执行其他信号...

⚠️阻塞≠忽略

信号被阻塞与信号被忽略是两个不同的概念:

信号被阻塞表示信号不会被进程接收,如果产生则处于未决状态(Pending);

信号被忽略是进程对信号的一种处理方式,表示进程接收信号时忽略该信号,不做处理;

📖2.阻塞的应用

🔖函数原型

sigprocmask() 是一个用于设置或获取当前进程阻塞信号集的系统调用。它可以阻塞、解除阻塞或替换进程的阻塞信号集。

#include <signal.h>int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);

参数:

① how:指定对信号集的操作方式,可以取以下值:

        SIG_BLOCK:将 set 中的信号添加到当前阻塞信号集中;

        SIG_UNBLOCK:从当前阻塞信号集中移除 set 中的信号;

        SIG_SETMASK:将当前阻塞信号集替换set

② set:指向包含要阻塞或解除阻塞的信号集的指针。

③ oldset:存储之前的阻塞信号集,为NULL表示不作存储。

返回值:

成功返回 0;失败返回 -1,并设置errno。

除了上述函数,我们还需要一系列的信号集操作函数:

#include <signal.h>
// sigset_t set; // 调用下面函数前,需要先定义一个函数集变量
int sigemptyset(sigset_t *set);
int sigfillset(sigset_t *set);
int sigaddset (sigset_t *set, int signo);
int sigdelset(sigset_t *set, int signo);
int sigismember(const sigset_t *set, int signo);

下面对这些函数进行一一介绍:

① sigemptyset

初始化set所指向的信号集,使其中所有信号的对应bit清零,表示该信号集不包含任何有效信号

② sigfillset

初始化set所指向的信号集,使其中所有信号的对应bit置位,表示该信号集的有效信号包括系统支持的所有信号

③ sigaddset

向set指向的信号集中添加信号signo,即将signo对应的标志位置为1。

④ sigdelset

将set指向的信号集中的signo信号删除,即将signo对应的标志位置为0。

⑤ sigismember

判断set指向的信号集中是否存在信号signo,也可以理解成返回signo对应的标志位。

🔖示例

设置阻塞信号2与40,观察发生信号时的状态:

#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>void printsigset(sigset_t *set)
{for(int i=0;i<64;++i){if(sigismember(set,i))putchar('1');elseputchar('0');}cout<<endl;
}int main()
{cout<<"This is a process,pid is: "<<getpid()<<endl;sigset_t m,p;sigemptyset(&m);sigaddset(&m,2);sigaddset(&m,40);sigprocmask(SIG_BLOCK,&m,NULL);while(1){sigpending(&p);printsigset(&p);sleep(3);}return 0;
}

此时阻塞了信号2与40,并通过打印未决信号集,观察情况:

发现发送信号2和40后,未决信号集的第2位和第40位置为1,表示信号2和40为未决信号。

📚五、可重入函数

信号处理函数执行过程中可能被任意中断,因此信号处理函数需要使用可重入函数,以避免数据竞争、资源冲突和未定义行为。

📖1.定义

可重入函数是指函数在执行过程中可以被中断,并且在中断期间又可以安全地被调用,且两次调用互不干扰。

🔖特性

① 无全局变量的依赖: 函数内部不依赖于全局或静态变量,或者对这些变量的操作是线程安全的(原子性操作)

② 无静态数据的修改: 函数不能修改其内部的静态变量,因为多次调用可能导致状态不一致

③ 无不可重入函数调用: 函数只能调用其他可重入函数

④ 使用局部资源: 函数仅使用局部变量或参数,不会影响其他调用的状态。 

🔖示例

情况一:两个线程并发地执行以下代码,假设a是全局变量,初始为0,那么输出结果会是什么?

   void foo(){a=a+1;printf("%d ",a);}

✅答案:1 1 、1 2 、2 1 、2 2 都有可能

分析:

a=a+1并不是原子性操作,怎么理解呢?

原子性操作只有未执行执行完毕两种情况,而a=a+1的操作流程是:

①将a从全局数据区提取到寄存器中;②在寄存器中完成+1操作;③将a返回到全局数据区

线程之间是共享虚拟地址空间的(除了栈区,线程之间各自拥有一块栈区空间),所以它们在访问全局变量a的时候会产生竞争访问,它们可能同时将a提取到各自的寄存器中,也有可能进程1先提取,修改完毕返回后进程2才提取,此时进程2提取的就是已经经历过一次+1的a:

情况二:那如果此时a是局部变量呢?

✅答案:只会是 1 1。

因为线程之间各自拥有一块栈空间,a是局部变量的话,那么在两个线程之间各有一份,即线程会在各自的栈空间中提取a到寄存器并返回+1后的结果,不存在竞争访问问题,执行结果只会是1 1。

情况一的foo函数是不可重入函数,因为存在竞争访问;情况二的foo函数是可重入函数,因为没有使用全局变量(只存在于栈空间),不存在竞争访问。

📖2.volatile关键字

看下面代码:

#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>int flag = 0;
void handler(int sig)
{if(sig == 2){cout<<"change flag 0 to 1"<<endl;flag = 1;}
}int main()
{signal(2,handler);while(!flag);exit(0);
}

在标准情况下键入CTRL-C ,2号信号被捕捉,执行自定义动作,修改flag=1,while条件不满足,退出循 环,进程退出:

# makefile
test:volatile.cppg++ -o $@ $^ #标准情况.PHONY:clean
clean:rm -f test

但是在优化情况下,键入CTRL-C ,2号信号被捕捉,执行自定义动作,修改flag=1,但是while条件依旧满足,进程继续运行!

test:volatile.cppg++ -o $@ $^ -O1 #优化情况.PHONY:clean
clean:rm -f test

这是为什么呢?

🔖-O1优化 

-O1是编译器的一个优化级别,在当前级别下,编译器会对代码进行性能优化, 原本存储在全局数据区的变量flag被直接存储到了寄存器,signal函数调用中,对flag值的修改是在另一个寄存器中进行的,而while循环中对flag的判断还是在原先的寄存器中提取,导致signal函数中对flag的修改没有作用到while循环中,键入CTRL-C最终不会退出进程。

为了解决编译器优化带来的这一潜在问题,我们可以使用volatile关键字修饰flag的定义,表示flag不被优化,这样就保证了signal中对flag的修改可以同步到while循环中:

#include <iostream>
using namespace std;
#include <unistd.h>
#include <signal.h>volatile int flag = 0; // volatile修饰
void handler(int sig)
{if(sig == 2){cout<<"change flag 0 to 1"<<endl;flag = 1;}
}int main()
{signal(2,handler);while(!flag);exit(0);
}

在优化情况下,键入CTRL-C ,2号信号被捕捉,执行自定义动作,修改flag=1,while条件不满足,退出循 环,进程退出:

test:volatile.cppg++ -o $@ $^ -O1 #优化情况.PHONY:clean
clean:rm -f test


以上就是【文件操作的艺术——从基础到精通】的全部内容,欢迎指正~ 

码文不易,还请多多关注支持,这是我持续创作的最大动力!  

相关文章:

【Linux】深入理解进程信号机制:信号的产生、捕获与阻塞

&#x1f3ac; 个人主页&#xff1a;谁在夜里看海. &#x1f4d6; 个人专栏&#xff1a;《C系列》《Linux系列》《算法系列》 ⛰️ 时间不语&#xff0c;却回答了所有问题 目录 &#x1f4da;前言 &#x1f4da;一、信号的本质 &#x1f4d6;1.异步通信 &#x1f4d6;2.信…...

Vue3.0使用JavaScript脚本实现Vue Router路由:页面跳转、获取URL参数

Vue 使用 Vue Router 路由系列文章: 《Vue使用Vue Router路由:开发单页应用》 《Vue使用Vue Router路由:通过URL传递与获取参数》 《Vue3.0使用JavaScript脚本实现Vue Router路由:页面跳转、获取URL参数》 1、路由基础 在单页 Web 应用中,整个项目只有一个 HTML 文件,不…...

2025山东科技大学考研专业课复习资料一览

[冲刺]2025年山东科技大学020200应用经济学《814经济学之西方经济学[宏观部分]》考研学霸狂刷870题[简答论述计算题]1小时前[强化]2025年山东科技大学085600材料与化工《817物理化学》考研强化检测5套卷22小时前[冲刺]2025年山东科技大学030100法学《704综合一[法理学、国际法学…...

lambda 表达式 闭包写法

lambda 表达式 1.用于 匿名委托函数表达 2.用于linq 查询表达 匿名方法表达 &#xff08;参数&#xff09;》{ 逻辑} 比如 &#xff08;x,y&#xff09;>{return xy;} 如果一个参数可不带&#xff08;&#xff09;&#xff0c;如果逻辑简单可以不{} 比如 x>x 如果没有参…...

什么是正则化?Regularization: The Stabilizer of Machine Learning Models(中英双语)

正则化&#xff1a;机器学习模型的稳定器 1. 什么是正则化&#xff1f; 正则化&#xff08;Regularization&#xff09;是一种在机器学习模型训练中&#xff0c;通过约束模型复杂性以防止过拟合的技术。 它的核心目标是让模型不仅在训练集上表现良好&#xff0c;还能在测试集上…...

【西门子PLC.博途】——面向对象编程及输入输出映射FC块

当我们做面向对象编程的时候&#xff0c;需要用到输入输出的映射。这样建立的变量就能够被复用&#xff0c;从而最大化利用了我们建立的udt对象。 下面就来讲讲映射是什么。 从本质上来说&#xff0c;映射就是拿实际物理对象对应程序虚拟对象&#xff0c;假设程序对象是I0.0&…...

CS61a.1 textbook1.2 编程要素

1.structure and interpretation of computer programs Python 内置了对各种常见编程活动的支持&#xff0c; 例如&#xff0c;操作文本、显示图形以及通过 互联网。Python 代码行 >>> from urllib.request import urlopen是一个 import 语句&#xff0c;用于加载用…...

计算机毕业设计Django+Tensorflow音乐推荐系统 音乐可视化 卷积神经网络CNN LSTM音乐情感分析 机器学习 深度学习 Flask

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…...

使用国内镜像网站在线下载安装Qt(解决官网慢的问题)——Qt

国内镜像网站 中国科学技术大学&#xff1a;http://mirrors.ustc.edu.cn/qtproject/清华大学&#xff1a;https://mirrors.tuna.tsinghua.edu.cn/qt/北京理工大学&#xff1a;http://mirror.bit.edu.cn/qtproject/ 南京大学&#xff1a;https://mirror.nju.edu.cn/qt腾讯镜像&…...

乳腺癌多模态诊断解释框架:CNN + 可解释 AI 可视化

乳腺癌多模态诊断解释框架&#xff1a;CNN 可解释 AI 可视化 论文大纲理解1. 确认目标2. 分析过程&#xff08;目标-手段分析&#xff09;3. 实现步骤4. 效果展示 结构分析1. 层级结构分析叠加形态&#xff08;从基础到高级&#xff09;构成形态&#xff08;部分到整体&#x…...

MySQL篇之对MySQL进行参数优化,提高MySQL性能

1. MySQL参数优化说明 MySQL 参数调优是提高数据库性能的重要手段之一。通过调整 MySQL 的配置参数&#xff0c;可以优化查询速度、提升并发处理能力、减少资源消耗等。 MySQL 的性能优化涉及到多个方面&#xff0c;包括内存管理、磁盘 I/O、查询优化、连接管理、复制配置等。…...

Scratch节日 | 快乐圣诞节——用编程传递节日祝福! ✨

今天为大家推荐一款充满节日气氛的Scratch项目——《快乐圣诞节》&#xff01;这款圣诞主题动画贺卡项目不仅让小朋友们学习编程知识&#xff0c;还提供了一种用创意传递祝福的方式。通过编程打造星星闪烁的圣诞树&#xff0c;播放经典圣诞音乐&#xff0c;制作一张属于自己的节…...

android studio更改应用图片,和应用名字。

更改应用图标&#xff0c;和名字 先打开AndroidManifest.xml文件。 更改图片文件名字&#xff08; 右键-->构建-->重命名&#xff08;R&#xff09;&#xff09;...

PHP8.4下webman直接使用topthink/think-orm

环境信息 操作系统win11php 8.4.1webman-framework ^1.6.8MySQL 8.4.3topthink/think-orm ^3.0 说明 PHP8.3以下版本 直接使用webman提供的webman/think-orm更方便。 PHP 环境换为 8.4 使用webman/think-orm 报了个错&#xff1b;所以换topthink/think-orm&#xff0c;根据文…...

uniapp 微信小程序 功能入口

单行单独展示 效果图 html <view class"shopchoose flex jsb ac" click"routerTo(要跳转的页面)"><view class"flex ac"><image src"/static/dyd.png" mode"aspectFit" class"shopchooseimg"&g…...

Halcon 机器视觉案例 之 连接件测量

第一篇 机器视觉案例 之 连接件测量 文章目录 第一篇 机器视觉案例 之 连接件测量1.案例要求2.实现思路2.1 读取单张图片并创建图像模板2.2 画出圆和直线2.3 创建测量模型2.4 循环读取多张图片并查找图像中连接件位置2.5 根据偏移量补偿使得测量模型移动至指定位置 3.实现效果4…...

druid与pgsql结合踩坑记

最近项目里面突然出现一个怪问题&#xff0c;数据库是pgsql&#xff0c;jdbc连接池是alibaba开源的druid&#xff0c;idea里面直接启动没问题&#xff0c;打完包放在centos上和windows上cmd窗口都能直接用java -jar命令启动&#xff0c;但是放到国产信创系统上就是报错&#xf…...

Windows环境 (Ubuntu 24.04.1 LTS ) 国内镜像,用apt-get命令安装RabbitMQ,java代码样例

一、环境 Windows11 WSL(Ubuntu 24.04.1) 二、思路 1 用Windows中的Ubuntu安装RabbitMQ&#xff0c;贴近Linux的线上环境&#xff1b; 2 RabbitMQ用erlang语言编写的&#xff0c;先安装erlang的运行环境&#xff1b; 2 用Linux的apt-get命令安装&#xff0c;解决软件依赖…...

RabbitMQ的核心组件有哪些?

大家好&#xff0c;我是锋哥。今天分享关于【RabbitMQ的核心组件有哪些&#xff1f;】面试题。希望对大家有帮助&#xff1b; RabbitMQ的核心组件有哪些&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 RabbitMQ是一个开源的消息代理&#xff08;Messag…...

mysql免安装版配置教程

一、将压缩包解压至你想要放置的文件夹中&#xff0c;注意&#xff1a;绝对路径中要避免出现中文 二、在解压目录下新建my.ini文件&#xff0c;已经有的就直接覆盖 my.ini文件内容 [mysqld] # 设置3306端口 port3306 # 设置mysql的安装目录 basedirD:\\tools\\mysql-8.1.0-win…...

ubuntu+ros新手笔记(三):21讲没讲到的MoveIt2

1 安装MoveIt2 安装参照在ROS2中&#xff0c;通过MoveIt2控制Gazebo中的自定义机械手 安装 MoveIt2可以选择自己编译源码安装&#xff0c;或者直接从二进制安装。 个人建议直接二进制安装&#xff0c;可以省很多事。 sudo apt install ros-humble-moveitmoveit-setup-assistan…...

Charles简单压力测试

01、接口请求次数&#xff0c;并发量&#xff0c;请求延迟时间均可配置 选中需要进行测试的接口&#xff0c;鼠标右键选中【repeat advance】 02、设置并发参数 下面的图中&#xff0c;选择了1个接口&#xff0c;每次迭代中1个接口同时请求&#xff0c;迭代1000次&#xff08…...

决策树的生成与剪枝

决策树的生成与剪枝 决策树的生成生成决策树的过程决策树的生成算法 决策树的剪枝决策树的损失函数决策树的剪枝算法 代码 决策树的生成 生成决策树的过程 为了方便分析描述&#xff0c;我们对上节课中的训练样本进行编号&#xff0c;每个样本加一个ID值&#xff0c;如图所示…...

对象克隆与单例模式的实现

一、引言 在 C 编程中&#xff0c;对象克隆和单例模式是两个非常重要的概念。对象克隆可以帮助我们快速创建具有相同状态的对象副本&#xff0c;而单例模式则可以确保一个类只有一个实例&#xff0c;并提供全局访问点。本文将详细介绍 C 中如何实现对象的克隆以及单例模式。 …...

在 Linux 系统中,让 apt 使用 HTTP 代理

在 Linux 系统中&#xff0c;要让 apt 使用 HTTP 代理&#xff0c;有几种方法可以实现&#xff1a; ### 1. 临时设置代理 你可以通过设置环境变量来临时为 apt 命令设置代理。这种方法不需要修改任何配置文件&#xff0c;只需在命令行中设置环境变量即可。例如&#xff1a; …...

中国气象局:2024年第二批“气象数据要素×”典型案例(附下载)

11月25日&#xff0c;中国气象局通报2024年第二批“气象数据要素”典型案例。此次遴选充分结合首届“数据要素”大赛获奖案例&#xff0c;旨在进一步号召各级气象部门充分学习借鉴先进经验和做法&#xff0c;持续推动高价值气象数据产品开发利用&#xff0c;挖掘气象数据要素应…...

Android绘图Path基于LinearGradient线性渐变,Kotlin(1)

Android绘图Path基于LinearGradient线性渐变&#xff0c;Kotlin&#xff08;1&#xff09; import android.content.Context import android.graphics.Canvas import android.graphics.Color import android.graphics.LinearGradient import android.graphics.Paint import and…...

基础入门-APP应用微信小程序原生态开发H5+Vue技术WEB封装打包反编译抓包点

知识点&#xff1a; 1、基础入门-APP应用-开发架构安全问题 2、基础入门-小程序应用-开发架构安全问题 通用&#xff1a; 1、反编译-得到源码-源码提取资产&#xff08;泄漏的配置信息&#xff09;-安全测试 2、抓包-资产-安全测试 一、演示案例-移动App-开发架构-原生&H…...

vue+ts提交数据时不提交(剔除)某项数据

关键代码 // 假设这是原始要提交的数据数组 const dataArray [{areaConfId: ,areaName: ,cityInfo: [],provinceList: [],cityList: []} ];// 使用map方法遍历数组中的每个对象&#xff0c;通过解构赋值和对象展开运算符去除cityInfo字段 const newDataArray dataArray.map…...

乐凡信息智能安全管控方案:助力油气田行业安全管控多方位升级

我国油田地域广阔&#xff0c;分布着大量各种油井&#xff0c;油井开采设备的连续稳定运行是保证石油开采的首要条件。然而&#xff0c;由于油田多位于特殊地理环境中&#xff0c;因而实现油井之间的通信首要问题就是要克服地理环境所带来的限制&#xff0c;传统通信系统的建设…...

Jenkins搭建并与Harbor集成上传镜像

Jenkins介绍 Jenkins 是一个开源的自动化服务器&#xff0c;广泛用于持续集成&#xff08;CI&#xff09;和持续交付&#xff08;CD&#xff09;的实践中。它帮助开发人员自动化构建、测试和部署过程&#xff0c;从而提高开发效率、代码质量和项目交付速度。通过丰富的插件支持…...

FutureCompletableFuture实战

1. Callable&Future&FutureTask介绍 直接继承Thread或者实现Runnable接口都可以创建线程&#xff0c;但是这两种方法都有一个问题就是&#xff1a;没有返回值&#xff0c;也就是不能获取执行完的结果。因此java1.5就提供了Callable接口来实现这一场景&#xff0c;而Fu…...

Amazon与Shopee平台对比:跨境卖家如何选对平台打开市场?

在跨境电商领域&#xff0c;选择合适的平台对卖家能否成功打开市场至关重要。如今&#xff0c;Amazon和Shopee成为了众多卖家的热门选择。一个以全球化布局和高端市场著称&#xff0c;一个则专注东南亚新兴市场的潜力。两者各有优势&#xff0c;但也需要根据卖家的业务模式、目…...

【项目实战】redis实现websocket分布式消息推送服务

由于redis并非专业的MQ中间件&#xff0c;消息的防丢失策略并不完整&#xff0c;存在丢失消息的可能。该方案为在再pc web管理平台的右下角弹出&#xff0c;显示新接收到的消息数&#xff0c;哪怕没有收到这个通知&#xff0c;也可以自己在消息中心看看。所以对可靠性要求不高。…...

(自用)配置文件优先级、SpringBoot原理、Maven私服

配置优先级 之前介绍过SpringBoot中支持三类配置文件.properties、.yml和.yaml&#xff0c;他们三者之间也是有着优先级顺序的&#xff0c;为.properties➡.yml➡.yaml。 同时SpringBoot为了增强程序的拓展性&#xff0c;除了支持配置文件属性配置&#xff0c;还支持Java系统属…...

在windows系统中使用labelimg对图片进行标注之工具安装及简单使用

一.背景 还是之前的主题&#xff0c;使用开源软件为公司搭建安全管理平台&#xff0c;从视觉模型识别安全帽开始。我是从运行、训练、标注倒过来学习的。本次主要是学习标注工具labelimg的安装及简单使用。 二.下载 LabelImg是一款广受欢迎的开源图像标注工具&#xff0c;为计…...

数字图像处理技术期末复习

1. 已知图像的分辨率和深度&#xff0c;怎么求图像的存储空间&#xff08;位&#xff0c;字节&#xff0c;KB&#xff09;&#xff1f; 题目&#xff1a; 已知图像的分辨率和深度&#xff0c;怎么求图像的存储空间&#xff08;位&#xff0c;字节&#xff0c;KB&#xff09;&a…...

点云空洞的边界识别提取 pso-bp 神经网络的模型来修复点云空洞 附python代码

代码是一个Python程序,用于处理3D点云数据,特别是检测和修复点云中的孔洞区域。 1. **导入库**: - `numpy`:用于数学运算。 - `open3d`:用于处理3D数据和可视化。 - `torch`:PyTorch库,用于深度学习。 - `torch.nn`和`torch.optim`:PyTorch的神经网络和优…...

【AutoDL】通过【SSH远程连接】【vscode】

小帅碎碎念 0. 起因1. SSH信息获取2. 给你的vscode安装支持SSH远程连接的插件3. SSH远程连接入口4. 输入密码登陆5. 总结 0. 起因 之前使用AutoDL和Jupyter进行代码编辑和执行确实很方便&#xff0c;尤其是对于交互式数据分析项目。然而&#xff0c;也存在一些限制和不便之处&…...

ubuntu22.04编译安装Opencv4.8.0+Opencv-contrib4.8.0教程

本章教程,主要记录在Ubuntu22.04版本系统上编译安装安装Opencv4.8.0+Opencv-contrib4.8.0的具体过程。 一、下载opencv和opencv-contrib包 wget https://github.com/opencv/opencv/archive/refs/tags/4.8.0.zip wget https://github.com/opencv/opencv_contrib/archive/refs/…...

短链接服务

一 功能描述 1.短链接是将长连接转化为短连接使得链接变得美观清爽&#xff0c;让用户点击率更高&#xff0c;同时规避原始链接中一些关键词、域名屏蔽等问题&#xff0c;最终利用短链每次跳转都需要经过后端的特性&#xff0c;在跳转过程中做异步埋点&#xff0c;用于效果数据…...

【Vue3学习】setup语法糖中的ref,reactive,toRef,toRefs

在 Vue 3 的组合式 API&#xff08;Composition API&#xff09;中&#xff0c;ref、reactive、toRef 和 toRefs 是四个非常重要的工具函数&#xff0c;用于创建和管理响应式数据。 一、ref 用ref()包裹数据,返回的响应式引用对象&#xff0c;包含一个 .value 属性&#xff0…...

Halcon中dots_image(Operator)算子原理及应用详解

在HALCON中&#xff0c;dots_image算子是一个用于增强图像中圆点效果的强大工具&#xff0c;特别适合于点的分割&#xff0c;以及OCR&#xff08;光学字符识别&#xff09;应用程序中增强点状印刷字体。以下是对dots_image (ImageResult, DotImage, 5, ‘dark’, 2)算子原理及应…...

【C语言】库函数常见的陷阱与缺陷(四):内存内容操作函数[5]--memchr

C语言中的memchr函数用于在内存块中搜索一个特定的字符(实际上是unsigned char类型的值),并返回该字符第一次出现的指针。虽然这个函数在内存搜索中非常有用,但它也存在一些陷阱。 一、功能与用法 功能:memchr函数在指定的内存块中搜索第一次出现的特定字符,并返回一个…...

【P2P】【Go】采用go语言实现udp hole punching 打洞 传输速度测试 ping测试

服务器端 udpserver/main.go package mainimport ("fmt""net""sync""sync/atomic" )var (clientCounter uint64 0 // 客户端连接计数器mu sync.Mutex )func main() {addr, err : net.ResolveUDPAddr("udp", &q…...

【附源码】Electron Windows桌面壁纸开发中的 CommonJS 和 ES Module 引入问题以及 Webpack 如何处理这种兼容

背景 在尝试让 ChatGPT 自动开发一个桌面壁纸更改的功能时&#xff0c;发现引入了一个 wallpaper 库&#xff0c;这个库的入口文件是 index.js&#xff0c;但是 package.json 文件下的 type:"module"&#xff0c;这样造成了无论你使用 import from 还是 require&…...

【SpringBoot 调度任务】

在 Spring Boot 中实现调度任务&#xff08;Scheduled Tasks&#xff09;&#xff0c;通过使用 EnableScheduling 和 Scheduled 注解来完成。 添加依赖启用调度任务支持创建调度任务运行应用程序 添加依赖 pom.xml 文件中有以下依赖项&#xff1a; <dependency><gro…...

Android v4和v7冲突

android.useAndroidXtrue android.enableJetifiertruev4转成AndroidX...

【HarmonyOS之旅】HarmonyOS开发基础知识(一)

目录 1 -> 应用基础知识 1.1 -> 用户应用程序 1.2 -> 用户应用程序包结构 1.3 -> Ability 1.4 -> 库文件 1.5 -> 资源文件 1.6 -> 配置文件 1.7 -> pack.info 1.8 -> HAR 2 -> 配置文件简介 2.1 -> 配置文件的组成 3 -> 配置文…...

【排序算法】——插入排序

目录 前言 简介 基本思想 1.直接插入排序 2.希尔排序 代码实现 1.直接插入排序 2.希尔排序 总结 1.时空复杂度 2.稳定性 尾声 前言 排序(Sorting) 是计算机程序设计中的一种重要操作&#xff0c;它的功能是将一个数据元素&#xff08;或记录&#xff09;的任意序列&…...