c语言单工通信,Linux下C语言实现半双工的UDP通信

------------恢复内容开始------------

Linux下C语言实现半双工的UDP通信

1、单向通信:又称为单工通信,即只能有一个方向的通信而没有反方向的交互。无线电广播或有线电广播以及电视广播就属于这种类型。

单向通信只需要一条信道,而双向交替通信或双向同时通信则都需要两条信道(每个方向各一条)。显然,双向同时通信的传输效率最高。不过应当指出,虽然电信局为打电话的用户提供了双向同时通信的信道,但有效的电话交谈一般都还是双方交替通信。当双方发生争吵时往往就是采用双向同时通信的方式。

2、半双工通信,是指数据可以沿两个方向传送.但同一时刻一个信道只允许单方向传送,因此义被称为双向交替通信,如图中(b)所示。例如,无线对讲机就是一种半双工设备,在同一时间内只允许一方讲话。

3、全双工通信,是指同时发生在两个方向上的一种数据传输方式,如图中(c)所示。电话机就是一种全双工设备,其通话双方可以同时进行对话。计算机之间的高速数据通信也是这种方式。

双向交替通信又称为半双工通信,即通信的双方都可以发送信息,但不能双方同时发送(当然也就不能同时接收)。这种通信方式是一方发送另一方接收,过一段时间后再反过来。

server.c

#include

#include

#include

#include

#include

#include

#include

#include

#define MYPORT 8887

#define ERR_EXIT(m) \

do { \

perror(m); \

exit(EXIT_FAILURE); \

} while (0)

void echo_ser(int sock)

{

char recvbuf[1024] = {0};

struct sockaddr_in peeraddr;

char close_str[1024]={"q"};

socklen_t peerlen;

int n;

while (1)

{

peerlen = sizeof(peeraddr);//获得地址长度

memset(recvbuf, 0, sizeof(recvbuf));//清除数组中的数据

//4.接收数据,recvfrom返回值为接受的字节数

n = recvfrom(sock, recvbuf, sizeof(recvbuf), 0,

(struct sockaddr *)&peeraddr, &peerlen);

if (n <= 0)

{

if (errno == EINTR)

continue;

ERR_EXIT("recvfrom error");

}

//5,发送数据,sendto返回值为发送的字节数,出错则返回-1

else if(n > 0)

{

printf("接收到的数据:%s\n",recvbuf);

if(memcmp(recvbuf,close_str,1)==0){

printf("接受到关闭信号~\n");

sendto(sock, "close ok~", 9, 0,

(struct sockaddr *)&peeraddr, peerlen);

break;

}

sendto(sock, recvbuf, n, 0,

(struct sockaddr *)&peeraddr, peerlen);

printf("回送的数据:%s\n",recvbuf);

}

}

close(sock);

}

int main(void)

{

//1.创建套接字

int sock;

if ((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)

ERR_EXIT("socket error");

//2.创建服务器地址和端口号并设置

struct sockaddr_in servaddr;

memset(&servaddr, 0, sizeof(servaddr));

servaddr.sin_family = AF_INET;//设置地址类型为IP

servaddr.sin_port = htons(MYPORT);//设置端口号

servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//自动获取地址

//3.绑定套接字

printf("监听%d端口\n",MYPORT);

if (bind(sock, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)//绑定套接字和地址变量

ERR_EXIT("bind error");

echo_ser(sock);

return 0;

}

client.c

#include

#include

#include

#include

#include

#include

#include

#include

#include

#define MYPORT 8887

char* SERVERIP = "127.0.0.1";

#define ERR_EXIT(m) \

do \

{ \

perror(m); \

exit(EXIT_FAILURE); \

} while(0)

void echo_cli(int sock)

{

//2.设置远程服务器的地址信息和端口号

struct sockaddr_in servaddr;

memset(&servaddr, 0, sizeof(servaddr));

servaddr.sin_family = AF_INET;

servaddr.sin_port = htons(MYPORT);

servaddr.sin_addr.s_addr = inet_addr(SERVERIP);

int ret;

char sendbuf[1024] = {0};

char recvbuf[1024] = {0};

//3.向服务器发送数据

while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)//读入字符串

{

printf("向服务器发送:%s\n",sendbuf);

//sendto返回值为发送字节的长度,出错返回-1

sendto(sock, sendbuf, strlen(sendbuf), 0, (struct sockaddr *)&servaddr, sizeof(servaddr));

//recvfrom返回值为接受的字节数

ret = recvfrom(sock, recvbuf, sizeof(recvbuf), 0, NULL, NULL);

if (ret == -1)

{

if (errno == EINTR)

continue;

ERR_EXIT("recvfrom");

}

printf("从服务器接收:%s\n",recvbuf);

//清除数组中的数据

memset(sendbuf, 0, sizeof(sendbuf));

memset(recvbuf, 0, sizeof(recvbuf));

}

close(sock);

}

int main(void)

{

//1.创建套接字

int sock;

if ((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)

ERR_EXIT("socket");

echo_cli(sock);

return 0;

}

运行结果:

cbc38f06458ecd5d8106fb73310f4818.png

------------恢复内容结束------------

来源:oschina

链接:https://my.oschina.net/u/4346770/blog/4288671

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值