udp通訊中的connect()和bind()函式
阿新 • • 發佈:2019-02-02
udp是一個基於無連線的通訊協議,通訊基本模型如下:
可以看出,不論是在客戶端還是伺服器,connect()似乎用不上,bind()在客戶端也用不上,但是事實並非如此。
1. udp客戶端使用connect()函式
udp客戶端建立了socket後可以直接呼叫sendto()函式向伺服器傳送資料,但是需要在sendto()函式的引數中指定目的地址/埠,但是可以呼叫connect()函式先指明目的地址/埠,然後就可以使用send()函式向目的地址傳送資料了,因為此時套接字已經包含目的地址/埠,也就是send()函式已經知道包含目的地址/埠。
//用sendto()函式傳送資料的udp客戶端程式
int main(int argc, char *argv[])
{
int sd;
struct sockaddr_in svr_addr;
int ret;
socklen_t addrlen = sizeof(struct sockaddr_in);
char buf[BUFSZ] = {};
if ((sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
{
perror("socket");
exit(EXIT_FAILURE);
}
//sendto()函式需要指定目的埠/地址
svr_addr.sin_family = AF_INET;
svr_addr.sin_port = htons(PORT);
svr_addr.sin_addr.s_addr = inet_addr("192.168.1.166");
while (1)
{
memset(buf, 0, BUFSZ);
printf("ple input: ");
fgets(buf, BUFSZ, stdin);
sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
ret = recvfrom(sd, buf, BUFSZ, 0 , (struct sockaddr* )&svr_addr, &addrlen);
printf("client: IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(svr_addr.sin_addr), ntohs(svr_addr.sin_port), buf);
}
close(sd);
return 0;
}
send()函式傳送資料的udp客戶端程式:
//用send()函式傳送資料的udp客戶端程式
int main(int argc, char *argv[])
{
int sd;
struct sockaddr_in svr_addr;
int ret;
socklen_t addrlen = sizeof(struct sockaddr_in);
char buf[BUFSZ] = {};
if ((sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
{
perror("socket");
exit(EXIT_FAILURE);
}
//先呼叫connect()函式,為套接字指定目的地址/埠
svr_addr.sin_family = AF_INET;
svr_addr.sin_port = htons(PORT);
svr_addr.sin_addr.s_addr = inet_addr("192.168.1.166");
connect(sd, (struct sockaddr* )&svr_addr, addrlen);
while (1)
{
memset(buf, 0, BUFSZ);
printf("ple input: ");
fgets(buf, BUFSZ, stdin);
//sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
send(sd, buf, BUFSZ, 0);
ret = recvfrom(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, &addrlen);
printf("client: IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(svr_addr.sin_addr), ntohs(svr_addr.sin_port), buf);
}
close(sd);
return 0;
}
2. udp客戶端程式使用bind()函式
udp伺服器呼叫了bind()函式為伺服器套接字繫結本地地址/埠,這樣使得客戶端的能知道它發資料的目的地址/埠,伺服器如果單單接收客戶端的資料,或者先接收客戶端的資料(此時通過recvfrom()函式獲取到了客戶端的地址資訊/埠)再發送資料,客戶端的套接字可以不繫結自身的地址/埠,因為udp在建立套接字後直接使用sendto(),隱含操作是,在傳送資料之前作業系統會為該套接字隨機分配一個合適的udp埠,將該套接字和本地地址資訊繫結。
但是,如果伺服器程式就緒後一上來就要傳送資料給客戶端,那麼伺服器就需要知道客戶端的地址資訊和埠,那麼就不能讓客戶端的地址資訊和埠號由客戶端所在作業系統分配,而是要在客戶端程式指定了。怎麼指定,那就是用bind()函式:
//為客戶端繫結埠和地址資訊
int main(int argc, char *argv[])
{
int sd;
struct sockaddr_in svr_addr, cli_addr;
int ret;
socklen_t addrlen = sizeof(struct sockaddr_in);
char buf[BUFSZ] = {};
if ((sd = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
{
perror("socket");
exit(EXIT_FAILURE);
}
//繫結地址資訊
cli_addr.sin_family = AF_INET;
cli_addr.sin_port = htons(9693);
cli_addr.sin_addr.s_addr = 0;
if ((ret = bind(sd, (struct sockaddr* )&cli_addr, addrlen)) < 0)
{
perror("bind");
exit(EXIT_FAILURE);
}
svr_addr.sin_family = AF_INET;
svr_addr.sin_port = htons(PORT);
svr_addr.sin_addr.s_addr = inet_addr("192.168.1.166");
while (1)
{
memset(buf, 0, BUFSZ);
printf("ple input: ");
fgets(buf, BUFSZ, stdin);
sendto(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, addrlen);
ret = recvfrom(sd, buf, BUFSZ, 0, (struct sockaddr* )&svr_addr, &addrlen);
printf("client: IPAddr = %s, Port = %d, buf = %s\n", inet_ntoa(svr_addr.sin_addr), ntohs(svr_addr.sin_port), buf);
}
close(sd);
return 0;
}
3. udp伺服器程式使用connect()函式
如上所述,connect()函式可以用來指明套接字的目的地址/埠號,那麼若udp伺服器可以使用connect,將導致伺服器只接受這特定一個主機的請求。