sockaddr_in , sockaddr , in_addr區別Socket編程函數集(非常有用)

一、sockaddr和sockaddr_in在字節長度上都爲16個BYTE,可以進行轉換

struct   sockaddr   {  
                unsigned   short   sa_family;    //2 
                char   sa_data[14];     //14
        };  
  上面是通用的socket地址,具體到Internet   socket,用下面的結構,二者可以進行類型轉換           
  struct   sockaddr_in   {  
                short   int   sin_family;     //2
                unsigned   short   int   sin_port;     //2
                struct   in_addr   sin_addr;     ‘//4
                unsigned   char   sin_zero[8];     //8
        };  
        struct   i
n_addr就是32位IP地址。  
        struct   in_addr   {  
                union {
                        struct { u_char s_b1,s_b2,s_b3,s_b4; } S_un_b;
                        struct { u_short s_w1,s_w2; } S_un_w;
                        u_long S_addr; 
                } S_un;

                #define s_addr  S_un.S_addr
        };  

或者;

struct in_addr {
    in_addr_t s_addr;
};
結構體in_addr 用來表示一個32位的IPv4地址
 inet_addr()是將一個點分制的IP地址(如192.168.0.1)轉換爲上述結構中需要的32位二進制方式的IP地址(0xC0A80001)。//server_addr.sin_addr.s_addr=htonl(INADDR_ANY); 

通常的做法是填值的時候使用sockaddr_in結構而作爲函數(如bin, accept, connect等)的參數傳入的時候轉換成sockaddr結構就行了,畢竟都是16個字符長。

通常server的用法是:  
  int   sockfd;  
  struct   sockaddr_in   my_addr;  
//賦值時用這個結構
  sockfd   =   socket(AF_INET,   SOCK_STREAM,   0);      
  my_addr.sin_family   =   AF_INET;     
  my_addr.sin_port   =   htons(MYPORT);     
  my_addr.sin_addr.s_addr   =   inet_addr("192.168.0.1");     
  bzero(&(my_addr.sin_zero),   8);         
  bind(sockfd,   (struct   sockaddr   *)&my_addr,   sizeof(struct   sockaddr));//用(struct   sockaddr   *)轉換即滿足要求

//int accept(int s,struct sockaddr * addr,int * addrlen);//這三個函數的第二個參數結構都爲struct sockaddr,所以一般做法都如上所示。

//int bind(int sockfd,struct sockaddr * my_addr,int addrlen);

//int connect (int sockfd,struct sockaddr * serv_addr,int addrlen);

----------------------------------------

通常client的用法是:  
  int   sockfd;  
  struct   sockaddr_in   my_addr;  
//賦值時用這個結構
  sockfd   =   socket(AF_INET,   SOCK_STREAM,   0);      
  my_addr.sin_family   =   AF_INET;     
  my_addr.sin_port   =   htons(MYPORT);     
  my_addr.sin_addr.s_addr   =   inet_addr("192.168.0.1");     
  bzero(&(my_addr.sin_zero),   8);         
  connect(sockfd,   (struct   sockaddr   *)&my_addr,   sizeof(struct   sockaddr));//用(struct   sockaddr   *)轉換即滿足要求




二 、說個小知識,爲什麼一般情況下都用serv.sin_addr.s_addr=htonl(INADDR_ANY)

  比如你的機器有三個ip   
  192.168.1.1   
  202.202.202.202   
  61.1.2.3   
    
  如果你serv.sin_addr.s_addr=inet_addr("192.168.1.1");   
    
  然後監聽100端口   
    
  這時其他機器只有connect   192.168.1.1:100端口才能成功。   
  connect   202.202.202.202:100和connect   61.1.2.3:100都會失敗。   
    
  如果serv.sin_addr.s_addr=htonl(INADDR_ANY);   的話,無論連接哪個ip都可以連上的,這就是爲什麼這樣選擇的理由


三、另外一個關於這連個結構的一個解釋,覺得也還不錯。

struct sockaddr 是一個通用地址結構,這是爲了統一地址結構的表示方法,統一接口函數,使不同的地址結構可以被bind() , connect() 等函數調用;struct sockaddr_in中的in 表示internet,就是網絡地址,這只是我們比較常用的地址結構,屬於AF_INET地址族,他非常的常用,以至於我們都開始討論它與 struct sockaddr通用地址結構的區別。另外還有struct sockaddr_un 地址結構,剩下的地址結構我就不知道了。我們可以認爲 struct sockaddr_in 和 struct sockaddr_un 是 struct sockaddr 的子集。
 
struct sockaddr {
unsigned short sa_family; /* address family, AF_xxx */
char sa_data[14]; /* 14 bytes of protocol address */
};

裏的sa_data裏的IP地址取出來,折騰半天還是失敗了。在CSDN上發現2003年時曾有人跟我一樣傻 哈哈

struct sockaddr {
unsigned short sa_family; /* address family, AF_xxx */
char sa_data[14]; /* 14 bytes of protocol address */
};
sa_family是地址家族,一般都是“AF_xxx”的形式。好像通常大多用的是都是AF_INET。
sa_data是14字節協議地址。
此數據結構用做bind、connect、recvfrom、sendto等函數的參數,指明地址信息。
但一般編程中並不直接針對此數據結構操作,而是使用另一個與sockaddr等價的數據結構
sockaddr_in(在netinet/in.h中定義):
struct sockaddr_in {
short int sin_family; /* Address family */
unsigned short int sin_port; /* Port number */
struct in_addr sin_addr; /* Internet address */
unsigned char sin_zero[8]; /* Same size as struct sockaddr */
};
struct in_addr {
unsigned long s_addr;
};
typedef struct in_addr {
union {
struct{
unsigned char s_b1,
s_b2,
s_b3,
s_b4;
} S_un_b;
struct {
unsigned short s_w1,
s_w2;
} S_un_w;
unsigned long S_addr;
} S_un;
} IN_ADDR;
sin_family指代協議族,在socket編程中只能是AF_INET
sin_port存儲端口號(使用網絡字節順序)
sin_addr存儲IP地址,使用in_addr這個數據結構
sin_zero是爲了讓sockaddr與sockaddr_in兩個數據結構保持大小相同而保留的空字節。
s_addr按照網絡字節順序存儲IP地址
sockaddr_in和sockaddr是並列的結構,指向sockaddr_in的結構體的指針也可以指向
sockadd的結構體,並代替它。也就是說,你可以使用sockaddr_in建立你所需要的信息,
在最後用進行類型轉換就可以了bzero((char*)&mysock,sizeof(mysock));//初始化
mysock結構體名
mysock.sa_family=AF_INET;
mysock.sin_addr.s_addr=inet_addr("192.168.0.1");
……
等到要做轉換的時候用:
(struct sockaddr*)mysock

四、備註常用的socket函數集:

accept(接受socket連線)
相關函數 socketbindlistenconnect
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int accept(int s,struct sockaddr * addr,int * addrlen);
函數說明 accept()用來接受參數ssocket連線。參數ssocket必需先經bind()listen()函數處理過,當有連線進來時accept()會返回一個新的socket處理代碼,往後的數據傳送與讀取就是經由新的socket處理,而原來參數ssocket能繼續使用accept()來接受新的連線要求。連線成功時,參數addr所指的結構會被系統填入遠程主機的地址數據,參數addrlenscokaddr的結構長度。關於結構sockaddr的定義請參考bind()
返回值 成功則返回新的socket處理代碼,失敗返回-1,錯誤原因存於errno中。
錯誤代碼
EBADF參數非合法socket處理代碼。
EFAULT參數addr指針指向無法存取的內存空間。
ENOTSOCK參數s爲一文件描述詞,非socket
EOPNOTSUPP指定的socket並非SOCK_STREAM
EPERM防火牆拒絕此連線。
ENOBUFS 系統的緩衝內存不足。
ENOMEM核心內存不足。
範例 參考listen()
bind(對socket定位)
相關函數 socketacceptconnectlisten
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int bind(int sockfd,struct sockaddr * my_addr,int addrlen);
函數說明
bind()用來設置給參數sockfdsocket一個名稱。此名稱由參數my_addr指向一sockaddr結構,對於不同的socket domain定義了一個通用的數據結構。
struct sockaddr {
unsigned short int sa_family;
char sa_data[14];
};
sa_family 爲調用socket()時的domain參數,即AF_xxxx值。
sa_data 最多使用14個字符長度。
sockaddr結構會因使用不同的socket domain而有不同結構定義,例如使用AF_INET domain,其socketaddr結構定義便爲
struct socketaddr_in {
unsigned short int sin_family;
uint16_t sin_port;
struct in_addr sin_addr;
unsigned char sin_zero[8];
};
struct in_addr {
uint32_t s_addr;
};
sin_family 即爲sa_family
sin_port 爲使用的port編號
sin_addr.s_addr IP 地址
sin_zero 未使用。
參數說明 addrlensockaddr的結構長度。
返回值 成功則返回0,失敗返回-1,錯誤原因存於errno中。
錯誤代碼
EBADF參數sockfd 非合法socket處理代碼。
EACCESS權限不足
ENOTSOCK參數sockfd爲一文件描述詞,非socket
範例 參考listen()
connect(建立socket連線)
相關函數 socketbindlisten
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int connect (int sockfd,struct sockaddr * serv_addr,int addrlen);
函數說明 connect()用來將參數sockfd socket 連至參數serv_addr 指定的網絡地址。結構sockaddr請參考bind()。參數addrlensockaddr的結構長度。
返回值 成功則返回0,失敗返回-1,錯誤原因存於errno中。
錯誤代碼
EBADF參數sockfd 非合法socket處理代碼
EFAULT參數serv_addr指針指向無法存取的內存空間
ENOTSOCK參數sockfd爲一文件描述詞,非socket
EISCONN參數sockfdsocket已是連線狀態
ECONNREFUSED連線要求被server端拒絕。
ETIMEDOUT企圖連線的操作超過限定時間仍未有響應。
ENETUNREACH無法傳送數據包至指定的主機。
EAFNOSUPPORT sockaddr結構的sa_family不正確。
EALREADY socket爲不可阻斷且先前的連線操作還未完成。
範例
/* 利用socketTCP client
此程序會連線TCP server,並將鍵盤輸入的字符串傳送給server
* TCP server範例請參考listen()。
*/
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
 
#define PORT 1234
#define SERVER_IP “127.0.0.1”
main()
{
int s;
struct sockaddr_in addr;
char buffer[256];
if ((s = socket(AF_INET,SOCK_STREAM,0))<0)
{
perror(“socket”);
exit(1);
}
 
/* 填寫sockaddr_in結構*/
bzero(&addr,sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(PORT);
addr.sin_addr.s_addr = inet_addr(SERVER_IP); /* 嘗試連線*/
if (connect(s,&addr,sizeof(addr))<0)
{
perror(“connect”);
exit(1);
}
 
/* 接收由server端傳來的信息*/
recv(s,buffer,sizeof(buffer),0);
printf(“%s\n”,buffer);
while(1)
{
bzero(buffer, sizeof(buffer));
/* 從標準輸入設備取得字符串*/
read(STDIN_FILENO, buffer, sizeof(buffer));
/* 將字符串傳給server*/
if (send(s, buffer, sizeof(buffer), 0)<0)
{
perror(“send”);
exit(1);
}
}
}
執行
$ ./connect
Welcome to server!
hi I am client! /*鍵盤輸入*/
/*<Ctrl+C>中斷程序*/
endprotoent(結束網絡協議數據的讀取)
相關函數 getprotoentgetprotobynamegetprotobynumbersetprotoent
表頭文件 #include<netdb.h>
定義函數 void endprotoent(void);
函數說明 endprotoent()用來關閉由getprotoent()打開的文件。
返回值 無返回值
範例 參考getprotoent()
endservent(結束網絡服務數據的讀取)
相關函數 getserventgetservbynamegetservbyportsetservent
表頭文件 #include<netdb.h>
定義函數 void endservent(void);
函數說明 endservent()用來關閉由getservent()所打開的文件。
返回值 無返回值
範例 參考getservent()
getsockopt(取得socket狀態)
相關函數 setsockopt
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int getsockopt(int s,int level,int optname,void* optval,socklen_t* optlen);
函數說明 getsockopt()會將參數s所指定的socket狀態返回。參數optname代表欲取得何種選項狀態,而參數optval則指向欲保存結果的內存地址,參數optlen則爲該空間的大小。參數leveloptname請參考setsockopt()
返回值 成功則返回0,若有錯誤則返回-1,錯誤原因存於errno
錯誤代碼
EBADF參數並非合法的socket處理代碼
ENOTSOCK參數s爲一文件描述詞,非socket
ENOPROTOOPT參數optname指定的選項不正確
EFAULT參數optval指針指向無法存取的內存空間
範例
#include<sys/types.h>
#include<sys/socket.h>
main()
{
int s,optval,optlen = sizeof(int);
if ((s = socket(AF_INET,SOCK_STREAM,0))<0)
perror(“socket”);
getsockopt(s,SOL_SOCKET,SO_TYPE,&optval,&optlen);
printf(“optval = %d\n”,optval);
close(s);
}
執行 optval = 1 /*SOCK_STREAM的定義正是此值*/
htonl(將32位主機字符順序轉換成網絡字符順序)
相關函數 htonsntohlntohs
表頭文件 #include<netinet/in.h>
定義函數 unsigned long int htonl(unsigned long int hostlong);
函數說明 Htonl()用來將參數指定的32hostlong 轉換成網絡字符順序。
返回值 返回對應的網絡字符順序。
範例 參考getservbyport()connect()
htons(將16位主機字符順序轉換成網絡字符順序)
相關函數 htonlntohlntohs
表頭文件 #include<netinet/in.h>
定義函數 unsigned short int htons(unsigned short int hostshort);
函數說明 htons()用來將參數指定的16hostshort轉換成網絡字符順序。
返回值 返回對應的網絡字符順序。
範例 參考connect()
inet_addr(將網絡地址轉成二進制的數字)
相關函數 inet_atoninet_ntoa
表頭文件
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
定義函數 unsigned long int inet_addr(const char *cp);
函數說明 inet_addr()用來將參數cp所指的網絡地址字符串轉換成網絡所使用的二進制數字。網絡地址字符串是以數字和點組成的字符串,例如:“163.13.132.68”
返回值 成功則返回對應的網絡二進制的數字,失敗返回-1
inet_aton(將網絡地址轉成網絡二進制的數字)
相關函數 inet_addrinet_ntoa
表頭文件
#include<sys/scoket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
定義函數 int inet_aton(const char * cp,struct in_addr *inp);
函數說明
inet_aton()用來將參數cp所指的網絡地址字符串轉換成網絡使用的二進制的數字,然後存於參數inp所指的in_addr結構中。
結構in_addr定義如下
struct in_addr {
unsigned long int s_addr;
};
返回值 成功則返回非0值,失敗則返回0
inet_ntoa(將網絡二進制的數字轉換成網絡地址)
相關函數 inet_addrinet_aton
表頭文件
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
定義函數 char * inet_ntoa(struct in_addr in);
函數說明 inet_ntoa()用來將參數in所指的網絡二進制的數字轉換成網絡地址,然後將指向此網絡地址字符串的指針返回。
返回值 成功則返回字符串指針,失敗則返回NULL
listen(等待連接)
相關函數 socketbindacceptconnect
表頭文件 #include<sys/socket.h>
定義函數 int listen(int s,int backlog);
函數說明 listen()用來等待參數socket連線。參數backlog指定同時能處理的最大連接要求,如果連接數目達此上限則client端將收到ECONNREFUSED的錯誤。Listen()並未開始接收連線,只是設置socketlisten模式,真正接收client端連線的是accept()。通常listen()會在socket()bind()之後調用,接着才調用accept()
返回值 成功則返回0,失敗返回-1,錯誤原因存於errno
附加說明 listen()只適用SOCK_STREAMSOCK_SEQPACKETsocket類型。如果socketAF_INET則參數backlog 最大值可設至128
錯誤代碼
EBADF參數sockfd非合法socket處理代碼
EACCESS權限不足
EOPNOTSUPP指定的socket並未支援listen模式。
範例
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
 
#define PORT 1234
#define MAXSOCKFD 10
main()
{
int sockfd,newsockfd,is_connected[MAXSOCKFD],fd;
struct sockaddr_in addr;
int addr_len = sizeof(struct sockaddr_in);
fd_set readfds;
char buffer[256];
char msg[ ] =”Welcome to server!”;
if ((sockfd = socket(AF_INET,SOCK_STREAM,0))<0)
{
perror(“socket”);
exit(1);
}
bzero(&addr,sizeof(addr));
addr.sin_family =AF_INET;
addr.sin_port = htons(PORT);
addr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(sockfd,&addr,sizeof(addr))<0)
{
perror(“connect”);
exit(1);
}
if (listen(sockfd,3)<0)
{
perror(“listen”);
exit(1);
}
for (fd=0; fd<MAXSOCKFD; fd++)
is_connected[fd]=0;
while(1)
{
FD_ZERO(&readfds);
FD_SET(sockfd,&readfds);
for (fd=0; fd<MAXSOCKFD; fd++)
if (is_connected[fd])
FD_SET(fd,&readfds);
if (!select(MAXSOCKFD, &readfds, NULL, NULL, NULL))
continue;
for (fd=0;fd<MAXSOCKFD;fd++)
if (FD_ISSET(fd,&readfds))
{
if (sockfd = =fd)
{
if((newsockfd = accept (sockfd,&addr,&addr_len))<0)
perror(“accept”);
write(newsockfd,msg,sizeof(msg));
is_connected[newsockfd] =1;
printf(“cnnect from %s\n”,inet_ntoa(addr.sin_addr));//獲取客戶端IP地址
}
else
{
bzero(buffer, sizeof(buffer));
if (read(fd,buffer,sizeof(buffer))<=0)
{
printf(“connect closed.\n”);
is_connected[fd]=0;
close(fd);
}
else
printf(“%s”,buffer);
}
}
}
}
執行
$ ./listen
connect from 127.0.0.1
hi I am client
connected closed.
ntohl(將32位網絡字符順序轉換成主機字符順序)
相關函數 htonlhtonsntohs
表頭文件 #include<netinet/in.h>
定義函數 unsigned long int ntohl(unsigned long int netlong);
函數說明 ntohl()用來將參數指定的32netlong轉換成主機字符順序。
返回值 返回對應的主機字符順序。
範例 參考getservent()
ntohs(將16位網絡字符順序轉換成主機字符順序)
相關函數 htonlhtonsntohl
表頭文件 #include<netinet/in.h>
定義函數 unsigned short int ntohs(unsigned short int netshort);
函數說明 ntohs()用來將參數指定的16netshort轉換成主機字符順序。
返回值 返回對應的主機順序。
範例 參考getservent()
recv(經socket接收數據)
相關函數 recvfromrecvmsgsendsendtosocket
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int recv(int s,void *buf, int len, unsigned int flags);
函數說明 recv()用來接收遠端主機經指定的socket傳來的數據,並把數據存到由參數buf 指向的內存空間,參數len爲可接收數據的最大長度。
參數
返回值
flags一般設0。其他數值定義如下:
MSG_OOB接收以out-of-band 送出的數據。
MSG_PEEK返回來的數據並不會在系統內刪除,如果再調用recv()會返回相同的數據內容。
MSG_WAITALL強迫接收到len大小的數據後才能返回,除非有錯誤或信號產生。
MSG_NOSIGNAL此操作不願被SIGPIPE信號中斷返回值成功則返回接收到的字符數,失敗返回-1,錯誤原因存於errno中。
錯誤代碼
EBADF參數s非合法的socket處理代碼
FAULT參數中有一指針指向無法存取的內存空間
ENOTSOCK參數s爲一文件描述詞,非socket
EINTR被信號所中斷
EAGAIN此動作會令進程阻斷,但參數ssocket爲不可阻斷
ENOBUFS系統的緩衝內存不足。
ENOMEM核心內存不足
EINVAL傳給系統調用的參數不正確。
範例 參考listen()
recvfrom(經socket接收數據)
相關函數 recvrecvmsgsendsendtosocket
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int recvfrom(int s,void *buf,int len,unsigned int flags ,struct sockaddr *from ,int *fromlen);
函數說明 recv()用來接收遠程主機經指定的socket 傳來的數據,並把數據存到由參數buf 指向的內存空間,參數len 爲可接收數據的最大長度。參數flags 一般設0,其他數值定義請參考recv()。參數from用來指定欲傳送的網絡地址,結構sockaddr 請參考bind()。參數fromlensockaddr的結構長度。
返回值 成功則返回接收到的字符數,失敗則返回-1,錯誤原因存於errno中。
錯誤代碼
EBADF參數s非合法的socket處理代碼
EFAULT參數中有一指針指向無法存取的內存空間。
ENOTSOCK參數s爲一文件描述詞,非socket
EINTR被信號所中斷。
EAGAIN此動作會令進程阻斷,但參數ssocket爲不可阻斷。
ENOBUFS系統的緩衝內存不足
ENOMEM核心內存不足
EINVAL傳給系統調用的參數不正確。
範例
/*利用socketUDP client
此程序會連線UDP server,並將鍵盤輸入的字符串傳給server
* UDP server 範例請參考sendto()。
*/
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<sys/typs.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>
 
#define PORT 2345
#define SERVER_IP “127.0.0.1”
 
main()
{
int s,len;
struct sockaddr_in addr;
int addr_len =sizeof(struct sockaddr_in);
char buffer[256];
/* 建立socket*/
if ((s = socket(AF_INET,SOCK_DGRAM,0))<0)
{
perror(“socket”);
exit(1);
}
/* 填寫sockaddr_in*/
bzero(&addr,sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(PORT);
addr.sin_addr.s_addr = inet_addr(SERVER_IP);
while(1)
{
bzero(buffer,sizeof(buffer));
/* 從標準輸入設備取得字符串*/
len =read(STDIN_FILENO,buffer,sizeof(buffer));
/* 將字符串傳送給server*/
sendto(s,buffer,len,0,&addr,addr_len);
/* 接收server端返回的字符串*/
len = recvfrom(s,buffer,sizeof(buffer),0,&addr,&addr_len);
printf(“receive: %s”,buffer);
}
}
執行
(先執行udp server 再執行udp client)
hello /*從鍵盤輸入字符串*/
receive: hello /*server端返回來的字符串*/
recvmsg(經socket接收數據)
相關函數 recvrecvfromsendsendtosendmsgsocket
表頭文件
#include<sys/types.h>
#include<sys/socktet.h>
定義函數 int recvmsg(int s,struct msghdr *msg,unsigned int flags);
函數說明 recvmsg()用來接收遠程主機經指定的socket傳來的數據。參數s爲已建立好連線的socket,如果利用UDP協議則不需經過連線操作。參數msg指向欲連線的數據結構內容,參數flags一般設0,詳細描述請參考send()。關於結構msghdr的定義請參考sendmsg()
返回值 成功則返回接收到的字符數,失敗則返回-1,錯誤原因存於errno中。
錯誤代碼
EBADF參數s非合法的socket處理代碼。
EFAULT參數中有一指針指向無法存取的內存空間
ENOTSOCK參數s爲一文件描述詞,非socket
EINTR被信號所中斷。
EAGAIN此操作會令進程阻斷,但參數ssocket爲不可阻斷。
ENOBUFS系統的緩衝內存不足
ENOMEM核心內存不足
EINVAL傳給系統調用的參數不正確。
範例 參考recvfrom()
send(經socket傳送數據)
相關函數 sendtosendmsgrecvrecvfromsocket
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int send(int s,const void * msg,int len,unsigned int falgs);
函數說明
send()用來將數據由指定的socket 傳給對方主機。參數s爲已建立好連接的socket。參數msg指向欲連線的數據內容,參數len則爲數據長度。參數flags一般設0,其他數值定義如下
MSG_OOB傳送的數據以out-of-band 送出。
MSG_DONTROUTE取消路由表查詢
MSG_DONTWAIT設置爲不可阻斷運作
MSG_NOSIGNAL此動作不願被SIGPIPE 信號中斷。
返回值 成功則返回實際傳送出去的字符數,失敗返回-1。錯誤原因存於errno
錯誤代碼
EBADF參數非合法的socket處理代碼。
EFAULT參數中有一指針指向無法存取的內存空間
ENOTSOCK參數s爲一文件描述詞,非socket
EINTR被信號所中斷。
EAGAIN此操作會令進程阻斷,但參數ssocket爲不可阻斷。
ENOBUFS系統的緩衝內存不足
ENOMEM核心內存不足
EINVAL傳給系統調用的參數不正確。
範例 參考connect()
sendmsg(經socket傳送數據)
相關函數 sendsendtorecvrecvfromrecvmsgsocket
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int sendmsg(int s,const strcut msghdr *msg,unsigned int flags);
函數說明
sendmsg()用來將數據由指定的socket傳給對方主機。參數s爲已建立好連線的socket,如果利用UDP協議則不需經過連線操作。參數msg 指向欲連線的數據結構內容,參數flags一般默認爲0,詳細描述請參考send()
結構msghdr定義如下
struct msghdr {
void *msg_name; /*Address to send to /receive from . */
socklen_t msg_namelen; /* Length of addres data */
strcut iovec * msg_iov; /* Vector of data to send/receive into */
size_t msg_iovlen; /* Number of elements in the vector */
void * msg_control; /* Ancillary dat */
size_t msg_controllen; /* Ancillary data buffer length */
int msg_flags; /* Flags on received message */
};
返回值 成功則返回實際傳送出去的字符數,失敗返回-1,錯誤原因存於errno
錯誤代碼
EBADF參數非合法的socket處理代碼。
EFAULT參數中有一指針指向無法存取的內存空間
ENOTSOCK參數s爲一文件描述詞,非socket
EINTR被信號所中斷。
EAGAIN此操作會令進程阻斷,但參數ssocket爲不可阻斷。
ENOBUFS系統的緩衝內存不足
ENOMEM核心內存不足
EINVAL傳給系統調用的參數不正確。
範例 參考sendto()
sendto(經socket傳送數據)
相關函數 send , sendmsg,recv , recvfrom , socket
表頭文件
#include < sys/types.h >
#include < sys/socket.h >
定義函數
int sendto ( int s , const void * msg, int len, unsigned int flags, const struct sockaddr * to , int tolen ) ;
函數說明 sendto() 用來將數據由指定的socket傳給對方主機。參數s爲已建好連線的socket,如果利用UDP協議則不需經過連線操作。參數msg指向欲連線的數據內容,參數flags 一般設0,詳細描述請參考send()。參數to用來指定欲傳送的網絡地址,結構sockaddr請參考bind()。參數tolensockaddr的結果長度。
返回值 成功則返回實際傳送出去的字符數,失敗返回-1,錯誤原因存於errno 中。
錯誤代碼
EBADF參數s非法的socket處理代碼。
EFAULT參數中有一指針指向無法存取的內存空間。
WNOTSOCK canshu s爲一文件描述詞,非socket
EINTR被信號所中斷。
EAGAIN此動作會令進程阻斷,但參數ssoket爲補課阻斷的。
ENOBUFS系統的緩衝內存不足。
EINVAL傳給系統調用的參數不正確。
範例
#include < sys/types.h >
#include < sys/socket.h >
# include <netinet.in.h>
#include <arpa.inet.h>
#define PORT 2345 /*使用的port*/
 
main()
{
int sockfd,len;
struct sockaddr_in addr;
char buffer[256];
/*建立socket*/
if (sockfd=socket (AF_INET,SOCK_DGRAM,0))<0}
{
perror (“socket”);
exit(1);
}
/*填寫sockaddr_in 結構*/
bzero ( &addr, sizeof(addr) );
addr.sin_family=AF_INET;
addr.sin_port=htons(PORT);
addr.sin_addr=hton1(INADDR_ANY) ;
if (bind(sockfd, &addr, sizeof(addr))<0)
{
perror(“connect”);
exit(1);
}
while(1)
{
bezro(buffer,sizeof(buffer));
len = recvfrom(socket,buffer,sizeof(buffer), 0 , &addr &addr_len);
/*顯示client端的網絡地址*/
printf(“receive from %s\n “ , inet_ntoa( addr.sin_addr));
/*將字串返回給client*/
sendto(sockfd,buffer,len,0,&addr,addr_len);”
}
}
執行 請參考recvfrom()
setprotoent(打開網絡協議的數據文件)
相關函數 getprotobyname, getprotobynumber, endprotoent
表頭文件 #include <netdb.h>
定義函數 void setprotoent (int stayopen);
函數說明 setprotoent()用來打開/etc/protocols如果參數stayopen值爲1,則接下來的getprotobyname()getprotobynumber()將不會自動關閉此文件。
setservent(打開主機網絡服務的數據文件)
相關函數 getservent, getservbyname, getservbyport, endservent
表頭文件 #include < netdb.h >
定義函數 void setservent (int stayopen);
函數說明 setservent()用來打開/etc/services,如果參數stayopen值爲1,則接下來的getservbyname()getservbyport()將補回自動關閉文件。
setsockopt(設置socket狀態)
相關函數 getsockopt
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int setsockopt(int s,int level,int optname,const void * optval,,socklen_toptlen);
函數說明
setsockopt()用來設置參數s所指定的socket狀態。參數level代表欲設置的網絡層,一般設成SOL_SOCKET以存取socket層。參數optname代表欲設置的選項,有下列幾種數值:
SO_DEBUG打開或關閉排錯模式
SO_REUSEADDR允許在bind()過程中本地地址可重複使用
SO_TYPE返回socket形態。
SO_ERROR返回socket已發生的錯誤原因
SO_DONTROUTE送出的數據包不要利用路由設備來傳輸。
SO_BROADCAST使用廣播方式傳送
SO_SNDBUF設置送出的暫存區大小
SO_RCVBUF設置接收的暫存區大小
SO_KEEPALIVE定期確定連線是否已終止。
SO_OOBINLINE當接收到OOB 數據時會馬上送至標準輸入設備
SO_LINGER確保數據安全且可靠的傳送出去。
參數 optval代表欲設置的值,參數optlen則爲optval的長度。
返回值 成功則返回0,若有錯誤則返回-1,錯誤原因存於errno
附加說明
EBADF參數s並非合法的socket處理代碼。
ENOTSOCK參數s爲一文件描述詞,非socket
ENOPROTOOPT參數optname指定的選項不正確。
EFAULT參數optval指針指向無法存取的內存空間。
範例 參考getsockopt()
shutdown(終止socket通信)
相關函數 socketconnect
表頭文件 #include<sys/socket.h>
定義函數 int shutdown(int s,int how);
函數說明
shutdown()用來終止參數s所指定的socket連線。參數s是連線中的socket處理代碼,參數how有下列幾種情況:
how=0 終止讀取操作。
how=1 終止傳送操作
how=2 終止讀取及傳送操作
返回值 成功則返回0,失敗返回-1,錯誤原因存於errno
錯誤代碼
EBADF參數s不是有效的socket處理代碼。
ENOTSOCK參數s爲一文件描述詞,非socket
ENOTCONN參數s指定的socket並未連線。
socket(建立一個socket通信)
相關函數 acceptbindconnectlisten
表頭文件
#include<sys/types.h>
#include<sys/socket.h>
定義函數 int socket(int domain,int type,int protocol);
函數說明
socket()用來建立一個新的socket,也就是向系統註冊,通知系統建立一通信端口。參數domain 指定使用何種的地址類型,完整的定義在/usr/include/bits/socket.h 內,底下是常見的協議:
PF_UNIX/PF_LOCAL/AF_UNIX/AF_LOCAL UNIX 進程通信協議。
PF_INET?AF_INET Ipv4網絡協議。
PF_INET6/AF_INET6 Ipv6 網絡協議。
PF_IPX/AF_IPX IPX-Novell協議。
PF_NETLINK/AF_NETLINK核心用戶接口裝置。
PF_X25/AF_X25 ITU-T X.25/ISO-8208 協議。
PF_AX25/AF_AX25業餘無線AX.25協議。
PF_ATMPVC/AF_ATMPVC存取原始ATM PVCs
PF_APPLETALK/AF_APPLETALK appletalkDDP)協議。
PF_PACKET/AF_PACKET初級封包接口
參數
type有下列幾種數值:
SOCK_STREAM提供雙向連續且可信賴的數據流,即TCP。支持
OOB 機制,在所有數據傳送前必須使用connect()來建立連線狀態。
SOCK_DGRAM使用不連續不可信賴的數據包連接
SOCK_SEQPACKET提供連續可信賴的數據包連接
SOCK_RAW提供原始網絡協議存取
SOCK_RDM提供可信賴的數據包連接
SOCK_PACKET提供和網絡驅動程序直接通信。
protocol用來指定socket所使用的傳輸協議編號,通常此參考不用管它,設
返回值 成功則返回socket處理代碼,失敗返回-1
錯誤代碼
EPROTONOSUPPORT參數domain指定的類型不支持參數typeprotocol指定的協議
ENFILE核心內存不足,無法建立新的socket結構;
EMFILE進程文件表溢出,無法再建立新的socket
EACCESS權限不足,無法建立typeprotocol指定的協議
ENOBUFS/ENOMEM 內存不足
EINVAL參數domain/type/protocol不合法
範例 參考connect()


發佈了369 篇原創文章 · 獲贊 587 · 訪問量 533萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章