0521_网络编程5

练习1:

 

TFTP通信过程总结

  1. 服务器在69号端口等待客户端的请求
  2. 服务器若批准此请求,则使用 临时端口 与客户端进行通信。
  3. 每个数据包的编号都有变化(从1开始)
  4. 每个数据包都要得到ACK的确认,如果出现超时,则需要重新发送最后的数据包或ACK包
  5. 数据长度以512Byte传输的,小于512Byte的数据意味着数据传输结束。
#include <myhead.h>
#define SER_PORT 69
void download(int fd,struct sockaddr_in sin);
void upload(int fd,struct sockaddr_in sin);
int main(int argc, const char *argv[])
{/*1.判断终端输入 */if(argc !=2 ){printf("ip未输入\n");exit(-1);}/*2. 创建套接字 */int fd;if((fd=socket(AF_INET,SOCK_DGRAM,0))<0){perror("socket");exit(1);}/*3. 向结构体中录入服务器信息 */struct sockaddr_in sin;sin.sin_family=AF_INET;sin.sin_port=htons(SER_PORT);sin.sin_addr.s_addr=inet_addr(argv[1]);/*4. 功能实现 */while(1){/*4.1 打印提示表 */printf("**********************\n");printf("********1.下载********\n");printf("********2.上传********\n");printf("********3.退出********\n");printf("**********************\n");/*4.2. 文件传输 */int S_opcode=0;scanf("%d",&S_opcode);while(getchar()!=10);switch (S_opcode){/*7.1 下载文件  */case 1:download(fd,sin);break;/*7.2 上传文件 */case 2:upload(fd,sin);break;case 3:goto _EXIT;break;default:printf("输入不正确\n");break;}}/*5. 退出 */
_EXIT:close(fd);return 0;
}
/* 下载 */
void download(int fd,struct sockaddr_in sin){int num=1;int flag=0;int wpid=-1;char Rsever_buf[516]={0};/*1. 与服务端进行数据传输 *//*1.1 初始化 */char Wsever_buf[516]="";//写入服务端的字符串/*1.2 第一次数据建立准备 *//*1.2.1 文件操作码选择 */short *opcode=(short *)Wsever_buf;*opcode=htons(1);/*1.2.2 文件名传输 */char *filename=Wsever_buf+2;printf("请输入文件名:");char filestr[32]="";fgets(filestr,32,stdin);filestr[strlen(filestr)-1]=0;strcpy(filename,filestr);/*1.2.3 传输模式 */char *mode=filename+strlen(filename)+1;strcpy(mode,"octet");/*1.2.4 数据大小统计 */int size=2+strlen(filename)+strlen(mode)+2;/*2. 第一次读写请求传输 */if(sendto(fd,Wsever_buf,size,0,(struct sockaddr *)&sin,sizeof(sin))<0){perror("sendto");exit(-1);}	while(1){/*3. 读取服务端数据包 */bzero(Rsever_buf,sizeof(Rsever_buf));socklen_t addrlen=sizeof(sin);ssize_t recv_len=-1;if((recv_len=recvfrom(fd,Rsever_buf,sizeof(Rsever_buf),0,(struct sockaddr *)&sin,&addrlen))<0){perror("recvfrom");exit(-1);}/*3. 判断操作码及编号 */short *r_opcode =(short *)Rsever_buf;short *r_num=r_opcode+1;switch (htons(*r_opcode)){/*3.1 正确数据包处理*/case 3:if(*r_num==htons(num)){printf("正在下载中\n");num++;break;}else{printf("数据包编号错误\n");exit(-1);}/*3.2 错误数据包提示 */case 5:if(0==htons(*r_num)){printf("未定义,差错错误信息\n");return;}else if(1==htons(*r_num)){printf("File not found\n");return;}else if(2==htons(*r_num)){printf("Access violation\n");return;}else if(3==htons(*r_num)){printf("Disk full or allocation exceeded\n");return;}else if(4==htons(*r_num)){printf("illegal TFTP operation\n");return;}else if(5==htons(*r_num)){printf("Unknown transfer ID\n");return;}else if(6==htons(*r_num)){printf("File already exists\n");return;}else if(7==htons(*r_num)){printf("No such user\n");return;}else if(8==htons(*r_num)){printf("Unsupported option(s) requested\n");return;}break;default:break;}/*4. 打开文件 */if(0==flag){wpid=open(filename,O_WRONLY | O_CREAT |O_TRUNC,0666);if(wpid<0){perror("open");exit(-1);}flag=1;}/*5. 写入文件 */if(write(wpid,Rsever_buf+4,recv_len-4)<0){perror("write");break;}/*6. ACK确认回馈 */char ACK[4]={0};short *w_opcode=(short *)ACK;*w_opcode=htons(4);short *w_num=w_opcode+1;*w_num=*r_num;/*7. 传输ACK确认包 */if(sendto(fd,ACK,4,0,(struct sockaddr *)&sin,sizeof(sin))<0){perror("sendto");exit(-1);}/*8. 判断文件是否下载结束 */if(recv_len<516){printf("文件下载完成\n");close(fd);break;}}return;
}
/*上传*/
void upload(int fd,struct sockaddr_in sin){int num=0;int flag=0;char Rsever_buf[516]={0};/*1. 与服务端进行数据传输 *//*1.1 初始化 */char Wsever_buf[516]="";//写入服务端的字符串/*1.2 第一次数据建立准备 *//*1.2.1 文件操作码选择 */short *opcode=(short *)Wsever_buf;*opcode=htons(2);/*1.2.2 文件名传输 */char *filename=Wsever_buf+2;printf("请输入文件名:");char filestr[32]="";fgets(filestr,32,stdin);filestr[strlen(filestr)-1]=0;strcpy(filename,filestr);/*1.2.2.1. 打开文件 */int wpid=-1;if((wpid=open(filename,O_RDONLY))<0){printf("文件名不存在,请重新输入\n");return;}/*1.2.3 传输模式 */char *mode=filename+strlen(filename)+1;strcpy(mode,"octet");/*1.2.4 数据大小统计 */int size=2+strlen(filename)+strlen(mode)+2;/*2. 第一次读写请求传输 */if(sendto(fd,Wsever_buf,size,0,(struct sockaddr *)&sin,sizeof(sin))<0){perror("sendto");exit(-1);}	while(1){/*3. 读取服务端数据包 */bzero(Rsever_buf,sizeof(Rsever_buf));socklen_t addrlen=sizeof(sin);ssize_t recv_len=-1;if((recv_len=recvfrom(fd,Rsever_buf,sizeof(Rsever_buf),0,(struct sockaddr *)&sin,&addrlen))<0){perror("recvfrom");exit(-1);}/*3. 判断操作码及编号 */short *r_opcode =(short *)Rsever_buf;short *r_num=r_opcode+1;switch (htons(*r_opcode)){/*3.1 正确数据包处理*/case 4:if(*r_num==htons(num)){printf("正在上传中\n");num++;*r_num=htons(num);break;}else{printf("数据包编号错误\n");break;}/*3.2 错误数据包提示 */case 5:if(0==htons(*r_num)){printf("未定义,差错错误信息\n");return;}else if(1==htons(*r_num)){printf("File not found\n");return;}else if(2==htons(*r_num)){printf("Access violation\n");return;}else if(3==htons(*r_num)){printf("Disk full or allocation exceeded\n");return;}else if(4==htons(*r_num)){printf("illegal TFTP operation\n");return;}else if(5==htons(*r_num)){printf("Unknown transfer ID\n");return;}else if(6==htons(*r_num)){printf("File already exists\n");return;}else if(7==htons(*r_num)){printf("No such user\n");return;}else if(8==htons(*r_num)){printf("Unsupported option(s) requested\n");return;}break;default:break;}/*4. 读取文件内容 */int retval=-1;if((retval=read(wpid,Rsever_buf+4,512))<0){perror("read");break;}else if(0==retval){printf("文件上传完毕\n");return;}/*5. 文件上传 */*r_opcode=htons(3);if(sendto(fd,Rsever_buf,retval+4,0,(struct sockaddr *)&sin,sizeof(sin))<0){perror("sendto");return;}}return;
}

 练习2:流式域套接字

TCP服务端实现

#include <myhead.h>
int main(int argc, const char *argv[])
{int sfd=-1;if((sfd=socket(AF_UNIX,SOCK_STREAM,0))<0){perror("socket");exit(-1);}if(access("./unix",F_OK)==0){                                                      if(unlink("./unix")<0){perror("unlink");exit(-1);}}struct sockaddr_un sun;sun.sun_family=AF_UNIX;strcpy(sun.sun_path,"./unix");if(bind(sfd,(struct sockaddr *)&sun,sizeof(sun))<0){perror("bind");exit(-1);}if(listen(sfd,128)<0){perror("listen");exit(-1);}struct sockaddr_un cun;socklen_t addlen=sizeof(cun);int newfd=accept(sfd,(struct sockaddr *)&cun,&addlen);if(newfd<0){perror("accept");exit(-1);}printf("[%s]:发来连接请求\n",cun.sun_path);char rbuf[128]="";while(1){bzero(rbuf,sizeof(rbuf));int res=recv(newfd,rbuf,sizeof(rbuf),0);if(0==res){perror("客户端已下线 \n");break;}printf("[%s]:%s\n",cun.sun_path,rbuf);strcat(rbuf,"*_*");send(newfd,rbuf,strlen(rbuf),0);printf("发送成功\n");}close(newfd);close(sfd);return 0;
}

TCP客户端实现

#include <myhead.h>
int main(int argc, const char *argv[])
{int sfd=-1;if((sfd=socket(AF_UNIX,SOCK_STREAM,0))<0){perror("socket");exit(-1);}if(access("./linux",F_OK)==0){if(unlink("./linux")<0){perror("unlink");exit(-1);}}struct sockaddr_un sun;sun.sun_family=AF_UNIX;strcpy(sun.sun_path,"./linux");if(bind(sfd,(struct sockaddr *)&sun,sizeof(sun))<0){perror("bind");exit(-1);}struct sockaddr_un cun;cun.sun_family=AF_UNIX;strcpy(cun.sun_path,"./unix");if(connect(sfd,(struct sockaddr *)&cun,sizeof(cun))<0){                                                          perror("connect");exit(-1);}printf("连接成功\n");char rbuf[128]="";char wbuf[128]="";while(1){fgets(wbuf,sizeof(wbuf),stdin);wbuf[strlen(wbuf)-1]=0;send(sfd,wbuf,strlen(wbuf),0);printf("发送成功\n");bzero(rbuf,sizeof(rbuf));recv(sfd,rbuf,sizeof(rbuf),0);printf("服务器发来的消息为:%s\n",rbuf);}close(sfd);return 0;
}

练习3:报式域套接字

UDP客户端实现

#include <myhead.h>
int main(int argc, const char *argv[])
{int fd=-1;if((fd=socket(AF_UNIX,SOCK_DGRAM,0))<0){perror("socket");exit(-1);}if(access("./unix",F_OK)==0){if(unlink("./unix")<0){perror("unlink");exit(-1);}}struct sockaddr_un sun;sun.sun_family=AF_UNIX;strcpy(sun.sun_path,"./unix");if(bind(fd,(struct sockaddr *)&sun,sizeof(sun))<0){perror("bind");exit(-1);}char rbuf[128]=""; struct sockaddr_un cun;socklen_t addrlen;while(1){bzero(rbuf,sizeof(rbuf));recvfrom(fd,rbuf,sizeof(rbuf),0,(struct sockaddr *)&cun,&addrlen);printf("[%s]:%s\n",cun.cun_path,rbuf);}close(fd);return 0;
}                                                                                    

UDP客户端实现

#include <myhead.h>
int main(int argc, const char *argv[])
{int fd=-1;if((fd=socket(AF_UNIX,SOCK_DGRAM,0))<0){perror("socket");exit(-1);}if(access("./linux",F_OK)==0){if(unlink("./linux")<0){perror("unlink");exit(-1);}}struct sockaddr_un sun;sun.sun_family=AF_UNIX;strcpy(sun.sun_path,"./linux");if(bind(fd,(struct sockaddr *)&sun,sizeof(sun))<0){perror("bind");exit(-1);}char wbuf[128]="";struct sockaddr_un cun;cun.sun_family=AF_UNIX;strcpy(cun,sun_path,"./unix");char rbuf[128]="";while(1){bzero(wbuf,sizeof(wbuf));bzero(rbuf,sizeof(rbuf));fgets(wbuf,sizeof(wbuf),stdin);wbuf[strlen(wbuf)-1]=0;sendto(fd,wbuf,strlen(wbuf),0,(struct sockaddr *)&cun,sizeof(cun));printf("发送成功\n");}close(fd);return 0;
}                                                                                     

面试题: 

1.项目中如何实现TCP的并发?
答:
在项目中实现 TCP 的并发通常涉及创建多个独立的执行流程来处理同时发起的多个 TCP 连接。常用的方法有以下几种:

1. **多进程**:服务端为每个新的客户端连接创建一个新的进程。这种方法的优点是简单易理解,每个进程独立运行,容错性好。但缺点是进程创建和切换的开销较大,不适合大量并发连接。

2. **多线程**:服务端为每个新的客户端连接创建一个新的线程。相比多进程,多线程的开销较小,因为线程间切换的成本低于进程间切换,共享资源的管理也比较方便。但多线程程序设计相对复杂,需要妥善处理同步和并发问题。

3. **I/O 多路复用**:通过选择性地阻塞在多个 I/O 上,例如使用 select、poll 或 epoll(在 Linux 上)等系统调用,服务器可以在单线程或少量线程中管理多个客户端连接。当某个 I/O 流准备好进行读写操作时,系统会通知服务器程序,从而进行相应的处理。

在实现 TCP 并发时,还需要考虑到负载均衡、资源管理、错误处理等问题,以确保服务器的稳定和性能。选择哪一种方法取决于具体的应用需求、性能要求和开发资源。


2.TCP通信过程中的三次握手?
答:
TCP的三次握手应用于建立连接:
第一次握手:客户端向服务器发送一个SYN包(同步包),其中包含客户端的初始序列号。此时,客户端进入SYN_SEND状态,等待服务器的确认。
第二次握手:服务器收到SYN包后,会确认客户端的SYN(通过发送一个ACK包,其中ack=客户端的序列号+1),同时服务器也会发送一个自己的SYN包,其中包含服务器的初始序列号。这个包通常被称为SYN+ACK包。此时,服务器进入SYN_RECV状态。
第三次握手:客户端收到服务器的SYN+ACK包后,会再次发送一个ACK包以确认收到服务器的SYN包(ack=服务器的序列号+1)。发送完这个ACK包后,客户端和服务器都进入ESTABLISHED状态,表示连接已建立成功。
简记:同步包→同步包+应答包→应答包

3.四次挥手的过程?
答:
TCP的四次挥手应用于释放连接,需要进行的四个步骤,具体步骤如下:
第一次挥手:当一方(假设为客户端)决定关闭连接时,它会发送一个FIN(Finish)报文段给对方。
第二次挥手:服务器收到FIN报文段后,会对这个FIN报文段进行确认,然后自己也发送一个ACK报文段。
第三次挥手:服务器在发送完ACK报文段后,会等待所有的数据传输完成,然后发送一个FIN报文。
第四次挥手:客户端收到服务器的FIN报文段后,会对其进行确认,并发送一个ACK报文段给服务器。
简记:FIN包→应答包→FIN包→应答包
                         
4.tcp\IP协议分几层?tcp\IP是哪一层?
答:
TCP/IP 协议模型通常分为四层,这些层次分别是:

1. **网络接口层**:也称为链接层或网络访问层,它负责通过网络媒体发送和接收数据包,包括定义电缆的物理和电气标准。

2. **互联网层**:它主要负责数据包在网络中的传输,确保数据包能够跨越多个网络从源到达目的地。互联网协议(IP)就处在这一层。

3. **传输层**:这一层提供端到端的通信服务,确保数据的成功传输。TCP(传输控制协议)和 UDP(用户数据报协议)都处于这一层。

4. **应用层**:它负责处理特定的应用程序细节,提供了网络服务与最终用户应用程序之间的接口。例如,HTTP、FTP、SMTP 等协议都工作在这一层上。

从协议的名称“TCP/IP”来看,“TCP”指的是传输控制协议,位于传输层;“IP”指的是互联网协议,位于互联网层。因此,“TCP/IP”实际上涉及到模型中的两个不同层次:传输层和互联网层。
​​​​​​​
5.UDP为什么丢包,怎样处理?丢包发生在哪一层?为什么?
答:
​​​​​​​​​​​​​​UDP(用户数据报协议)丢包的原因包括:

1. **无连接性**:UDP 是一种无连接的协议,不像 TCP 那样在数据传输之前建立连接。这意味着UDP不保证可靠传输,数据包可能会在传输过程中丢失。

2. **无拥塞控制**:UDP 本身不进行拥塞控制。当网络拥塞时,UDP 继续以相同的速率发送数据包,这可能导致网络设备(如路由器)的缓冲区溢出,从而丢包。

3. **无序传输**:UDP 不保证数据包的顺序,所以数据包到达顺序可能与发送顺序不同,接收方需要处理可能的乱序问题。

4. **无错误校验重传机制**:TCP 有检验和及确认应答机制,并会重传损坏的数据包。而 UDP 只有基本的检验和功能,但不会重传损坏或丢失的数据包。

处理 UDP 丢包的方法包括:

- **应用层重传机制**:在需要可靠性的应用中,可以在应用层实现重传机制。如果在预定的时间内没有收到数据包的应答,发送方可以重发数据包。

- **序列号**:在数据包中引入序列号,以便接收方能够检测到丢包事件,并且能够将乱序到达的数据包重新排列成正确的顺序。

- **冗余数据**:使用前向错误校正(FEC)等技术,在传输数据中加入额外的冗余数据,这样即使某些数据包丢失,接收方也可能恢复出原始数据。

- **调节发送速率**:根据网络的实时状况调整数据的发送速率,避免因网络拥塞造成丢包。

- **使用其他协议**:对于需要高可靠性的应用,可以考虑使用更可靠的协议,如 TCP,或者更先进的协议,例如 QUIC。

- **应用选择重传(Selective Acknowledgment, SACK)**:这是一种在接收端告知发送端只重传丢失的数据包,而不是所有后续数据的机制。

丢包发生在网络层原因:

网络拥堵:当网络中的数据流量超过了网络设备(如路由器或交换机)的处理能力时,这些设备的缓冲区可能会满,导致无法继续接收新的数据包。当缓冲区满时,新到达的数据包将被丢弃。网络拥堵是导致丢包最常见的原因之一。
路由问题:数据包在网络层按照路由算法确定的路径从源头路由到目标。如果路由配置错误或是动态路由算法遇到问题,可能导致数据包被错误地发送到不存在或无法到达的目的地,从而导致丢包。
硬件故障:网络设备的物理损坏或故障也可能导致丢包。例如,路由器的接口损坏,导线问题,或其他硬件故障都可能阻止数据包的正确传输。
链路质量差:无线网络或长距离传输中的信号衰减、干扰、噪声等问题也是导致丢包的常见原因。在这些情况下,数据包在传输过程中可能会变得不完整或损坏,接收端设备无法正确解析这些数据包,导致丢包。
包过大:如果数据包的大小超过了网络中间设备的最大传输单元(MTU),而且数据包没有设定允许分片或者路径上的某个设备不支持分片,那么这个数据包会被丢弃。
安全策略:为了安全考虑,一些网络设备可能配置了包过滤或防火墙规则,用于阻止特定类型的流量。符合这些规则的数据包将被丢弃。

 6.TCP是同步还是异步?谈谈你对同步异步的理解?
答:
​​​​​​​​​​​​​​TCP(传输控制协议)本身在底层是同步的,因为它需要建立连接,在发送方和接收方之间传输数据,并确认数据包已接收。TCP确保数据准确无误地从一个网络端点传递到另一个网络端点。

对于**同步**(Synchronous)来说,它通常指的是操作的发起者必须等待这个操作完成才能继续进行后续的操作。在同步操作中,任务按照它们发生的顺序依次执行,每个任务的完成是下一个任务开始的前提。例如,当你进行网络请求,在得到响应之前,程序将不会执行后续代码,这是同步操作的特点。

而**异步**(Asynchronous)操作则不同,异步意味着操作的发起者不必等待操作完成就可以继续执行后续的操作。在异步执行中,可以启动一个任务,并且在它完成之前就开始执行其他任务。当该任务完成时,程序会在适当的时间处理完成的操作。这种方式允许程序在等待长时间操作(如文件读写、网络请求等)的同时,继续执行其他代码,从而优化程序的整体性能。

系统中的同步和异步操作常常结合起来使用,以实现更高效、可靠的数据处理和通信。应用程序根据不同的需要,可能采用同步或异步模式来处理任务,或者结合使用二者的特点。例如,在一个TCP服务中,尽管TCP协议是同步的,但应用层可以采用异步编程模型来管理多个TCP连接,以提高系统的并发能力和性能。

7.什么是TCP的沾包现象,如何解决?
答:
​​​​​​​​​​​​​​TCP的粘包现象(sticky packets phenomenon)是指在使用TCP进行通信时,由于TCP是一个面向字节流的协议,它会将应用程序的数据看作一串无结构的字节流,数据包之间的界限并不是固定的。因此,当发送方连续快速发送多个较小的消息时,TCP可能将它们合并为一个较大的数据包来提高效率,同样,接收方也可能一次性读取并合并多个数据包。这就可能导致接收端不能确定每个数据包的边界,即发生粘包。

解决粘包问题一般有以下几种方法:

1. **消息定界**:为每个消息设置边界,如使用特殊的分隔符或序列来标识消息的开始和结束。接收方在接收数据时,根据这些定界符来分割原始字节流,从而恢复出独立的消息。

2. **固定长度**:每个消息都采用固定的长度,这样接收方就可以在接收到对应长度的数据后进行处理。

3. **消息长度字段**:在消息的开始处添加一个长度字段,这个长度字段指示了整个消息的长度,接收方首先读取长度字段,然后读取相应长度的数据作为一个消息。

4. **组合方法**:在消息开始处既指定消息的长度又包含一些结束符,确保接收方能准确无误地分割消息。

5. **应用层协议设计**:设计实用的应用层协议来处理消息的封包和解包。例如,HTTP、FTP等应用层协议都有自己的处理方式来防止粘包问题。

6. **利用现成的框架和库**:许多网络编程库(如Netty、asyncio等)提供了解决粘包问题的机制,无需手动处理粘包,只需要按库的规定编写逻辑即可。

在应用层实现这些策略可以避免TCP粘包现象,确保数据能够正确、完整地在客户端和服务器之间传输。

8.组播和广播的区别?
答:
​​​​​​​​​​​​​​组播(Multicast)和广播(Broadcast)是网络通信中用于发送消息给多个目的地的两种机制:

1. **广播**:
   - 广播是一对所有的通信方式,其中一台主机发送的消息将传送给同一局域网(LAN)上的所有其他主机。
   - 广播消息通常用在没有特定目标的情况下,例如,当新设备加入网络并寻址DHCP服务器以获取IP地址时,它会发送广播消息。
   - 广播地址对于所有设备都是公开和可识别的,就像把信件扔进一个房间让每个人都能拿去阅读。
   - 在广播中,网络中的每台设备即使对消息不感兴趣也会收到消息,这可能导致无用的网络流量和设备处理。

2. **组播**:
   - 组播允许将消息同时发送给一组指定的目的地(即组播组中的主机)。只有加入该组的主机才会接收组播消息。
   - 组播地址仅适用于那些已经表明愿意接收该特定组播组消息的设备。
   - 组播有效地支持了多点传输,仅将消息传递给感兴趣的监听者,这样就减少了网络的拥堵和无效流量。
   - 组播广泛用于流媒体、实时应用和网络服务中,例如视频会议和在线游戏,典型的例子有IGMP(Internet Group Management Protocol)和多播路由。

总结来说,广播是将消息发送给所有人,而组播是将消息仅发送给一组感兴趣的监听者。组播比广播更有效率,因为它减少了不必要的数据传输。

9.阻塞IO和非阻塞IO的区别?
答:
​​​​​​​​​​​​​​阻塞IO(Blocking IO)和非阻塞IO(Non-blocking IO)主要区别体现在程序如何等待IO操作完成:

1. **阻塞IO**:
   - 在阻塞IO模型中,当一个IO操作(如读取或写入)开始时,如果数据没有准备好,程序会被挂起(即阻塞),直到数据准备好并且IO操作完成。
   - 在数据准备阶段,程序做不了其他事情,它会等待在那里直到整个IO操作完全完成后才会继续执行。
   - 阻塞IO简化了编程模型,因为在发起IO请求之后,你只需要等待结果,不必担心在这个过程中如何处理其他任务。

2. **非阻塞IO**:
   - 非阻塞IO模型中,IO操作不会导致程序挂起等待。如果数据没有准备好,IO调用会立即返回一个指示数据尚未准备好的状态。
   - 程序可以继续执行后续代码,它可以定期地检查IO操作是否可以进行,或者可以进行其他工作,这样很好地利用了等待时间。
   - 非阻塞IO通常与事件驱动或轮询机制一起使用,程序可以在多个IO操作之间有效地切换,提高了程序的整体效率和响应性。

非阻塞IO更适用于处理高并发情况,尤其在需要同时管理多个连接或者进行多个并行IO操作时。但这种模型编程难度较高,因为要管理和调度多个IO操作,而且可能需要额外的机制(如IO复用或事件通知)来检测IO状态。

10.并发和并行的区别?
答:
并发(Concurrency)和并行(Parallelism)是计算机科学中两个关键概念,常常用来描述处理多个任务的能力,它们之间有着本质的区别:

1. **并发**:
   - 并发是指一个处理单元(如CPU)在同一时间段内处理多个任务的能力。它更多地关注多任务的交错执行,即任务在单个核心上通过上下文切换实现多任务的“同时”处理。
   - 并发不是真正的同时执行多个操作,而是从宏观角度看,使得每个任务都有机会按顺序执行,从而给用户一种多个任务同时处理的错觉。
   - 并发更多地应用于任务处理的组织和调度上,常用于单核或多核处理器上的任务管理,以提高资源的使用效率和系统的响应速度。

2. **并行**:
   - 并行是指多个处理单元(如多个CPU核心)同时执行多个任务或操作。在并行计算中,每个处理单元都在同一时刻独立地执行不同的任务,实现了真正的同时处理。
   - 并行计算侧重于通过增加计算资源来增加计算速度,适用于有大量计算需要分割执行的场景,如大规模科学计算、图像处理等。
   - 在多核和多处理器系统中,并行计算可以显著提高计算效率和处理速度,特别是对于可以明确分解和独立执行的任务。

简而言之,**并发是任务交错执行**(在单个核心上“同时”处理多个任务),**并行是多个核心同时执行任务**。并发的目的在于充分利用有限的计算资源以提高效率,而并行的目标是通过增加计算资源来减少计算时间。

思维导图: 

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

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

相关文章

骑车不戴头盔监测摄像机

骑行是一种健康的出行方式&#xff0c;但是在骑行途中不戴头盔存在安全隐患&#xff0c;容易造成头部受伤。为了规范骑行行为&#xff0c;保障骑行安全&#xff0c;可以考虑使用骑车不戴头盔监测摄像机进行监测和识别。这种摄像机可以通过智能识别技术&#xff0c;实时监测骑自…

装机数台,依旧还会心念i5-12600KF的性能和性价比优势:

近几个月的时间中&#xff0c; 装机差不多4台电脑&#xff0c;由于工作需要&#xff0c;计划年中再增添一台。 目前市场上英特尔CPU促销非常火爆&#xff0c;第12代、第13代以及第14代的产品在年中有适当的优惠。 年中也是装机的旺季&#xff0c;各种相关配件也相对便宜一些。…

PS系统教学02

多个图片同时进行打开 在素材库里面选中两张图片&#xff0c;直接拖进PS软件中&#xff0c;此时会显示其中一张。当按下回车键会显示另一张。 当图层过多&#xff0c;需要进行选择&#xff0c;其中某一张图片&#xff0c;按住Ctrl键&#xff0c;进行选择点击&#xff0c;可以移…

制造企业如何通过PLM系统实现BOM管理的飞跃

摘要 在当今快速变化的制造行业中&#xff0c;产品生命周期管理&#xff08;PLM&#xff09;系统的应用已成为企业提升效率、降低成本和增强竞争力的关键。本文将探讨PLM系统如何通过其先进的BOM&#xff08;物料清单&#xff09;管理功能&#xff0c;帮助制造企业在整个产品生…

idea+tomcat+mysql 从零开始部署Javaweb项目(保姆级别)

文章目录 新建一个项目添加web支持配置tomcat优化tomcat的部署运行tomcatidea数据库连接java连接数据库 新建一个项目 new project&#xff1b;Java&#xff1b;选择jdk的版本&#xff1b;next&#xff1b;next&#xff1b;填写项目名字&#xff0c;选择保存的路径&#xff1b;…

【Linux 网络编程】协议的分层知识!

文章目录 1. 计算机网络背景2. 认识 "协议"3. 协议分层 1. 计算机网络背景 网络互联: 多台计算机连接在一起, 完成数据共享; &#x1f34e;局域网&#xff08;LAN----Local Area Network&#xff09;: 计算机数量更多了, 通过交换机和路由器连接。 &#x1f34e; 广…

基于 Arm 虚拟硬件的 TinyMaix 超轻量级神经网络推理框架的项目实践

本实验过程中所显示的优惠价格及费用报销等相关信息仅在【Arm AI 开发体验创造营】体验活动过程中有效&#xff0c;逾期无效&#xff0c;请根据实时价格自行购买和体验。同时&#xff0c;感谢本次体验活动 Arm 导师 Liliya 对于本实验手册的共创与指导。 详见活动地址&#xff…

Vue使用axios实现调用后端接口

准备后端接口 首先&#xff0c;我已经写好一个后端接口用来返回我的用户数据&#xff0c;并用Postman测试成功如下&#xff1a; 以我的接口为例&#xff0c;接口地址为&#xff1a;http://localhost:8080/user/selectAll 返回Json为&#xff1a; {"code": "2…

docker制作高版本jdk17镜像踩坑

1、创建目录并下载jdk上传到服务器中 从jdk官网下载jdk17镜像&#xff0c;提示&#xff1a;下载到本地用xftp上传到服务器&#xff08;速度会快点&#xff09; jdk官网&#xff1a;https://www.oracle.com/java/technologies/downloads/#graalvmjava21 创建目录&#xff0c;将…

Ubuntu系统编译内核——deb安装 / install安装

摘要 本文简要记录两种编译内核的方法&#xff1a; 打包成deb模块安装&#xff08;推荐&#xff09;&#xff1b;直接make install安装&#xff1b; 更推荐使用——打包成deb模块安装&#xff0c;因为可以方便的拷贝下次其他机器使用。 1. 编译环境准备 系统&#xff1a;lin…

强化学习——学习笔记3

一、强化学习都有哪些分类&#xff1f; 1、基于模型与不基于模型 根据是否具有环境模型&#xff0c;强化学习算法分为两种&#xff1a;基于模型与不基于模型 基于模型的强化学习(Model-based RL)&#xff1a;可以简单的使用动态规划求解&#xff0c;任务可定义为预测和控制&am…

cesium 实现自定义弹窗并跟随场景移动

cesium 添加点位自定义弹窗跟随场景移动 完整代码演示可直接copy使用 1 效果图&#xff1a; 2 深入理解 就是原始点位的数据 id>property 点位真实渲染到球体上的笛卡尔坐标系 id>_polyline 的路径下 可以通过 3 代码示例 <!DOCTYPE html> <html lang"…

【数据分享】2017-2023年全球范围10米精度土地覆盖数据

土地覆盖数据是我们在各项研究中都非常常用的数据&#xff0c;土地覆盖数据的来源也有很多。之前我们分享过欧空局发布的2020年和2021年的10米分辨率的土地覆盖数据,也分享过我国首套1米分辨率的土地覆盖数据&#xff08;均可查看之前的文章获悉详情&#xff09;&#xff01; …

管道液位传感器可以检测哪些液体?

管道液位传感器是一种专门用于检测流动性比较好的液体的传感器装置。它采用光学感应原理&#xff0c;不涉及任何机械运动&#xff0c;具有长寿命、安装方便和微功耗的特点。相比传统机械式液位传感器&#xff0c;光电管道传感器有效解决了低精度和卡死失效等问题&#xff0c;同…

Django 解决 CSRF 问题

在 Django 出现 CSRF 问题 要解决这个问题&#xff0c;就得在 html 里这么修改 <!DOCTYPE html> <html><head></head><body><form action"/login/" method"post">{% csrf_token %}</form></body> </…

短视频脚本创作的五个方法 沈阳短视频剪辑培训

说起脚本&#xff0c;我们大概都听过影视剧脚本、剧本&#xff0c;偶尔可能在某些综艺节目里听过台本。其中剧本是影视剧拍摄的大纲&#xff0c;用来指导影视剧剧情的走向和发展&#xff0c;而台本则是综艺节目流程走向的指导大纲。 那么&#xff0c;短视频脚本是什么&#xf…

探析GPT-4o:技术之巅的跃进

如何评价GPT-4o? 简介&#xff1a;最近&#xff0c;GPT-4o横空出世。对GPT-4o这一人工智能技术进行评价&#xff0c;包括版本间的对比分析、GPT-4o的技术能力以及个人感受等。 随着人工智能领域的不断发展&#xff0c;GPT系列模型一直处于行业的前沿。最近&#xff0c;GPT-4…

前端实习记录——git篇(一些问题与相关命令)

1、版本控制 &#xff08;1&#xff09;版本回滚 git log // 查看版本git reset --mixed HEAD^ // 回滚到修改状态&#xff0c;文件内容没有变化git reset --soft HEAD^ // 回滚暂存区&#xff0c;^的个数代表几个版本git reset --hard HEAD^ // 回滚到修改状态&#xff…

生态农业:引领未来农业新篇章

生态农业&#xff0c;正以其独特的魅力和创新理念&#xff0c;引领着未来农业发展的新篇章。在这个充满变革的时代&#xff0c;我们需要更加关注农业的可持续发展&#xff0c;而生态农业正是实现这一目标的重要途径。 生态农业产业的王总说&#xff1a;生态农业强调生态平衡和可…

【文档+源码+调试讲解】古典舞在线交流平台的设计与实现

摘 要 随着互联网技术的发展&#xff0c;各类网站应运而生&#xff0c;网站具有新颖、展现全面的特点。因此&#xff0c;为了满足用户古典舞在线交流的需求&#xff0c;特开发了本古典舞在线交流平台。 本古典舞在线交流平台应用Java技术&#xff0c;MYSQL数据库存储数据&…