函數(shù)說(shuō)明:socket()用來(lái)建立一個(gè)新的socket, 也就是向系統(tǒng)注冊(cè), 通知系統(tǒng)建立一通信端口. 參數(shù)domain 指定使用何種的地址類型, 完整的定義在/usr/include/bits/socket.h 內(nèi), 底下是常見的協(xié)議:
創(chuàng)新互聯(lián)建站主要從事網(wǎng)站設(shè)計(jì)制作、成都網(wǎng)站建設(shè)、網(wǎng)頁(yè)設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)西固,十余年網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專業(yè),歡迎來(lái)電咨詢建站服務(wù):13518219792
PF_UNIX/PF_LOCAL/AF_UNIX/AF_LOCAL UNIX 進(jìn)程通信協(xié)議
PF_INET?AF_INET Ipv4 網(wǎng)絡(luò)協(xié)議
PF_INET6/AF_INET6 Ipv6 網(wǎng)絡(luò)協(xié)議
PF_IPX/AF_IPX IPX-Novell 協(xié)議
PF_NETLINK/AF_NETLINK 核心用戶接口裝置
PF_X25/AF_X25 ITU-T X. 25/ISO-8208 協(xié)議
PF_AX25/AF_AX25 業(yè)余無(wú)線AX. 25 協(xié)議
PF_ATMPVC/AF_ATMPVC 存取原始 ATM PVCs
PF_APPLETALK/AF_APPLETALK appletalk (DDP)協(xié)議
PF_PACKET/AF_PACKET 初級(jí)封包接口
參數(shù) type 有下列幾種數(shù)值:
1、SOCK_STREAM 提供雙向連續(xù)且可信賴的數(shù)據(jù)流, 即TCP. 支持 OOB 機(jī)制, 在所有數(shù)據(jù)傳送前必須使用connect()來(lái)建立連線狀態(tài).
2、SOCK_DGRAM 使用不連續(xù)不可信賴的數(shù)據(jù)包連接
3、SOCK_SEQPACKET 提供連續(xù)可信賴的數(shù)據(jù)包連接
4、SOCK_RAW 提供原始網(wǎng)絡(luò)協(xié)議存取
5、SOCK_RDM 提供可信賴的數(shù)據(jù)包連接
6、SOCK_PACKET 提供和網(wǎng)絡(luò)驅(qū)動(dòng)程序直接通信. protocol 用來(lái)指定socket 所使用的傳輸協(xié)議編號(hào), 通常此參考不用管它, 設(shè)為0 即可.
返回值:成功則返回socket 處理代碼, 失敗返回-1.
錯(cuò)誤代碼:
1、EPROTONOSUPPORT 參數(shù)domain 指定的類型不支持參數(shù)type 或protocol 指定的協(xié)議
2、ENFILE 核心內(nèi)存不足, 無(wú)法建立新的socket 結(jié)構(gòu)
3、EMFILE 進(jìn)程文件表溢出, 無(wú)法再建立新的socket
4、EACCESS 權(quán)限不足, 無(wú)法建立type 或protocol 指定的協(xié)議
5、ENOBUFS/ENOMEM 內(nèi)存不足
6、EINVAL 參數(shù)domain/type/protocol 不合法
char *itoa(int x,char*string1,int radix);
char *itoa(int y,char*string2,int radix);
這兩行你寫成聲明了。應(yīng)該用
itoa(x, string1, 10);
itoa(y, string2, 10);
在函數(shù)調(diào)用的時(shí)候,只是把原本參數(shù)的值賦值給了函數(shù)中的形式參數(shù)。在exchange函數(shù)中操作的是形參,沒有對(duì)原本函數(shù)做出操作,所以輸出結(jié)果是 x = 4,y = 2 a = 2,b = 4。
這次在函數(shù)調(diào)用的時(shí)候是吧a,b的地址賦值給了px,py,所以交換的是真正的a,b變量此時(shí)輸出的結(jié)果是*px = 4,*py = 2
a = 4,b =2 。
這個(gè)取地址的引用傳遞雖然看起來(lái)像值傳遞,然而確實(shí)真正的改變?cè)瓉?lái)常量的值,就像下棋一樣操作的不是棋子,而是棋手。
使用socket()編程即可。
1、網(wǎng)絡(luò)的Socket數(shù)據(jù)傳輸是一種特殊的I/O,Socket也是一種文件描述符。Socket也具有一個(gè)類似于打開文件的函數(shù)調(diào)用Socket(),該函數(shù)返回一個(gè)整型的Socket描述符,隨后的連接建立、數(shù)據(jù)傳輸?shù)炔僮鞫际峭ㄟ^該Socket實(shí)現(xiàn)的。
2、下面用Socket實(shí)現(xiàn)一個(gè)windows下的c語(yǔ)言socket通信例子,這里客戶端傳遞一個(gè)字符串,服務(wù)器端進(jìn)行接收。
【服務(wù)器端】
#include "stdafx.h"
#include stdio.h
#include winsock2.h
#include winsock2.h
#define SERVER_PORT 5208 //偵聽端口
void main()
{
WORD wVersionRequested;
WSADATA wsaData;
int ret, nLeft, length;
SOCKET sListen, sServer; //偵聽套接字,連接套接字
struct sockaddr_in saServer, saClient; //地址信息
char *ptr;//用于遍歷信息的指針
//WinSock初始化
wVersionRequested=MAKEWORD(2, 2); //希望使用的WinSock DLL 的版本
ret=WSAStartup(wVersionRequested, wsaData);
if(ret!=0)
{
printf("WSAStartup() failed!\n");
return;
}
//創(chuàng)建Socket,使用TCP協(xié)議
sListen=socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (sListen == INVALID_SOCKET)
{
WSACleanup();
printf("socket() faild!\n");
return;
}
//構(gòu)建本地地址信息
saServer.sin_family = AF_INET; //地址家族
saServer.sin_port = htons(SERVER_PORT); //注意轉(zhuǎn)化為網(wǎng)絡(luò)字節(jié)序
saServer.sin_addr.S_un.S_addr = htonl(INADDR_ANY); //使用INADDR_ANY 指示任意地址
//綁定
ret = bind(sListen, (struct sockaddr *)saServer, sizeof(saServer));
if (ret == SOCKET_ERROR)
{
printf("bind() faild! code:%d\n", WSAGetLastError());
closesocket(sListen); //關(guān)閉套接字
WSACleanup();
return;
}
//偵聽連接請(qǐng)求
ret = listen(sListen, 5);
if (ret == SOCKET_ERROR)
{
printf("listen() faild! code:%d\n", WSAGetLastError());
closesocket(sListen); //關(guān)閉套接字
return;
}
printf("Waiting for client connecting!\n");
printf("Tips: Ctrl+c to quit!\n");
//阻塞等待接受客戶端連接
while(1)//循環(huán)監(jiān)聽客戶端,永遠(yuǎn)不停止,所以,在本項(xiàng)目中,我們沒有心跳包。
{
length = sizeof(saClient);
sServer = accept(sListen, (struct sockaddr *)saClient, length);
if (sServer == INVALID_SOCKET)
{
printf("accept() faild! code:%d\n", WSAGetLastError());
closesocket(sListen); //關(guān)閉套接字
WSACleanup();
return;
}
char receiveMessage[5000];
nLeft = sizeof(receiveMessage);
ptr = (char *)receiveMessage;
while(nLeft0)
{
//接收數(shù)據(jù)
ret = recv()(sServer, ptr, 5000, 0);
if (ret == SOCKET_ERROR)
{
printf("recv() failed!\n");
return;
}
if (ret == 0) //客戶端已經(jīng)關(guān)閉連接
{
printf("Client has closed the connection\n");
break;
}
nLeft -= ret;
ptr += ret;
}
printf("receive message:%s\n", receiveMessage);//打印我們接收到的消息。
}
// closesocket(sListen);
// closesocket(sServer);
// WSACleanup();
}
【客戶端】
#include "stdafx.h"
#include stdio.h
#include stdlib.h
#include winsock2.h
#define SERVER_PORT 5208 //偵聽端口
void main()
{
WORD wVersionRequested;
WSADATA wsaData;
int ret;
SOCKET sClient; //連接套接字
struct sockaddr_in saServer; //地址信息
char *ptr;
BOOL fSuccess = TRUE;
//WinSock初始化
wVersionRequested = MAKEWORD(2, 2); //希望使用的WinSock DLL的版本
ret = WSAStartup(wVersionRequested, wsaData);
if(ret!=0)
{
printf("WSAStartup() failed!\n");
return;
}
//確認(rèn)WinSock DLL支持版本2.2
if(LOBYTE(wsaData.wVersion)!=2 || HIBYTE(wsaData.wVersion)!=2)
{
WSACleanup();
printf("Invalid WinSock version!\n");
return;
}
//創(chuàng)建Socket,使用TCP協(xié)議
sClient = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (sClient == INVALID_SOCKET)
{
WSACleanup();
printf("socket() failed!\n");
return;
}
//構(gòu)建服務(wù)器地址信息
saServer.sin_family = AF_INET; //地址家族
saServer.sin_port = htons(SERVER_PORT); //注意轉(zhuǎn)化為網(wǎng)絡(luò)節(jié)序
saServer.sin_addr.S_un.S_addr = inet_addr("192.168.1.127");
//連接服務(wù)器
ret = connect(sClient, (struct sockaddr *)saServer, sizeof(saServer));
if (ret == SOCKET_ERROR)
{
printf("connect() failed!\n");
closesocket(sClient); //關(guān)閉套接字
WSACleanup();
return;
}
char sendMessage[]="hello this is client message!";
ret = send (sClient, (char *)sendMessage, sizeof(sendMessage), 0);
if (ret == SOCKET_ERROR)
{
printf("send() failed!\n");
}
else
printf("client info has been sent!");
closesocket(sClient); //關(guān)閉套接字
WSACleanup();
}