相關函數 socket,bind,listen,connect
表頭文件 #include<sys/types.h>
#include<sys/socket.h>
定義函數 int accept(int s,struct sockaddr * addr,int * addrlen);
函數說明 accept()用來接受參數s的socket連線。參數s的socket必需先經bind()、listen()函數處理過,當有連線進來時 accept()會返回一個新的socket處理代碼,往後的數據傳送與讀取就是經由新的socket處理,而原來參數s的socket能繼續使用 accept()來接受新的連線要求。連線成功時,參數addr所指的結構會被系統填入遠程主機的地址數據,參數addrlen為scokaddr的結構長度。關於結構sockaddr的定義請參考bind()。
返回值 成功則返回新的socket處理代碼,失敗返回-1,錯誤原因存於errno中。
錯誤代碼 EBADF 參數s 非合法socket處理代碼。
EFAULT 參數addr指針指向無法存取的內存空間。
ENOTSOCK 參數s為一文件描述詞,非socket。
EOPNOTSUPP 指定的socket並非SOCK_STREAM。
EPERM 防火牆拒絕此連線。
ENOBUFS 系統的緩沖內存不足。
ENOMEM 核心內存不足。
范例 參考listen()。
bind(對socket定位)
相關函數 socket,accept,connect,listen
表頭文件 #include<sys/types.h>
#include<sys/socket.h>
定義函數 int bind(int sockfd,struct sockaddr * my_addr,int addrlen);
函數說明 bind()用來設置給參數sockfd的socket一個名稱。此名稱由參數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 未使用。
參數 addrlen為sockaddr的結構長度。
返回值 成功則返回0,失敗返回-1,錯誤原因存於errno中。
錯誤代碼 EBADF 參數sockfd 非合法socket處理代碼。
EACCESS 權限不足
ENOTSOCK 參數sockfd為一文件描述詞,非socket。
范例 參考listen()
connect(建立socket連線)
相關函數 socket,bind,listen
表頭文件 #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()。參數addrlen為sockaddr的結構長度。
返回值 成功則返回0,失敗返回-1,錯誤原因存於errno中。
錯誤代碼 EBADF 參數sockfd 非合法socket處理代碼
EFAULT 參數serv_addr指針指向無法存取的內存空間
ENOTSOCK 參數sockfd為一文件描述詞,非socket。
EISCONN 參數sockfd的socket已是連線狀態
ECONNREFUSED 連線要求被server端拒絕。
ETIMEDOUT 企圖連線的操作超過限定時間仍未有響應。
ENETUNREACH 無法傳送數據包至指定的主機。
EAFNOSUPPORT sockaddr結構的sa_family不正確。
EALREADY socket為不可阻斷且先前的連線操作還未完成。
范例 /* 利用socket的TCP 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(結束網絡協議數據的讀取)
相關函數 getprotoent,getprotobyname,getprotobynumber,setprotoent
表頭文件 #include<netdb.h>
定義函數 void endprotoent(void);
函數說明 endprotoent()用來關閉由getprotoent()打開的文件。
返回值
范例 參考getprotoent()
endservent(結束網絡服務數據的讀取)
相關函數 getservent,getservbyname,getservbyport,setservent
表頭文件 #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則為該空間的大小。參數level、optname請參考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的定義正是此值*/