在linux下用网络通信个人觉得是件比较好玩的事情。在用socket编程时,发现可以先不了解UDP,TCP传输层及传输层以下的事情,是件轻松的事情,当然后续会把tcp ip详解卷给补上。同时也发现了一本很不错的书籍 《unix网络编程卷一》,来学习linux C/C++网络编程。
UDP程序所用套接字
先介绍UDP中用到的socket的几个函数:
/************************************************/
#include <sys/types.h>
#include <sys/socket.h>
int socket(int domain, int type, int protocol);
/************************************************/
为了执行网络I/O,一个进程必须要做的第一件事情就是调用socket函数。
arg1 domain -- 协议族类,此处一致设置为AF_INET,为IPV4协议族。
arg2 type -- 套接字类型,SOCK_STREAM--用于TCP,字节流套接字,
SOCK_DGRAM--用于UDP, 数据报套接字。
arg3 protocol --协议类型常值,设为0,根据domain 和type可以默认设定系统协议。
retn(返回值) 套接字描述符--sockfd,成功返回非负描述符,出错为-1。
/***************************************************************************/
#include <sys/types.h>
#include <sys/socket.h>
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
/***************************************************************************/
bind函数把一个本地协议地址赋予一个套接字,对于网际协议,协议地址是32位IPv4地址与16位的TCP或UDP端口号的组合。
arg2 addr -- 指向特定于协议的地址结构指针,结构体成员sin_addr和sin_port,ip地址和端口号。
arg3 addrlen -- 该地址结构的长度。
retn(返回值) 成功则为0, 出错则为-1。
[b]调用bind函数可以指定一个端口号,一个IP地址或者两者都指定,也可以两者都不指定。
sin_addr.s_addr选择为统配地址时常值INADDR_ANY, 有内核决定IPv4地址即本机IPv4地址。
sin_port选择为0时, 有内核决定临时端口号。[/b]
/**********************************************************************/
#include <sys/types.h>
#include <sys/socket.h>
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr *src_addr, socklen_t *addrlen);
/***********************************************************************/
recvfrom函数使用数据报协议(UDP)接收的典型用途。
argv2 -- buf,读出缓冲区指针
argv3 -- len,读出缓冲区大小,
argv4 -- flags,总设置为0
argv5 -- src_addr,,数据报发送者的协议地址套接字地址结构
argv6 -- addrlen,套接字结构地址中字节数指针。
retn -- 成功为写字节数,出错为-1。
/*************************************************************************/
#include <sys/types.h>
#include <sys/socket.h>
ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);
/*************************************************************************/
sendto函数使用数据报协议(UDP)发送的典型用途。
argv2 -- buf, 写入缓冲区指针。
argv3 -- len, 写入缓冲区大小。
argv4 -- flags,总设置为0。
argv5 -- dest_addr, 数据报目的协议地址套接字地址结构
argv6 -- addrlen,套接字地址结构字节数。
retn -- 成功为写字节数,出错为-1。
如下代码为UDP服务器:
[code=c]
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#define BUFFERSIZE 1500
char recvbuf[BUFFERSIZE];
int main (int argc, char *argv[])
{
int sockfd;
uint32_t recvNum;
uint32_t addrlen;
struct sockaddr_in addr_local;
struct sockaddr addr_remort;
char sendBuffer[] = "Linux Received.";
sockfd = socket(AF_INET, SOCK_DGRAM, 0);
if (sockfd < 0)
{
printf ("Socket Descriptor Error.\r\n");
return 1;
}
bzero(&addr_local, sizeof(addr_local));
addr_local.sin_family = AF_INET;
addr_local.sin_port = htons(8080);
addr_local.sin_addr.s_addr = htonl(INADDR_ANY); //local IP addr
if (bind(sockfd, (struct sockaddr *)&addr_local, sizeof(addr_local)) < 0)
{
printf ("Socket Binding Error.\r\n");
return 1;
}
for (;;)
{
recvNum = recvfrom(sockfd, recvbuf, sizeof(recvbuf), 0, (struct sockaddr*)&addr_remort, &addrlen);
if (recvNum > 0)
{
sendto(sockfd, sendBuffer, sizeof(sendBuffer), 0, (struct sockaddr*)&addr_remort, addrlen);
printf ("Recv Data: %s \r\n", recvbuf);
}
}
return 0;
}
[/code]