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個字符長。

通常的用法是:??
? 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);

二 、說個小知識,為什么一般情況下都用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參數s?非合法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參數s?并非合法的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()用來等待參數s?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參數s?非合法的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參數s?非合法的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()



本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/253261.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/253261.shtml
英文地址,請注明出處:http://en.pswp.cn/news/253261.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Python 購物車

購物車需求&#xff1a; 可購買的商品信息顯示 顯示購物車內的商品信息、數量 購物車內的商品數量進行增加、減少 用戶余額的充值 用戶購買完成進行結賬&#xff0c;將最終余額回寫到用戶文件中   流程圖&#xff1a; 代碼&#xff1a; 1、主文件 def login():# 驗證用戶帳號…

認對畫對MOS管

MOS管我們在設計電路中經常用的一種無源器件。 首先介紹下&#xff0c;在原理圖和PCB以及實物PCBA中如何辨別各種MOS管&#xff0c;作為應用好的先決條件&#xff0c;必須認對畫對管子。 1. MOS管的GSD三極在原理圖和PCB上怎么判定&#xff1a; G極(gate)—柵極&#xff0c;原…

Jmeter之BeanShell

在Jmeter中各種分類組件中都有相應的BeanShell組件&#xff0c;這里簡單的說明一下Beanshell的使用。 一、概念 BeanShell是一種符合Java語法的腳本語言&#xff0c;也有自己的一些特定語法 二、內置變量 Jmeter在它的Beanshell中內置了變量&#xff0c;用戶可以通過這些變量與…

HDU 1875 暢通工程再續

傳送門&#xff1a;http://acm.hdu.edu.cn/showproblem.php?pid1875 簡單的最小生成樹 #include <iostream> #include <cstdio> #include <cmath> using namespace std;const int maxn100005; const double INF1.0e20;struct Node{double x,y; } isl[maxn];…

C語言變長數組 struct中char data[0]的用法

摘要&#xff1a;在實際的編程中&#xff0c;我們經常需要使用變長數組&#xff0c;但是C語言并不支持變長的數組。此時&#xff0c;我們可以使用結構體的方法實現C語言變長數組。 struct MyData {int nLen;char data[0];}; 在結構中&#xff0c;data是一個數組名&#xff1b;但…

MOS管的實際應用

繼上一篇“認對畫對MOS管”后&#xff0c;現在小結一下MOS管的具體應用&#xff1a; 應用MOS管前&#xff0c;理解MOS管每個參數的具體意義后&#xff0c;再額外注意一下管子本身的體二極管&#xff0c;本身Vf1.6V&#xff0c;導通后管子本身阻抗一般是mΩ級&#xff1b;管子廠…

imp導入前對當前用戶清庫腳本

--清空當前用戶所有表begin for i in ( select drop table || a.tab_name as sqls from (select distinct t.tab_name from (select Lower(table_name) as tab_name from user_tables) t) a ) loop dbms_output.put_line(i.sqls); execute immediate i.sqls; end loop;end;/--清…

Spring - Spring Boot Spring Cloud

Spring -> Spring Boot > Spring Cloud 這幾天剛剛上班&#xff0c;公司用的是Spring Cloud&#xff0c;接觸不多。我得趕快學起來。 想學習就必須得知道什么是微服務&#xff0c;什么是Spring Boot&#xff0c;什么是Spring Cloud&#xff0c;以及兩者之間有什么關系&am…

C語言 · 前10名

算法提高 前10名 時間限制&#xff1a;1.0s 內存限制&#xff1a;256.0MB問題描述數據很多&#xff0c;但我們經常只取前幾名&#xff0c;比如奧運只取前3名。現在我們有n個數據&#xff0c;請按從大到小的順序&#xff0c;輸出前10個名數據。輸入格式兩行。第一行一個整數n…

ssacanf\Sprintf格式化字符串

一、sscanf sscanf() - 從一個  int sscanf(const char *buffer,const char *format,[argument ]...); buffer 存儲的數據 format 格式控制字符串 argument 選擇性設定字符串 sscanf會從buffer里讀進數據&#xff0c;依照argument的設定將數據寫回。字符串中讀進與指定格式相…

防火墻規則

1、iptables -t -L -n -t指定表格 -L 顯示目前表格的規則 -n 數字顯示2、iptables-save 以命令方式顯示規則3、清除清空filter從頭制定規則 ipatables -F 清除已經定義 iptables -X 清除自定義鏈 iptables -z 清除鏈統計和計數4、設定默認規則,當所有規則不匹…

JAVA中循環刪除list中元素的方法總結

印象中循環刪除list中的元素使用for循環的方式是有問題的&#xff0c;但是可以使用增強的for循環&#xff0c;然后今天在使用時發現報錯了&#xff0c;然后去科普了一下&#xff0c;再然后發現這是一個誤區。下面就來講一講。。伸手黨可直接跳至文末。看總結。。 JAVA中循環遍歷…

直流有刷電機與無刷電機的區別

首先介紹有刷電機與無刷電機工作原理&#xff0c;最后從調速方式及性能差異這兩個方面詳細的闡述了有刷電機與無刷電機的區別。 有刷電機與無刷電機工作原理 1、有刷電機 電機工作時&#xff0c;線圈和換向器旋轉&#xff0c;磁鋼和碳刷不轉&#xff0c;線圈電流方向的交替變化…

MapReduce詳解

MapReduce簡介 MapReduce是一種編程模型&#xff0c;用于大規模數據集&#xff08;大于1TB&#xff09;的并行運算。概念"Map&#xff08;映射&#xff09;"和"Reduce&#xff08;歸約&#xff09;"&#xff0c;是它們的主要思想。 MapReduce極大地方便了編…

JavaScriptBreak 語句 continue 語句

break 語句用于跳出循環。 continue 用于跳過循環中的一個迭代。 Break 語句 我們已經在本教程之前的章節中見到過 break 語句。它用于跳出 switch() 語句。 break 語句可用于跳出循環。 continue 語句跳出循環后&#xff0c;會繼續執行該循環之后的代碼&#xff08;如果有的話…

kernel mtd 分區與UBOOT 分區的理解

今天做內核移植&#xff0c;準備添加NAND flash的驅動&#xff0c;做到MTD分區時&#xff0c;想起在一本書上看到的一句話&#xff0c;說的是分區時每個區之間沒有間隙&#xff0c;前一個區的結束地址是后一個區的起始地址。可是當我看我的開發板的教程時&#xff0c;分區如下&…

運放的主要參數詳細介紹

1. 引言 運放的作用是調節和放大模擬信號&#xff0c;它是用途十分廣泛的器件&#xff0c;接入適當的反饋網絡&#xff0c;可用作精密的交流和直流放大器、有源濾波器濾波器、振蕩器振蕩器及電壓比較器。其應用領域包括但不限制通訊、電子、汽車、工業檢測等等&#xff0c;并將…

FastDFS 文件上傳工具類

FastDFS文件上傳工具類 import org.csource.common.NameValuePair;import org.csource.fastdfs.ClientGlobal;import org.csource.fastdfs.StorageClient1;import org.csource.fastdfs.StorageServer;import org.csource.fastdfs.TrackerClient;import org.csource.fastdfs.Tra…

MOS管的主要參數與重要特性

雙極性晶體管&#xff1a;NPN和PNP管&#xff1b; 單極性晶體管&#xff1a;場效應管&#xff08;MOSFET和JFET&#xff09;&#xff1b; MOS管相對三極管具有速度快、輸入阻抗高、噪聲低、動態范圍大、功耗小、容易集成等優點。 下面總結下其主要參數與重要特性&#xff0c…

【Codeforces Round #430 (Div. 2) B】Gleb And Pizza

【鏈接】點擊打開鏈接 【題意】 在這里寫題意【題解】 根據圓心到原點的距離這個東西判斷一下圓在不在那個環里面就好【錯的次數】 0【反思】 在這了寫反思【代碼】 #include <cstdio> #include <iostream> #include <algorithm> #include <cstring> #…