Linux下socket编程sendto命令详解 (linux socket sendto)
Linux下Socket编程sendto命令详解
在Linux下进行Socket编程需要掌握sendto函数的使用方法。sendto函数可以从一个已经打开的socket发送数据到指定的目的地址,可以是本地地址,也可以是远程地址。本文将详细介绍如何使用sendto函数。
sendto函数的基本用法
sendto函数的原型如下:
“`
ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,
const struct sockaddr *dest_addr, socklen_t addrlen);
“`
其中各个参数的意义如下:
– sockfd:已经打开的socket的描述符。
– buf:需要发送的数据。
– len:需要发送的数据的长度。
– flags:发送数据的标志。常见的标志有MSG_DONTWT和MSG_NOSIGNAL等。
– dest_addr:目的地址。可以是本地地址,也可以是远程地址。
– addrlen:目的地址的长度。
sendto函数的返回值为成功发送的字节数,发送失败返回-1,错误信息存储在errno中。
使用sendto函数发送数据
下面是一个简单的例子,演示如何使用sendto函数发送数据。该程序创建一个UDP socket,然后向指定的IP地址和端口发送一条信息。
“`c
#include
#include
#include
#include
#include
#include
#define IP_ADDR “localhost”
#define PORT 8080
#define BUF_SIZE 4096
int mn() {
int sockfd, ret;
char buf[BUF_SIZE];
struct sockaddr_in addr;
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd == -1) {
printf(“Create socket fled: %s\n”, strerror(errno));
exit(EXIT_FLURE);
}
printf(“Create socket success.\n”);
memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
inet_pton(AF_INET, IP_ADDR, &addr.sin_addr.s_addr);
addr.sin_port = htons(PORT);
strcpy(buf, “Hello, Socket”);
ret = sendto(sockfd, buf, strlen(buf), 0, (struct sockaddr *)&addr, sizeof(addr));
if (ret == -1) {
printf(“Send message fled: %s\n”, strerror(errno));
close(sockfd);
exit(EXIT_FLURE);
}
printf(“Send message success.\n”);
close(sockfd);
return 0;
}
“`
该程序的流程如下:
1. 调用socket函数创建UDP socket;
2. 设置目的地址和端口;
3. 发送一条信息;
4. 关闭socket。
在sendto函数中,之一个参数为sockfd,表示与目标主机连接的socket描述符。第二个参数为要发送的数据,即buf。第三个参数为要发送数据的长度len。第四个参数flags设置为0,表示默认标志。第五个参数为目标地址,为一个指向目标socket地址结构的指针。第六个参数addrlen表示目标地址结构的长度,可以使用sizeof(struct sockaddr_in)。
在sendto函数执行成功后,sendto函数返回已发送的字节数。如果出现错误,sendto返回-1,错误信息存储在errno中。
sendto函数的选项
在使用sendto函数时,可以设置一些选项。常见的选项如下:
– MSG_DONTWT:非阻塞模式;
– MSG_NOSIGNAL:在对方关闭后不发送SIGPIPE信号;
– MSG_MORE:发送更多数据;
– MSG_CONFIRM:要求确认无误。
以MSG_DONTWT为例,下面是一个例子:
“`c
#include
#include
#include
#include
#include
#include
#include
#define IP_ADDR “localhost”
#define PORT 8080
#define BUF_SIZE 4096
int mn() {
int sockfd, ret;
char buf[BUF_SIZE];
struct sockaddr_in addr;
int flags;
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd == -1) {
printf(“Create socket fled: %s\n”, strerror(errno));
exit(EXIT_FLURE);
}
printf(“Create socket success.\n”);
flags = fcntl(sockfd, F_GETFL, 0);
fcntl(sockfd, F_SETFL, flags|O_NONBLOCK);
memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
inet_pton(AF_INET, IP_ADDR, &addr.sin_addr.s_addr);
addr.sin_port = htons(PORT);
strcpy(buf, “Hello, Socket”);
ret = sendto(sockfd, buf, strlen(buf), MSG_DONTWT, (struct sockaddr *)&addr, sizeof(addr));
if (ret == -1) {
printf(“Send message fled: %s\n”, strerror(errno));
close(sockfd);
exit(EXIT_FLURE);
}
printf(“Send message success.\n”);
close(sockfd);
return 0;
}
“`
在这个例子中,我们通过fcntl函数设置了socket为非阻塞模式,然后在sendto函数中加入了MSG_DONTWT选项。
在sendto函数中使用MSG_DONTWT选项时,如果socket处于阻塞模式下,则sendto函数会返回EAGN错误,否则立即返回。因此,我们需要将socket设置为非阻塞模式。在程序中,我们首先使用fcntl函数获取socket文件描述符的标志,然后通过F_SETFL选项更新socket的标志,将其设置为非阻塞模式。