8.1IO进程线程

news2025/1/16 21:50:08

笔记

进程

一.多进程引入

1.1引入目的

程序员写程序时,一个程序可能由多个任务组成,如果使用的是单进程,或单任务,那么该任务执行阻塞时,其他任务就无法执行,必须等到该任务解除阻塞后,才能去执行其他任务。

多进程或多线程,可以解决同一个程序中多个任务并发执行的情况

1.2进程的概念

        1>进程是程序的一次执行过程

        2>进程是程序资源分配的基本单位,系统会分给每个进程分配4G的虚拟内存,分为0--3G的用户空间和3--4G的内核空间

                多个进程共享内核空间,用户空间相互独立

        3>进程是一个动态的过程,有生命周期的概念,分为创建态、就绪态、运行态、阻塞态、死亡态

        4>进程在内核空间中存储在一个名叫task_struct中的结构体中(PCB)

task_struct:

进程描述符:task struct包含了描述一个进程所需的所有信息。
进程状态:包括运行、就绪、阻塞等状态。
进程标识符:如进程ID(PID)。
进程调度信息:如优先级、调度策略等。
内存管理信息:如虚拟地址空间、页表等。
文件系统信息:如打开的文件、文件系统根目录等
信号处理:包括待处理信号和信号处理函数。
进程间通信:如消息队列、共享内存等IPC机制相关信息。
时间和定时器:如进程创建时间、CPU使用时间等。
线程信息:在Linux中,线程被视为轻量级进程,也用taskstruct表示。

        5>单核cpu处理多任务时,一般使用的是时间片轮询机制

        6>进程与程序的区别:程序是静态的,是存储在磁盘上的二进制代码

        7>进程是动态的,是有生命周期的7>进程的组成:进程控制块(PCB)、数据段、程序段

1.3 进程的种类

进程一共分为三大类:交互进程、批处理进程、守护进程

        1>交互进程:他是由shel控制,用于直接跟用户进行交互的进程。例如:vim编辑器、文本编辑器

        2>批处理进程:本质上维护了一个队列,被放入队列中的进程会统一被调度。例如gcc编译器的一步到位的编译

        3>守护进程:脱了了终端而存在的进程,随着系统的启动而开始,随着系统的结束而终止。例如:服务进程3

1.4 进程号的概念

每个进程在系统中都有一个唯一的标识位,用一个整数表示,这就是该进程的进程号(PID)

        1>PID(processID):当前进程的进程号

        2>PPID(parent process lD):当前进程的父进程的进程号

每个进程都是由其父进程进行拷贝赋值出来的。

可以进入 /proc目录中的每个数字都是现在正在执行的一个进程

1.5 特殊的进程
        

        1>0号进程:也成为 idel进程,他是操作系统启动后执行的第一个进程,这个进程也叫空闲进

程,当没有其他进程执行时,系统会默认执行该进程。1号进程和2号进程都是由0号进程创建出来

的。

        2>1号进程:也称init进程,该进程由0号进程产生,主要完成系统创建时一些软件硬件的初始化

工作。当其他进程的父进程死亡后,会托管其子进程

        3>2号进程:也称kthreadd,该进程由0号进程产生,也成为调度进程,当某个就绪进程时间片

轮到时,该进程负责进程的调度

工作

        4>孤儿进程:当前进程的父进程死亡后,但是当前进程还没有结束,那么当前进程称为孤儿进

程,孤儿进程会由1号进程收养57僵尸进程:当前进程已经死亡,但是其父进程没有为其收尸,那么

该进程为僵尸进程

1.6 进程的相关指令

        1>查看进程信息的命令:ps,跟不同的选项,执行不同的状态

                ps-ef:显示进程之间的关系

                ps-ajx:显示进程的状态

                ps-aux:可以查看进程资源使用情况

        2>top或htop        可以动态展示进程的占用情况 

        3>pstree:展示进程树,可以显示进程的父子关系

        4>查看给定进程的进程号:pidof 进程名ki:向进程发送信号,发信号的格式

                        ki -信号名(号)进程号

                        能够发送的信号号有:可以通过指令kil-l查看

1.7进程的状态

        1>主要状态一共有五个:创建态、就绪态、运行态、阻塞态、死亡态

        2>程序中的进程的状态显示:可以通过指令man ps查看进程的状态

                进程的状态由两部分组成:主状态和附加态

主状态
                D                不可中断的休眠态(usually I0)
                R                运行态(on run queue)
                S                可中断的体眠态(waiting for an event to complete)
                T                暂停态,会给出作业号进行控制
               W                已经弃用
                X                死亡态(should never be seen)
                Z                僵尸态

附加态

                <                高优先级的进程(not nice to other users)

                N                低优先级的进程(nice to other users)

                L                锁到内存中的进程(for real-time and custom Io)

               S                会话组组长,默认为当前终端

                I                包含多线程的进程(using CLONE_THREAD,like NPTL pthreads do)

               +                表示是前台运行的进程

        3>进程状态切换的实例

二.多进程编程

        多线程允许使用同一个线程体函数

2.1 fork进程的创建函数

        1> 在进程的创建过程,其实就是通过拷贝父进程的task struct结构体而来的,子进程中保存

了大部分的父进程的遗传基因,只需要修改少部分的内容,如子进程的进程号,子进程的父进程

号。

        2>子进程和父进程的资源用户空间是完全独立的,创建出子进程后,父子进程的资源相互独立,

互不影响

        3>进程的创建,通过fork函数完成

        4>当子进程创建后,会跟父进程一起执行fork后面的语句

#include<sys/types.h>

#include<unisted.h>

pid_t fork(void)

功能:拷贝父进程得到子进程

参数:无

返回值:成功时,父进程中返回子进程中的pid号,子进程中返回0,失败返回-1并置位错误码

注意:创建出的子进程跟父进程没有先后执行的顺序,遵循时间片轮询机制

2.2进程号的获取

#include<sys/types.h>

#include<unsitd.h>

pid_t getpid(void)

功能:获取当前进程的进程号

参数:无

返回值:当前进程的pid

pid_t getppid(void)

功能:获取当前进程的父进程的pid号

参数:无

返回值:当前进程的父进程的pid

2.3进程的退出(exit、_exit)

#include<stdlib.h>

void exit(int status);

功能:退出当前进程,并刷新当前进程打开的标准IO文件指针的缓冲区

参数:退出时的状态

        EXIT_SUCCESS:成功退出

        EXIT_FAILURE:失败退出

返回值:无

#include<stdlib.h>

void _exit(int status);

功能:退出当前进程,但是不刷新当前进程打开的标准IO文件指针的缓冲区

参数:退出时的状态

        EXIT_SUCCESS:成功退出

        EXIT_FAILURE:失败退出

返回值:无

2.4进程资源的回收

#include<sys/types.h>

#include<sys/wait.h>

pid_t wait(int *wstatus)

功能:阻塞回收子进程的资源,如果没有进程退出,那么就阻塞等待

参数:子进程退出时的状态,一般不接受,直接填NULL即可

返回值:成功返回回收的子进程的pid,失败返回-1并置位错误码

#include<sys/types.h>

 

#include<sys/wait.h>

pid_t waitpid(pid_t pid,int *wststus,int options)

功能:既可以阻塞也可以非阻塞形式回收僵尸进程

参数1:要回收的进程号

                >0:表示回收具体的某个进程

                =0:能够回收当前进程所在的进程组中的任意一个子进程

                =-1:回收任意一个子进程

                <-1:回收别的进程组(进程组id为给定的pid的绝对值)中的任意一个子进程

参数2:子进程退出时的状态,一般不接收,直接填NULL即可

参数3:表示是否阻塞回收僵尸进程

        0:表示阳塞        

        WNOHANG:表示非阳塞

返回值:

        >0如果成功回收了子进程资源,那么会返回该子进程的pid

        =0:表示以非阳塞的形式回收资源,但是没有回收到子进程

        =-1:失败返回-1并置位错误码

2.5创建三个进程

可以让父进程创建一个子进程,再由父进程或者子进程创建一个子进程

2.6写时拷贝技术

2.7特殊进程的验证

        1>僵尸进程

#include<myhead.h>

int main(int argc, const char *argv[])
{
    pid_t pid = fork();      //创建出子进程

    if(pid < 0)
    {
        perror("fork error");
        return -1;
    }else if(pid == 0)
    {
        printf("我是子进程\n");
        exit(EXIT_SUCCESS);         //退出子进程
    }

    //父进程内容
    printf("我是父进程\n");
    sleep(5);
    return 0;
}

        2>孤儿进程

#include<myhead.h>

int main(int argc, const char *argv[])
{
    pid_t pid = fork();      //创建出子进程

    if(pid < 0)
    {
        perror("fork error");
        return -1;
    }else if(pid > 0)
    {
        printf("我是父进程\n");
        exit(EXIT_SUCCESS);         //退出子进程
    }

    //父进程内容
    printf("我是子进程\n");
    sleep(5);
    return 0;
}

        3>创建守护进程

#include <myhead.h>

int main(int argc, const char *argv[])
{
	pid_t pid=fork();
	if(pid<0)
	{
		perror("fork error");
		return -1;
	}else if(pid>0)
	{
		//退出父进程
		exit(EXIT_SUCCESS);
	}

	//将自己设置成会话组组长
	setsid();
	//更改操作目录为根目录
	chdir("/");
	//获得创建文件的最大权限
	umask(0);
	//将标准输入输出出错重定向到long.txt文件中
	int fd =open("./long.txt",O_WRONLY|O_CREAT|O_APPEND,0664);
	if(-1==fd)
	{
		perror("open error");
		return -1;
	}
	//重定向
	dup2(fd,STDIN_FILENO);
	dup2(fd,STDOUT_FILENO);
	dup2(fd,STDERR_FILENO);
	//执行操作
	while(1)
	{
		printf("hello!\n");
		fflush(stdout);
		sleep(1);
	}
	close(fd);
	return 0;
}

线程

一.多线程基本概念

        1> 线程:也称为轻量版的进程(LWP),是更小的任务执行单元,是进程的一个执行路径

        2> 线程是任务器调度的最小单位

        3> 一个进程中可以包含多个线程,多个线程共享进程的资源。

        4> 线程几乎不占用资源,只是占用了很少的用于程序状态的资源(大概有8k左右)

        5> 由于多个线程共同使用进程的资源,导致,线程在操作上容易出现不安全的状态

        6> 线程操作开销较小、任务切换效率较高

        7> 一个进程中,至少要包含一个线程(主线程)

        8> 在有任务执行漫长的IO等待过程中,可以同时执行其他任务

        9> linux中不直接支持线程相关的支持库,需要引入第三方库,线程支持库

                sudo apt-get install manpages-posix manpages-posix-dev

                如果程序中使用的线程支持库中的函数,编译程序时,需要加上 -lpthread 选项

二.线程支持函数(多线程编程)

多线程允许使用同一个线程体函数

2.1pthread_create:创建线程
 #include <pthread.h>

       int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
 void *(*start_routine) (void *), void *arg);
       功能:在当前进程中,创建一个分支线程
       参数1:用于接收创建好的线程ID
       参数2:线程的属性,一般填NULL表示使用系统默认的属性创建线程
       参数3:线程体函数,需要传递一个函数,参数为void*,返回值为void*
       参数4:参数3的参数
       返回值:成功创建返回0,失败返回一个错误码,注意,不是内核提供的错误码   
#include<myhead.h>

//定义一个用于传递数据的结构体类型
struct Buf
{
    int num;
    double key;
};


//定义线程体函数
void *task(void *arg)
{
    int num = (*((struct Buf*)arg)).num;
    double key = ((struct Buf*)arg)->key;

    while(1)
    {
        printf("我是分支线程, num = %d, key=%.2lf\n", num, key);
        sleep(1);
    }
}


/************************主程序*****************************/
int main(int argc, const char *argv[])
{
    int num = 520;       //定义整形变量
    double key = 1314;   //定义浮点型数据

    struct Buf buf = {num, key};    //封装要传递的数据

    //定义变量存储线程号
    pthread_t tid = -1;

    //创建分支线程
    if(pthread_create(&tid, NULL, task, &buf) != 0)
    {
        printf("pthread_create error\n");
        return -1;
    }


    printf("我是主线程,tid = %#lx\n", tid);

    //while(1);
    getchar();

    return 0;
}
2.2pthread_self:线程号的获取
       #include <pthread.h>
       pthread_t pthread_self(void);
    功能:获取当前线程的线程号
    参数:无
    返回值:当前线程的线程号
2.3pthread_exit:线程退出函数
       #include <pthread.h>

       void pthread_exit(void *retval);
    功能:退出当前的线程
    参数:线程退出时的状态,一般填NULL
    返回值:无
2.4pthread_jion:线程资源回收函数
  #include <pthread.h>

       int pthread_join(pthread_t thread, void **retval);
    功能:阻塞等待给定线程的退出,并回收该线程的资源
    参数1:要回收的线程号
    参数2:接收线程退出时的状态
    返回值:成功返回0,失败返回错误码
2.5pthread_detach:线程分离太态
       #include <pthread.h>

       int pthread_detach(pthread_t thread);
    功能:将线程设置成分离态,设置了分离态的线程,退出后,由系统回收其资源
    参数:要被分离的线程id号
    返回值:成功返回0,失败返回错误码
2.6 pthread_setcancelstate:设置取消属性
     #include <pthread.h>

       int pthread_setcancelstate(int state, int *oldstate);
    功能:设置是否接收取消指令
    参数1:新的状态
        PTHREAD_CANCEL_ENABLE:可接受状态
        PTHREAD_CANCEL_DISABLE:不可接收状态
    参数2:线程的旧的状态容器,如果不愿意要之前的状态,填NULL即可
    返回值:成功返回0,失败返回错误码
   返回值:成功返回0,失败返回错误码

#include<myhead.h>



//定义线程体函数
void *task(void *arg)
{
    //设置线程不可取消状态
    pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, NULL);

    printf("我是分支线程,线程号为:%#lx\n", pthread_self());

    //exit(EXIT_SUCCESS);        //退出进程
    int count = 0;
    while(1)
    {
        printf("我真的还想再活五百年。。。。\n");
        sleep(1);
        count++;
        if(count == 10)
        {
            break;
        }
    }
    pthread_exit(NULL);           //退出线程
}

/************************主程序*****************************/
int main(int argc, const char *argv[])
{

    //定义变量存储线程号
    pthread_t tid = -1;

    //创建分支线程
    if(pthread_create(&tid, NULL, task, NULL) != 0)
    {
        printf("pthread_create error\n");
        return -1;
    }

    printf("我是主线程,tid = %#lx, 主线程线程号:%#lx\n", tid, pthread_self());

    /*回收分支线程的资源
    if(pthread_join(tid, NULL) == 0)
    {
        printf("成功回收了%#lx的资源\n", tid);
    }*/

    //将线程设置成分离态
    pthread_detach(tid);

    printf("线程已经分离\n");
    
    //休眠5秒
    sleep(5);
    //向分之线程中发送一个取消请求
    pthread_cancel(tid);


    //while(1);
    getchar();

    return 0;
}

作业

使用两个线程完成两个文件的拷贝,分支线程1拷贝前一半,分支线程2拷贝后一半,主线程回收两个分支线程的资源

#include <myhead.h>
//定义数据结构体
typedef struct data
{
	const char *argv1;
	const char *argv2;
	int size;
	int len;
}Data;
//定义复制函数
void* copy(void *data)
{
	//将数据结构体中的值取出
	const char*argv1=((Data*)data)->argv1;
	const char*argv2=((Data*)data)->argv2;
	int size=((Data*)data)->size;
	int len=((Data*)data)->len;
	int fp=-1;
	//打开被复制文件
	if((fp=open(argv1,O_RDONLY))==-1)
	{
		perror("open argv[1] error");
		return NULL;
	}
	int fd=-1;
	//打开准复制文件
	if((fd=open(argv2,O_WRONLY|O_CREAT|O_TRUNC,0664))==-1)
	{
		return NULL;
	}
	//移动光标到指定位置
	lseek(fp,size,SEEK_SET);
	lseek(fd,size,SEEK_SET);
	char temp=0;
	//按给定长度提取字符并写入到文件中
	for (int i=0; i<len;i++)
	{
		read(fp,&temp,sizeof(temp));
		write(fd,&temp,sizeof(temp));
	}
	//关闭文件
	close(fp);
	close(fd);
	printf("拷贝成功!\n");

}
int main(int argc, const char *argv[])
{
	//判断传入的文件是否为三个
	if(argc!=3)
	{
		perror("file input error");
		return-1;
	}
	//打开被复制文件
	int fp =-1;
	if((fp=open(argv[1],O_RDONLY))==-1)
	{
		perror("open argv[1] error");
		return-1;
	}
	//计算文件大小
	int size=lseek(fp,0,SEEK_END);
	//分支线程复制长度
	int len=size-size/2;
	//关闭文件
	close(fp);
	//定义传入函数的数据变量,不能只定义一个变量,因为传入函数的数据是地址传递,两个线程要调用两次函数,如果一个线程修改变量,另一个也会修改
	Data data1;
	Data data2;
	data1.argv1=argv[1];
	data1.argv2=argv[2];
	//分支线程复制位置的值赋给数据结构体
	data1.size=size/2;
	//分支线程复制长度的值赋给数据结构体
	data1.len=len;
	//创建分支线程
	pthread_t tid=-1;
	if(pthread_create(&tid,NULL,copy,&data1)!=0)
	{
		perror("pthread error");
		return -1;
	}
	//主线程复制需要的数据的值赋给数据结构体
	data2.argv1=argv[1];
	data2.argv2=argv[2];
	data2.size=0;
	data2.len=size/2;
	//调用复制函数
	copy(&data2);
	//阻塞回收分支线程资源
	pthread_join(tid,NULL);
	return 0;
}

思维导图

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

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

相关文章

【Linux】(26) 详解磁盘与文件系统:从物理结构到inode机制

目录 1.认识磁盘、 1.1 理论 1.2 磁盘的物理结构 CHS 寻址 1.3 磁盘的逻辑抽象结构 2. inode 结构 1.Boot Block 启动块 2.Super Block&#xff08;超级块&#xff09; 3.Group Descriptor Block&#xff08;块组描述符&#xff09; 4.Data Blocks (数据块) 5.Inode…

spring boot 实现 Stream 钉钉事件订阅

1: 参考链接 https://open.dingtalk.com/document/orgapp/develop-stream-mode-push-server 2&#xff1a;钉钉开放平台订阅配置 配置之后运行一下上面提供的链接 里面的main方法&#xff0c;验证通道 3&#xff1a;订阅启动方式 EventListenerThread eventListenerThrea…

玩转ComfyUI】基于函数计算一键部署AI生图平台 ComfyUI

登录阿里云 1.前往函数计算控制台。 在推荐您创建 AliyunFcDefaultRole默认角色对话框中&#xff0c;单击创建。 3.在角色快捷创建页面&#xff0c;单击同意授权。 4.在实验操作面板顶部&#xff0c;单击 图标打开新的标签页&#xff0c;然后在地址栏中输入如下地址并回车&…

会展信息服务预约小程序开展品牌管理

如舞台、漫展、商业展会等场景都需要会展服务&#xff0c;场地建设、方案提供、人员配备&#xff0c;尤其是高线城市&#xff0c;服务开展度比较高&#xff0c;客商双方为多次互需关系&#xff0c;服务方需要拓展品牌获得更多客源加以转化&#xff0c;而客户则需要找到靠谱且具…

LogonTracer图形化事件分析工具

LogonTracer这款工具是基于Python编写的&#xff0c;并使用Neo4j作为其数据库&#xff08;Neo4j多用于图形数据库&#xff09;&#xff0c;是一款用于分析Windows安全事件登录日志的可视化工具。它会将登录相关事件中的主机名&#xff08;或IP地址&#xff09;和帐户名称关联起…

vscode中运行不同语言的文件出现报错?

vscode是一个不错的语言编辑软件&#xff0c;可以利用插件运行不同的语言&#xff0c;从而大大减少了软件的下载数量。 今天在学习的时候出现了一个令人匪夷所思的报错问题&#xff1a;早些时候一直在运行Python语言的文件&#xff0c;后来想对比一下c语言与Python的不同。于是…

阿里云、华为云、腾讯云、亚马逊云与谷歌云全景推荐

探索未来&#xff0c;选择领先云服务 —— 阿里云、华为云、腾讯云、亚马逊云与谷歌云全景推荐 在这个数字化飞速发展的时代&#xff0c;云计算已成为企业和个人日常运营不可或缺的一部分。无论是小型初创公司还是全球性大企业&#xff0c;选择合适的云服务可以大幅提升业务效…

Redis——集合 SET

目录 1. 添加元素 SADD 2. 查看元素 SMEMBERS 3. 判断元素是否存在该集合 SISMEMBER 4. 删除元素 SREM 集合 SET 是一种无序集合&#xff1b;因此其与列表有以下区别&#xff1a; &#xff08;1&#xff09;列表是有序的&#xff0c;集合是无序的&#xff1b; &#xff0…

基于SpringBoot宠物店寄养管理系统 毕业设计-附源码94084

摘 要 宠物店寄养管理系统是一个基于计算机技术的应用系统&#xff0c;旨在为宠物店提供一种便捷、高效的方式来管理宠物店寄养服务。本研究通过使用现代化的软件开发技术和数据库管理方法&#xff0c;设计和实现了一个全面的宠物店寄养管理系统。该系统具有以下主要功能:宠物资…

ONLYOFFICE文档:为企业和开发者带来强大的文档编辑功能

本文给大家介绍一个开源项目&#xff1a;ONLYOFFICE文档&#xff0c;它能够为文档编辑、多人协作提供强大支持。无论你是个人使用&#xff0c;还是企业、商业开发&#xff0c;都能找到适合你的版本。 关于 ONLYOFFICE 文档 ONLYOFFICE 文档是一套功能强大的文档编辑器&#x…

DockerCompose部署示例

目录 前言 1. 初识DockerCompose 2. 安装DockerCompose 3. 部署微服务项目 1&#xff09;找一个目录&#xff0c;创建一个新的cloud-demo文件夹。 2&#xff09;在cloud-demo文件夹创建一个docker-compose.yml文件&#xff0c;然后编写下面内容&#xff1a; 3&#xff09…

手把手构建Netty

1.Netty基础 Netty是一个提供了易于使用的API的客户端、服务器框架&#xff1b; 并发高-NIO(非阻塞IO) 传输快-零拷贝: 分析&#xff1a; 使用了NIO的零拷贝&#xff1b;java中内存是分为堆和栈&#xff0c;还有字符串常量池等等&#xff1b; 如果有一些数据需要从IO中读取&am…

28、美国国家冰雪中心(NSIDC)海冰密集度月数据处理:方法二

文章目录 1. 前言2. polarstereo-lonlat-convert-py 介绍2.1 安装方法3. 代码4. 其他1. 前言 在前文中已经详细介绍了如何使用Python对NSIDC的海冰密集度数据进行处理,将其从极地投影转换成常用的经纬度投影,但是方法较为麻烦,本文将基于另一个库,使用更加简便的方法对其进…

JVM—垃圾收集算法和HotSpot算法实现细节

1、分代回收策略 分代的垃圾回收策略&#xff0c;是基于这样一个事实&#xff1a;不同的对象的生命周期是不一样的。因此&#xff0c;不同生命周期的对象可以采取不同的收集方式&#xff0c;以便提高回收效率。 分代垃圾回收采用分治的思想&#xff0c;进行代的划分&#xff0…

π 随机数计算圆周率

如下图&#xff0c;圆与正方形面积比值等于圆内的点和总点数的比值 Disatance是点到原点的距离&#xff1a;SQRT(A2xA2B2xB2) 以下是在Excel中1.5万个点计算圆周率的结果。 InCircle公式&#xff1a;IF(C2<1,1,0) π计算公式&#xff1a;4*SUM(D2:D15000)/COUNT(D2:D15000)

leetcode 2181.合并零之间的结点

1.题目要求: /*** Definition for singly-linked list.* struct ListNode {* int val;* struct ListNode *next;* };*/struct ListNode* mergeNodes(struct ListNode* head){struct ListNode* cur head;int count 0;//1.遍历结点&#xff0c;求出结点数while(cur){co…

前后端demo-WarehouseManagement

前端 数据库 其他 1.git下来&#xff0c;解决依赖问题&#xff0c;前端报错因为字体文件丢失&#xff0c;下载字体放到fonts文件夹字体.zip官方版下载丨最新版下载丨绿色版下载丨APP下载-123云盘 2.后端login验证&#xff0c;前端需要账号格式&#xff0c;linqq.com 3.自己…

【React】实现消息列表的删除

工作小记&#xff0c;第一次接触react项目 1.增加删除对话项的函数 hooks\use-conversation.ts // 删除对话项的函数const deleteConversation (id: string) > {setConversationList(prevList > prevList.filter(item > item.id ! id))}return {deleteConversation,.…

OZON户外运动装备热卖产品

OZON平台上的户外运动装备热卖产品涵盖了多个类别&#xff0c;这些产品不仅满足了俄罗斯消费者对户外运动的需求&#xff0c;还体现了市场趋势和消费者偏好的变化。以下是一些主要的热卖户外运动装备产品&#xff1a; OZON户外运动装备热卖产品地址&#xff1a;D。DDqbt。COm/…

第十九天培训笔记

上午 1 、构建 vue 发行版本 [rootserver eleme_web]# nohup npm run serve& // 运行 vue 项目 [rootserver eleme_web]# mkdir /eleme [rootserver eleme_web]# cp -r /root/eleme_web/dist/* /eleme/ // 将项目整体 移动到 /eleme 目录下 [rootserver eleme_web]# …