Linux进程编程(PS: exec族函数、system、popen函数)

2023-11-18

1.进程相关概念

程序和进程

程序是静态的概念,gcc xx.x -o pro,磁盘中生成的pro就是程序。

进程是程序的一次运行活动,通俗的讲就是程序跑起来了,系统中就多了一个进程。

查看系统中的进程

ps指令

查看系统中所有进程

ps -aux

结果:
在这里插入图片描述

查看系统中的init进程

ps -aux | grep init

结果:
在这里插入图片描述
即把ps -aux指令所有输出的结果通过管道导向grep进行搜索,查找init关键字的文本

-aux 显示所有包含其他使用者的进程
|管道符号
grep 用于查找文件里符合条件的字符串

top指令

类似windows的任务管理器,数据也是实时动态变化的。
在这里插入图片描述

进程标识符 使用getpid()获取

每一个进程都有一个非负整数标识唯一的ID,即为pid。

调用getpid()获取自身进程标识符,getppid()获取获取父进程标识符

其中系统所占用的进程标识符如下:

pid 进程名称 作用 说明
pid = 0 交换进程 用于进程调度 所有“同时”在运行的程序所占用的资源受到进程调度的影响
pid = 1 init进程 用于系统初始化 程序运行,内核加载完毕,文件系统起来的第一个进程就是init进程,读取配置文件然后再启动其他进程。(如ktv点歌机开机后看到的是点歌界面,而不是字符界面)
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
	//pid_t getpid(void);
	pid_t pid;
	pid=getpid();
	printf("我的pid是:%d\n",pid);

	return 0;
}

父进程,子进程

进程A创建了进程B,A就是父进程,B就是子进程

2.创建进程fork

pid_t fork(void);

fork函数调用成功,返回两次:返回值为0,代表当前进程是子进程,非负数为父进程,如果调用失败则返回-1
创建子进程的目的:复制父进程(此时两个或两个以上进程),父进程等待客户端服务请求,当这种请求到达时,父进程调用fork,让子进程去处理(QQ服务器 客户端 结合Socket网络编程)

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
        pid_t pid;

        pid =  getpid();

        fork();//此处开始,后面所有代码执行了两次 
        	   //fork()有两个返回值

        printf("my pid is %d\n",pid);

        return 0;
}

返回两次:
在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
        pid_t pid;

        pid =  getpid();

        fork();

        printf("my pid is %d,current pro id:%d\n",pid,getpid());

        return 0;
}                                    

在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
        pid_t pid;

        pid =  getpid();

        fork();

        if(pid == getpid()){
                printf("this is father print\n");
        }else{
                printf("this is child print,child pid = %d\n",getpid());
                                
        }

        
        return 0;
}                                                                                                                                                                                                              

在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
        pid_t pid;

        printf("father: pid = %d\n",getpid());

        pid =  fork();


        if(pid > 0){
                printf("this is father print, pid = %d\n",getpid());
        }else if(pid == 0){
                printf("this is child print,child pid = %d\n",getpid());

        }


        return 0;
}

在这里插入图片描述

应用场景(模拟网络等待):

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main()
{
        pid_t pid;
        int data;

        while(1){
                printf("please input a data\n");
                scanf("%d",&data);
                if(data == 1){
                        pid = fork();

                        if(pid > 0){

                        }else if(pid == 0){

                                while(1){
                                        printf("do net request,pid=%d\n",getpid());
                                        sleep(3);
                                }
                        }


                }else{
                        printf("wait, do nothing\n");
                }
        }




        return 0;
}

进程创建发生了什么——C程序的存储空间如何分配

详细参照博文:内存四区(代码区 静态区 栈区 堆区)

fork创建进程,所有的东西都进行了拷贝,包括全局变量、局部变量、代码等,各进程内改变变量的值互不影响。

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

详细说明:
在这里插入图片描述

3.创建进程vfork(区别fork)

vfork与fork的区别:
1.(共用)vfork直接使用父进程的存储空间,不拷贝(随着内核的发展,fork目前执行的是写时拷贝—copy on write 若子进程未改变原始变量的值时不会进行拷贝)
2.(等待)vfork保证子进程先运行,当子进程调用exit退出后,父进程才执行

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>

int main(){
        int cnt=0;
        pid_t pid;
        pid_t repid;
        pid=getpid();
        repid=vfork();
        if(repid > 0){
                while(1){
                        printf("cnt=%d\n",cnt);
                        printf("this is father pid,repid=%d\n",repid);
                        sleep(1);
                }
        }
        else if(repid == 0){
                while(1){
                        printf("this is child pid,repid=%d\n",repid);
                        sleep(1);
                        cnt++;
                        if(cnt==2){
 
                                exit(0);
                        }
                }
        }
        return 0;
}

运行结果:
在这里插入图片描述

4.进程退出

正常退出

1.main函数调用return

2.进程调用exit(),标准C库

3.进程调用_exit()或者_Exit(),系统调用

4.进程最后一个线程返回

5.最后一个线程调用pthread_exit

异常退出

1.调用abort

2.当进程收到某些信号时,如ctrl+c

3.最后一个线程对取消(cancellation)请求作出响应

注意:不管进程如何终止,最后都会执行内核中的同一段代码,这段代码为相应进程关闭所有打开描述符,释放它所使用的存储器等。

5.父进程等待子进程退出

父进程收集子进程退出状态(僵尸进程)

父进程等待子进程退出并收集子进程退出状态
子进程退出状态不被收集,会变成僵尸进程

僵尸进程的例子:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>


int main()
{
	pid_t pid;
	int i;
	int cnt=0;
	pid=fork();
	if(pid > 0){//父进程
		while(1){
			printf("这是父进程,pid=%d\n",getpid());
			printf("cnt=%d\n",cnt);
			sleep(2);//防止刷屏
		}	
	}else if(pid == 0){//子进程
		for(i=0;i<5;i++){//看看是不是保证子进程先运行,五次过后推出进入父进程
			printf("这是子进程,pid=%d,这是第%d次\n",getpid(),i+1);
			cnt++;
			sleep(1);//防止刷屏
		}
		exit(-1);

	}
	return 0;
}

在这里插入图片描述

此时运行的结果是子进程退出了,但是退出状态没有被收集,子进程成了僵尸进程(zomb)。

在这里插入图片描述

等待退出函数wait()

在这里插入图片描述

wait(int *status):
status参数,他是一个整型数指针。
非空:子进程退出状态放在它所指向的地址
空(NULL):不关心退出状态

在这里插入图片描述

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>


int main()
{
	pid_t pid;
	int i;
	int cnt=0;
	int status;
	pid=fork();
	if(pid > 0){//父进程
		wait(&status);
		printf("子进程退出,status:%d\n",WEXITSTATUS(status));//WEXITSTATUS()是解析子进程退出码的宏,下面有详细介绍
		while(1){
			
			printf("这是父进程,pid=%d\n",getpid());
			printf("cnt=%d\n",cnt);
			sleep(2);//防止刷屏
		}	
	}else if(pid == 0){//子进程
		for(i=0;i<5;i++){//看看是不是保证子进程先运行,五次过后推出进入父进程
			printf("这是子进程,pid=%d,这是第%d次\n",getpid(),i+1);
			cnt++;
			sleep(1);//防止刷屏
		}
		exit(3);//退出码返回给父进程中的wait进行收集

	}
	return 0;
}

在这里插入图片描述

解析出wait()对子进程退出码回收的宏

在这里插入图片描述

等待退出函数waitpid()

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
来个例子:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>



int main()
{
        pid_t pid;
        int i;
        int cnt=0;
        int status;
        pid=fork();
        if(pid > 0){//父进程
                waitpid(pid,&status,WNOHANG);//WNOHANG不阻塞
                printf("子进程退出,status:%d\n",WEXITSTATUS(status));
                while(1){

                        printf("这是父进程,pid=%d\n",getpid());
                        printf("cnt=%d\n",cnt);
                        sleep(2);//防止刷屏
                }
        }else if(pid == 0){//子进程
                for(i=0;i<5;i++){//看看是不是保证子进程先运行,五次过后推出进入父进程
                        printf("这是子进程,pid=%d,这是第%d次\n",getpid(),i+1);
                        cnt++;
                        sleep(1);//防止刷屏
                }
                exit(3);

        }
        return 0;
}

结果如下所示,发现子进程也变为僵尸进程。
在这里插入图片描述

在这里插入图片描述

父进程先于子进程退出(孤儿进程)

父进程如果不等待子进程退出 ,在子进程之前就结束了自己的生命,此时的子进程就叫做是孤儿进程
Linux避免系统存在太多的孤儿进程,init进程收留孤儿进程,变成孤儿进程的父进程

例子:

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>



int main()
{
	pid_t pid;
	int i;
	int status;
	int cunt=0;
	pid=fork();
	if(pid > 0){
		
			printf("这是父进程,pid=%d\n",getpid());
			printf("cunt=%d\n",cunt);
			sleep(2);
		
	}else if(pid == 0){
		for(i=0;i<5;i++){
			printf("这是子进程,pid=%d,我的父进程的pid=%d\n",getpid(),getppid());
			cunt++;
			sleep(2);
		}
		exit(3);
	}




	return 0;
}

运行结果:
在这里插入图片描述

最后再来一个综合的例子:
在这里插入图片描述

6.exec族函数(让子进程调用其他程序)

参照博文:https://blog.csdn.net/u014530704/article/details/73848573

execl,execlp

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
 
int main(){
 
        //int execl(const char *path, const char *arg, .../* (char  *) NULL */);
/*      path:可执行文件的路径名字
        arg:可执行程序所带的参数,第一个参数为可执行文件名字,没有带路径且arg必须以NULL结束
        file:如果参数file中包含/,则就将其视为路径名,否则就按 PATH环境变量,在它所指定的各目录中搜寻可执行文件
        
        exec族函数参数极难记忆和分辨,函数名中的字符会给我们一些帮助:
        
        l : 使用参数列表
        p:使用文件名,并从PATH环境进行寻找可执行文件
        v:应先构造一个指向各参数的指针数组,然后将该数组的地址作为这些函数的参数。
        e:多了envp[]数组,使用新的环境变量代替调用进程的环境变量
        exec函数族的函数执行成功后不会返回,调用失败时,会设置errno并返回-1,然后从原程序的调用点接着往下执行。 
*/
 
        printf("before execl\n");
        if(execl("/bin/ls","ls",NULL,NULL) == -1)//通过whereis指令找到ls命令的位置
        {
                printf("execl failed!\n");
                perror("why");
        }
                printf("after execl\n");
 
        return 0;
}
int main(){
 
        //int execlp(const char *file, const char *arg, .../* (char  *) NULL */);
        printf("before execl\n");
        if(execlp("ls","ls","-l",NULL) == -1)//不用找到命令的路径
        {
                printf("execl failed!\n");
                perror("why");
        }
                printf("after execl\n");
 
        return 0;
}

运行结果:
在这里插入图片描述

通过whereis指令查找ls、date(获取系统时间)系统指令的位置:

在这里插入图片描述
在这里插入图片描述

execv execvp

int main(){
 
        //int execv(const char *path, char *const argv[]);      
 
        char* argv[]={"ls",NULL,NULL};
        if(execv("/bin/ls",argv) == -1)
        {
                printf("execl failed!\n");
                perror("why");
        }
                printf("after execl\n");
 
        return 0;
}
int main(){
 
        // int execvp(const char *file, char *const argv[]);
 
        char* argv[]={"ls","-l",NULL};
        if(execvp("ls",argv) == -1)
        {
                printf("execl failed!\n");
                perror("why");
        }
                printf("after execl\n");
 
        return 0;
}

exec配合fork使用

应用场景:在执行A程序的过程中让子进程去执行B程序

代码B(用来修改配置文件) 编译生成程序名为change

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/wait.h>


int main()
{
	int fd;
	int n_write;
	int n_read;
	pid_t pid;
	char *readBuf;
	fd=open("test15.txt",O_RDWR);
	int size=lseek(fd,0,SEEK_END);//计算文件的大小
	lseek(fd,0,SEEK_SET);//光标重新到开头
	readBuf=(char *)malloc(sizeof(char)*size+8 );//+8防止溢出
	n_read=read(fd,readBuf,size);
	char *p=strstr(readBuf,"length=");
	if(p == NULL ){
		printf("没有找到\n");
		exit(-1);
	}
	p=p+strlen("length=");//指针往后移动大哦想要改的地方
	*p='9';//将里面的6改称9
	lseek(fd,0,SEEK_SET);
	n_write=write(fd,readBuf,strlen(readBuf));
	close(fd);
	exit(0);
}

代码A

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/wait.h>




int main()
{
	int fd;
	int n_write;
	int n_read;
	pid_t pid;
	int data;
	char *readBuf;
	while(1){
		printf("请输入一个数\n");
		scanf("%d",&data);
		if(data == 1){
			printf("成功进入\n");
			pid=fork();
			if(pid > 0){//父进程
				wait(NULL);//防止子进程变成僵尸进程
		}
			if(pid == 0){//子进程
				execl("./change","change",NULL,NULL);//获取系统时间也能实现
			}



		}else{
			printf("等待输入正确的指令\n");
		}
	}

	return 0;
}

初始的test15.txt
在这里插入图片描述

运行程序A后

在这里插入图片描述

system函数

本质上是对execl函数的二次封装,可查看其源码。实际上比execl更好用
和execl区别:system执行完该函数后,还会继续执行后面的函数 ,而exec则不会。

#include <stdlib.h>
int system(const char *command);
system()函数返回值如下:
成功,则返回进程的状态值
当sh不执行时,返回127//shell脚本
失败返回 -1
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>


int main(){
        pid_t pid;
        pid_t repid;
        int data;
        while(1){
                printf("please input a number:");
                scanf("%d",&data);
                if(data==1){
                        repid=fork();//创建进程
                        if(repid > 0){
                                wait(NULL);
                        }
                         if(repid == 0){
 
                                //execl("./changeNumToFile.out","changeNumToFile.c","changeNum.c",NULL);
                                system("./changeNumToFile.out");//执行完该函数后,还会执行后面的函数  exec就不会回来继续执行后面代码了,除非execl出错返回-1
                                exit(0);
                                }
                        }
                else{
                        printf("waiting!\n");
                }
        }
        return 0;

popen函数

比system在应用中的好处:可以获取运行的输出结果

#include <stdio.h>
FILE *popen(const char *command, const char *type);
int pclose(FILE *stream);
#include <stdio.h>
#include <stdlib.h>
 //      FILE *popen(const char *command, const char *type);
//       FILE *fopen(const char *path, const char *mode);
//      size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
int main(){
 
//      system("ls");
        FILE* fp;
        char ret[1024];
        fp=popen("ls -l","r");
        int n_read=fread(ret,1,1024,fp);
 
        if(n_read != -1){
                printf("read sucess!\n");
                printf("return n_read=%d,read data=%s\n",nread,ret);
        }
        else{
                printf("no read datas\n");
        }
        return 0;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Linux进程编程(PS: exec族函数、system、popen函数) 的相关文章

  • 在Linux伪终端中执行从一个终端发送到另一个终端的字符串

    假设我有一个终端 其中 tty 的输出是 dev pts 2 我想从另一个终端向第一个终端发送命令并执行它 使用 echo ls gt dev pts 2 仅在第一个终端中打印 ls 有没有办法执行字符串 不 终端不执行命令 它们只是数据的
  • 即使 makefile 和源代码存在,为什么“Build Project”在 Eclipse Helios CDT 中显示为灰色?

    我无法构建我的项目 我在 Eclipse Helios 中创建了一个新的 CDT 项目 并告诉它使用现有的源代码和 makefile 这两者都正确显示在 Package 和 Project 视图中 然而 项目 菜单中的 构建全部 和 构建项
  • 使用 MongoDB docker 镜像停止虚拟机而不丢失数据

    我已经在 AWS EC2 上的虚拟机中安装了官方的 MongoDB docker 映像 并且数据库上已经有数据 如果我停止虚拟机 以节省过夜费用 我会丢失数据库中包含的所有数据吗 在这些情况下我怎样才能让它持久 有多种选择可以实现此目的 但
  • 使用netcat将unix套接字传输到tcp套接字

    我正在尝试使用以下命令将 unix 套接字公开为 tcp 套接字 nc lkv 44444 nc Uv var run docker sock 当我尝试访问时localhost 44444 containers json从浏览器中 它不会加
  • 原生 Linux 应用程序可像 ResHacker 一样编辑 Win32 PE

    我想运行自动修改 dll服务 用户提交特定的 dll 我在服务器上修改它 然后用户可以下载 dll的修改版本 是否有任何本机 Linux 应用程序提供常见的 Win32 PE 修改功能 例如图标 字符串 加速器 对话等 至少提供命令行或脚本
  • 在 Ubuntu 上纯粹通过 bash 脚本安装 mysql 5.7

    我想要一个无需任何手动输入即可安装 MySQL 5 7 实例的 bash 脚本 我正在关注数字海洋教程 https www digitalocean com community tutorials how to install mysql
  • 静态链接共享对象?或者损坏的文件?

    我有一个从专有来源获得的库 我正在尝试链接它 但出现以下错误 libxxx so 文件无法识别 文件格式无法识别 Collect2 ld 返回 1 退出状态 确实 ldd libxxx so statically linked 这究竟意味着
  • 通过 SSH 将变量传递给远程脚本

    我正在通过 SSH 从本地服务器在远程服务器上运行脚本 首先使用 SCP 复制该脚本 然后在传递一些参数时调用该脚本 如下所示 scp path to script server example org another path ssh s
  • Windows 与 Linux 文本文件读取

    问题是 我最近从 Windows 切换到 Ubuntu 我的一些用于分析数据文件的 python 脚本给了我错误 我不确定如何正确解决 我当前仪器的数据文件输出如下 Header 有关仪器等的各种信息 Data 状态 代码 温度 字段等 0
  • linux下如何获取昨天和前天?

    我想在变量中获取 sysdate 1 和 sysdate 2 并回显它 我正在使用下面的查询 它将今天的日期作为输出 bin bash tm date Y d m echo tm 如何获取昨天和前天的日期 这是另一种方法 对于昨天来说 da
  • 为什么 ld 无法从 /etc/ld.so.conf 中的路径找到库?

    我想添加 opt vertica lib64进入系统库路径 所以我执行以下步骤 1 添加 opt vertica lib64 into etc ld so conf 然后运行ldconfig 2 检查 bash ldconfig p gre
  • 我应该使用哪个 Linux 发行版作为 Xen 主机? [关闭]

    Closed 这个问题是基于意见的 help closed questions 目前不接受答案 我为家庭办公室订购了一台服务器 我想用 Xen 对其进行分区 我认为这将使事情保持干净并且更容易维护 我将运行 MySQL PostgreSQL
  • Linux命令列出所有可用命令和别名

    是否有一个 Linux 命令可以列出该终端会话的所有可用命令和别名 就好像您输入 a 并按下 Tab 键一样 但针对的是字母表中的每个字母 或者运行 别名 但也返回命令 为什么 我想运行以下命令并查看命令是否可用 ListAllComman
  • 进程名称长度的最大允许限制是多少?

    进程名称允许的最大长度是多少 我正在读取进程名称 proc pid stat文件 我想知道我需要的最大缓冲区 我很确定有一个可配置的限制 但就是找不到它在哪里 根据man 2 prctl http man7 org linux man pa
  • 为 Linux 安装 R 包时出错

    我试图在 R 3 3 上安装一个名为 rgeos 的包 但是当我输入 install packages rgeos 但它返回给我以下错误 其他包也会发生同样的情况 但不是所有包 gt installing source package rg
  • python:numpy 运行脚本两次

    当我将 numpy 导入到 python 脚本中时 该脚本会执行两次 有人可以告诉我如何阻止这种情况 因为我的脚本中的所有内容都需要两倍的时间 这是一个例子 usr bin python2 from numpy import print t
  • 套接字:监听积压并接受

    listen sock backlog 在我看来 参数backlog限制连接数量 这是我的测试代码 server initialize the sockaddr of server server sin family AF INET ser
  • Scrapy FakeUserAgentError:获取浏览器时发生错误

    我使用 Scrapy FakeUserAgent 并在我的 Linux 服务器上不断收到此错误 Traceback most recent call last File usr local lib64 python2 7 site pack
  • 如何调用位于其他目录的Makefile?

    我正在尝试这样做 我想打电话给 make Makefile存在于其他目录中 abc可以使用位于不同目录中的 shell 脚本的路径 我该怎么做呢 由于 shell 脚本不允许我cd进入Makefile目录并执行make 我怎样才能编写she
  • 如何查明 Ubuntu 上安装了哪个版本的 GTK+?

    我需要确定 Ubuntu 上安装了哪个版本的 GTK 男人似乎不帮忙 这个建议 https stackoverflow com a 126145 会告诉您安装了哪个 2 0 的次要版本 不同的主要版本将具有不同的包名称 因为它们可以在系统上

随机推荐

  • JVM 一. 字节码与数据类型相关

    目录 一 字节码 数据类型相关 一 字节码 数据类型相关 字节码文件是跨平台的吗 是的 java虚拟机主要识别字节码文件 其实现在的java虚拟机已经不是单纯的java的 只要语言满足虚拟机的规范 都可以在这个虚拟机上运行 class文件中
  • 一、Kubernetes系列之介绍篇

    Kubernetes介绍 1 背景介绍 云计算飞速发展 IaaS PaaS SaaS Docker技术突飞猛进 一次构建 到处运行 容器的快速轻量 完整的生态环境 2 什么是kubernetes 首先 他是一个 全新的基于容器技术的分布式架
  • 微信回调模式配置企业服务器URL

    转载请标明出处 尊重他人劳动成果 谢谢 前几天微信推出了企业号 我就进去关注了一下 发现用途大大的多 就顺手整了一个测试号来试试 由于是新出的玩意儿 很多东西有文档也不到一定知道整 我这个配置就花了蛮久才找到失败的原因 最终是借用了浩然哥的
  • STL——queue模板类常见函数

    include
  • 【Python数据分析】Pandas按行遍历Dataframe

    Pandas按行遍历Dataframe的方法主要有两种 iterrows 和itertuples 具体用法如下 构建数据集 import pandas as pd import numpy as np N 20 dataset pd Dat
  • 08C++11多线程编程之unique_lock类模板

    08C 11多线程编程之unique lock类模板 前述 如果看懂了该篇文章 你对unique lock可以说随便使用 并且可以只看第5点的总结即可 1 unique lock概念 当不加参数时 和lock guard一样能自动上锁解锁
  • Unity 实现选框选中物体

    最近在看RTS游戏视频注意到了选框功能 就尝试做了一下 功能实现 脚本挂载到Camera上 要不然OnPostRender 函数无法调用 rectMat新建一个材质球 设置成默认的Sprites就可以了 using System Colle
  • Vue的安装和部署

    1 下载安装node js 官网下载地址 https nodejs org en 如果C盘较小 可以自定义安装到其他盘 解压完的目录如下 2 node v npm v 能看到版本号说明安装成功 3 在安装目录下 创建全局安装目录 node
  • Sqlite进阶之--附加数据库关联查询以及Pragma的相关使用

    数据库连接 基本的 Data Source c mydb db Version 3 此类库不支持版本 2 内存数据库 Data Source memory Version 3 New True SQLite 数据库通常存储在磁盘上 但数据库
  • 元组学习python复习资料

    a 第0个 2 3 d 倒1 1 倒1 2 这里我们区分好命名各元祖 1为第0个元祖 2为第1个元祖 csharp 再介绍一下元祖的概念 数据结构在python中包含 元祖 列表和字典 特点1 元祖一旦被创建不能被修改 a 第0个 2 3
  • 简洁直观的飞行器数学模型推导

    运动学方程 动力学方程 值得注意的是 对于非定轴和定轴转动 h r
  • HDOJ 7328 Snake —— 2023“钉耙编程”中国大学生算法设计超级联赛(5)(2023杭电多校第五场)

    题目链接 简要题意 题目相当于求 n n n个人排成 m m m个队伍 队伍内部有序 队伍之间无序 最长的长度不超过 k
  • ffmpeg最简单方式支持nvidia硬编解码

    提示 文章写完后 目录可以自动生成 如何生成可参考右边的帮助文档 文章目录 前言 一 nvidia硬编解码是什么 二 使用步骤 1 安装 2 确认 1 硬件解码器 2 硬件编码器 3 测试 总结 前言 因为工作内容的需要 之前写过一篇文章关
  • java 线程休眠 假死,java多线程-jstack线程阻塞问题排查

    线上高并发时 tomcat挂掉了 应用假死等问题 都可以使用jstack查看线程堆栈的问题 jstack参数解说 首先线程状态如下 New 当线程对象创建时存在的状态 此时线程不可能执行 Runnable 当调用thread start 后
  • IT项目管理(4)

    文章目录 联合同学做一个年级微信公众号加强各班相互了解 联合活动等 请编制项目章程和项目管理计划 指导该项目实施与运营 必须包含 WBS 和 甘特图 项目章程 测量目的 可测量的项目目标和相关的成功标准 高层级需求 高层级项目描述 边界定义
  • 【webots教程】你在webots搭建的第一个仿真环境

    系列文章目录 webots教程 简介与软硬件要求 webots教程 安装 webots教程 关于webots的超详细介绍 webots教程 你在webots搭建的第一个仿真环境 webots教程 编写你的第一个控制器 webots教程 简单
  • AI工程师职业经验指南——前新浪微博资深算法工程师告诉你怎样成为一名合格的推荐系统工程师

    本文转载自 程序员 杂志2017年11月期 推荐系统工程师成长路线图 Item based collaborative filtering recommendation algorithms 这篇文章发表于2001年 在Google学术上显
  • Android完全退出程序

    有过Android开发经验的人都知道 应用程序点击返回键或者代码显示调用了Activity finish 方法都无法完全退出 他们还在进程中 下面分享下Android客户端完全退出程序的方法 1 添加权限
  • WPS VAB支持库下载

    WPS VAB支持库下载 WPS VAB支持库下载 新的改变 功能快捷键 合理的创建标题 有助于目录的生成 如何改变文本的样式 插入链接与图片 如何插入一段漂亮的代码片 生成一个适合你的列表 创建一个表格 设定内容居中 居左 居右 Smar
  • Linux进程编程(PS: exec族函数、system、popen函数)

    目录 1 进程相关概念 程序和进程 查看系统中的进程 ps指令 top指令 进程标识符 使用getpid 获取 父进程 子进程 2 创建进程fork 进程创建发生了什么 C程序的存储空间如何分配 3 创建进程vfork 区别fork 4 进