unix下編寫socket程式的一般步驟(轉)
unix下編寫socket程式的一般步驟(轉)[@more@]在unix下寫socket程式可能是最方便,你只要掌握其一般步驟,就可以松的寫出面向傳輸層的應用。
1、理解幾個常用的socket函式
#include
#include
int socket(int domain,int type,int portocol);
domain指所使用的協議族(family)可以為AF_UNIX和AF_INET,一般只用AF_INET(指Internet)type指所用的傳輸型別,可以為SOCK_STERAM(面向連線的TCP),和SOCK_DGRAM(面向無連線的udp)
int bind(int s,const struct sockaddr *address,size_t address_len);
s為socket返回的檔案描述符
address為協議族名稱和其他資訊
具體結構為struct sockaddr_in{
short sin_family;/*協議族
u_short sin_port;/*埠*/
struct in_addr sin_addr;/*地址*/
char sin_zero[8];
};
int listen(int s,int backlog);
backlog為容許的請求數目
int accept(int s,struct sockaddr *address,int *address_len);
這裡的前兩個引數同上
addres_len是要傳遞一個記有結構大小的地址
int connect(int s,struct sockaddr *address,size_t address_len);
這裡的引數意義同bind
2.理解建立程式的一般呼叫過程
要建立一個處理連線的伺服器端程式,首先要呼叫socket函式建立一個socket,返回一個檔案控制程式碼fd,使以後對它的操作就象對普通檔案裝置一樣讀寫。
由於是伺服器端必須對一個斷口進行監聽其他機器的請求,所以接下去呼叫bind函式,傳入剛才的fd,定義好地址和埠,由於是要接受來自任何host的連線所以應講sin_addr賦為INADDR_ANY,port為你所設定的埠。
注意:這裡的地址和埠是網路位元組順序,所以要呼叫htonl,htons完成主機位元組順序
到網路位元組的轉變
接下來就是監聽listen,呼叫accept接受來自客戶端的請求,accpet返回連線後的檔案描述符,你就可以用它進行收發資訊(對應於read, write)這樣的一個過程就是socket->bind->listen->accpet->Read,write
而對於客戶端則是socket->connect->read,write
3.一個完整的程式
#include
#include
#include /*包含有htons等函式的標頭檔案*/
#include
#include
void main()
{
int listenfd,clifd;
long pid;
struct sockaddr_in myaddr,cliaddr;
int ret;
int len;
listenfd=socket(AF_INET,SOCK_STREAM,0);
if (listenfd<0)
{
perror("socket error");
exit(-1);
}
myaddr.sin_family=AF_INET;
myaddr.sin_addr.s_addr=htonl(INADDR_ANY);
myaddr.sin_port=htons(8888);
ret=bind(listenfd,(struct sockaddr *)&myaddr,sizeof(myaddr));
if (ret<0)
{
perror("bind error");
exit(-1);
}
listen(listenfd,10);
len=sizeof(struct sockaddr);
while(1)
{
clifd=accept(listenfd,(struct sockaddr*)&cliaddr,&len);
/*注意accept的第三個引數也是地址*/
if(clifd==-1)
{
perror("accept error");
continue;
}
printf("connect from %s %d ",inet_ntoa(cliaddr.sin_addr.s_addr),
ntohs(cliaddr.sin_port));
switch(pid=fork())
{
case 0: /*子程式*/
close(listenfd);
;/*子程式進行其他的操作*/
close(clifd);
exit(0);
break;
case -1:
perror("fork error");
break;
default:/*父程式*/
close(clifd);
break;
}
}
}
4.程式說明
該程式的功能是監聽8888埠的連線,對所有的對8888埠的連線顯示出地址和對方的埠號該程式在sco unix下除錯透過,在其他unix和linux平臺請注意inet_ntoa,htons函式所應在的標頭檔案的名稱
同時該程式用到了併發的觀點,因為accept,read,write均為阻塞(block)的函式,一旦程式block將不能處理其他請求,所以用主程式進行listen,由子程式進行負責對客戶端傳輸資料.
你可以在同一臺unix機器用telnet localhost 8888進行觀察程式會輸出connect from 127.0.0.1 xxxx
1、理解幾個常用的socket函式
#include
#include
int socket(int domain,int type,int portocol);
domain指所使用的協議族(family)可以為AF_UNIX和AF_INET,一般只用AF_INET(指Internet)type指所用的傳輸型別,可以為SOCK_STERAM(面向連線的TCP),和SOCK_DGRAM(面向無連線的udp)
int bind(int s,const struct sockaddr *address,size_t address_len);
s為socket返回的檔案描述符
address為協議族名稱和其他資訊
具體結構為struct sockaddr_in{
short sin_family;/*協議族
u_short sin_port;/*埠*/
struct in_addr sin_addr;/*地址*/
char sin_zero[8];
};
int listen(int s,int backlog);
backlog為容許的請求數目
int accept(int s,struct sockaddr *address,int *address_len);
這裡的前兩個引數同上
addres_len是要傳遞一個記有結構大小的地址
int connect(int s,struct sockaddr *address,size_t address_len);
這裡的引數意義同bind
2.理解建立程式的一般呼叫過程
要建立一個處理連線的伺服器端程式,首先要呼叫socket函式建立一個socket,返回一個檔案控制程式碼fd,使以後對它的操作就象對普通檔案裝置一樣讀寫。
由於是伺服器端必須對一個斷口進行監聽其他機器的請求,所以接下去呼叫bind函式,傳入剛才的fd,定義好地址和埠,由於是要接受來自任何host的連線所以應講sin_addr賦為INADDR_ANY,port為你所設定的埠。
注意:這裡的地址和埠是網路位元組順序,所以要呼叫htonl,htons完成主機位元組順序
到網路位元組的轉變
接下來就是監聽listen,呼叫accept接受來自客戶端的請求,accpet返回連線後的檔案描述符,你就可以用它進行收發資訊(對應於read, write)這樣的一個過程就是socket->bind->listen->accpet->Read,write
而對於客戶端則是socket->connect->read,write
3.一個完整的程式
#include
#include
#include /*包含有htons等函式的標頭檔案*/
#include
#include
void main()
{
int listenfd,clifd;
long pid;
struct sockaddr_in myaddr,cliaddr;
int ret;
int len;
listenfd=socket(AF_INET,SOCK_STREAM,0);
if (listenfd<0)
{
perror("socket error");
exit(-1);
}
myaddr.sin_family=AF_INET;
myaddr.sin_addr.s_addr=htonl(INADDR_ANY);
myaddr.sin_port=htons(8888);
ret=bind(listenfd,(struct sockaddr *)&myaddr,sizeof(myaddr));
if (ret<0)
{
perror("bind error");
exit(-1);
}
listen(listenfd,10);
len=sizeof(struct sockaddr);
while(1)
{
clifd=accept(listenfd,(struct sockaddr*)&cliaddr,&len);
/*注意accept的第三個引數也是地址*/
if(clifd==-1)
{
perror("accept error");
continue;
}
printf("connect from %s %d ",inet_ntoa(cliaddr.sin_addr.s_addr),
ntohs(cliaddr.sin_port));
switch(pid=fork())
{
case 0: /*子程式*/
close(listenfd);
;/*子程式進行其他的操作*/
close(clifd);
exit(0);
break;
case -1:
perror("fork error");
break;
default:/*父程式*/
close(clifd);
break;
}
}
}
4.程式說明
該程式的功能是監聽8888埠的連線,對所有的對8888埠的連線顯示出地址和對方的埠號該程式在sco unix下除錯透過,在其他unix和linux平臺請注意inet_ntoa,htons函式所應在的標頭檔案的名稱
同時該程式用到了併發的觀點,因為accept,read,write均為阻塞(block)的函式,一旦程式block將不能處理其他請求,所以用主程式進行listen,由子程式進行負責對客戶端傳輸資料.
你可以在同一臺unix機器用telnet localhost 8888進行觀察程式會輸出connect from 127.0.0.1 xxxx
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/10617731/viewspace-957686/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- PHP 編寫基本的 Socket 程式PHP
- 在Unix下用C編寫curses程式的一些常用模組(轉)
- Linux下安裝Oracle的一般步驟LinuxOracle
- Unix系列shell程式編寫
- debian下編譯2.6.13.2核心的步驟及感受(轉)編譯
- mapreduce的一般執行步驟
- RequisitPro管理需求的一般步驟UI
- 網路入侵方法與一般步驟 1 (轉)
- 網路入侵方法與一般步驟2 (轉)
- 網路入侵方法與一般步驟3 (轉)
- Unix系列shell程式編寫:下篇
- Unix系列shell程式編寫:中篇
- Unix系列shell程式編寫:上篇
- 2、編寫/修改許可權及執行Shell程式的步驟
- 定製燒錄SCO UNIX安裝光碟的步驟(轉)
- 全庫匯入的一般步驟
- Sybase 11.0.3 For SCO UNIX 5.0.5安裝配置步驟(轉)
- sql最佳化一般步驟SQL
- UNIX系統中安裝SAP的步驟
- BOOT0的主要程式碼兼Unix下彙編小節(轉)boot
- Linux系統下編譯Openssl 步驟:Linux編譯
- 學習C#的一般性步驟C#
- 一般安裝EBS 後的 操作步驟,徵集!
- HP unix無法進入CDE的排查步驟
- 抖音去水印詳細原理步驟及介面編寫
- SQL隱碼攻擊的原理及一般步驟SQL
- Spring整合Mybatis的一般步驟(IDEA版)SpringMyBatisIdea
- 總結一下使用pytorch搭建神經網路的一般步驟PyTorch神經網路
- 學習編譯原理的步驟編譯原理
- 用匯編編寫DOS下的記憶體駐留程式(5) (轉)記憶體
- 用匯編編寫DOS下的記憶體駐留程式(3) (轉)記憶體
- 用匯編編寫DOS下的記憶體駐留程式(4) (轉)記憶體
- Linux下編譯安裝Mysql 5.5的簡單步驟Linux編譯MySql
- 開源專案匯入eclipse的一般步驟Eclipse
- unix socket通訊
- 建立BAPI程式的步驟API
- 簡單的介紹UNIX下的常用編輯工具VI(轉)
- 編寫易讀的程式碼 (轉)