欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

如何基于C語(yǔ)言socket編程實(shí)現(xiàn)TCP通信

 更新時(shí)間:2019年05月20日 14:58:51   作者:lovekun1989  
本文介紹了如何基于C語(yǔ)言socket編程實(shí)現(xiàn)TCP通信,下面小編來(lái)簡(jiǎn)單介紹下

TCP/IP協(xié)議(Transmission Control Protocol/Internet Protocol)叫做傳輸控制/網(wǎng)際協(xié)議,又叫網(wǎng)絡(luò)通信協(xié)議。實(shí)際上,它包含上百個(gè)功能的協(xié)議,如ICMP(互聯(lián)網(wǎng)控制信息協(xié)議)、FTP(文件傳輸協(xié)議)、UDP(用戶數(shù)據(jù)包協(xié)議)、ARP(地址解析協(xié)議)等。TCP負(fù)責(zé)發(fā)現(xiàn)傳輸?shù)膯?wèn)題,一旦有問(wèn)題就會(huì)發(fā)出重傳信號(hào),直到所有數(shù)據(jù)安全正確的傳輸?shù)侥康牡亍?/p>

套接字(socket):在網(wǎng)絡(luò)中用來(lái)描述計(jì)算機(jī)中不同程序與其他計(jì)算機(jī)程序的通信方式。socket其實(shí)是一種特殊的IO借口,也是一種文件描述符。

套接字分為三類:

流式socket(SOCK_STREAM):流式套接字提供可靠、面向連接的通信流;它使用TCP協(xié)議,從而保證了數(shù)據(jù)傳輸?shù)恼_性和順序性。

數(shù)據(jù)報(bào)socket(SOCK_DGRAM):數(shù)據(jù)報(bào)套接字定義了一種無(wú)連接的服務(wù),數(shù)據(jù)通過(guò)相互獨(dú)立的保溫進(jìn)行傳輸,是無(wú)序的,并且不保證是可靠、無(wú)差錯(cuò)的。它使用的數(shù)據(jù)報(bào)協(xié)議是UDP。

原始socket:原始套接字允許對(duì)底層協(xié)議如IP或ICMP進(jìn)行直接訪問(wèn),它功能強(qiáng)大但使用復(fù)雜,主要用于一些協(xié)議的開(kāi)發(fā)。

套接字由三個(gè)參數(shù)構(gòu)成:IP地址,端口號(hào),傳輸層協(xié)議。

這三個(gè)參數(shù)用以區(qū)分不同應(yīng)用程序進(jìn)程間的網(wǎng)絡(luò)通信與連接。

套接字的數(shù)據(jù)結(jié)構(gòu):C語(yǔ)言進(jìn)行套接字編程時(shí),常會(huì)使用到sockaddr數(shù)據(jù)類型和sockaddr_in數(shù)據(jù)類型,用于保存套接字信息。

兩種結(jié)構(gòu)體分別表示如下:

struct sockaddr
{
 //地址族,2字節(jié)
 unsigned short sa_family;
 //存放地址和端口,14字節(jié)
 char sa_data[14];
}
 
struct sockaddr_in
{
 //地址族
 short int sin_family;
 //端口號(hào)(使用網(wǎng)絡(luò)字節(jié)序)
 unsigned short int sin_port;
 //地址
 struct in_addr sin_addr;
 //8字節(jié)數(shù)組,全為0,該字節(jié)數(shù)組的作用只是為了讓兩種數(shù)據(jù)結(jié)構(gòu)大小相同而保留的空字節(jié)
 unsigned char sin_zero[8]
}

對(duì)于sockaddr,大部分的情況下只是用于bind,connect,recvfrom,sendto等函數(shù)的參數(shù),指明地址信息,在一般編程中,并不對(duì)此結(jié)構(gòu)體直接操作。而是用sockaddr_in來(lái)代替。

兩種數(shù)據(jù)結(jié)構(gòu)中,地址族都占2個(gè)字節(jié),常見(jiàn)的地址族有:AF_INET,AF_INET6,AF_LOCAL。

這里要注意字節(jié)序的問(wèn)題,最好使用以下函數(shù)來(lái)對(duì)端口和地址進(jìn)行處理:

uint16_t htons(uint16_t host16bit) uint32_t htonl(uint32_t host32bit)
uint16_t ntohs(uint16_t net16bit) uint32_t ntohs(uint32_t net32bit)

將主機(jī)字節(jié)序改成網(wǎng)絡(luò)字節(jié)序。

使用socket進(jìn)行TCP通信時(shí),經(jīng)常使用的函數(shù)有:

下面是TCP通信的demo:

/*socket tcp服務(wù)器端*/
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
 
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
 
#define SERVER_PORT 5555
 
/*
 監(jiān)聽(tīng)后,一直處于accept阻塞狀態(tài),
 直到有客戶端連接,
 當(dāng)客戶端如數(shù)quit后,斷開(kāi)與客戶端的連接
 */
 
int main()
{
 //調(diào)用socket函數(shù)返回的文件描述符
	int serverSocket;
 //聲明兩個(gè)套接字sockaddr_in結(jié)構(gòu)體變量,分別表示客戶端和服務(wù)器
	struct sockaddr_in server_addr;
	struct sockaddr_in clientAddr;
	int addr_len = sizeof(clientAddr);
	int client;
	char buffer[200];
	int iDataNum;
 
 //socket函數(shù),失敗返回-1
 //int socket(int domain, int type, int protocol);
 //第一個(gè)參數(shù)表示使用的地址類型,一般都是ipv4,AF_INET
 //第二個(gè)參數(shù)表示套接字類型:tcp:面向連接的穩(wěn)定數(shù)據(jù)傳輸SOCK_STREAM
 //第三個(gè)參數(shù)設(shè)置為0
	if((serverSocket = socket(AF_INET, SOCK_STREAM, 0)) < 0)
	{
		perror("socket");
		return 1;
	}
 
	bzero(&server_addr, sizeof(server_addr));
 //初始化服務(wù)器端的套接字,并用htons和htonl將端口和地址轉(zhuǎn)成網(wǎng)絡(luò)字節(jié)序
	server_addr.sin_family = AF_INET;
	server_addr.sin_port = htons(SERVER_PORT);
 //ip可是是本服務(wù)器的ip,也可以用宏INADDR_ANY代替,代表0.0.0.0,表明所有地址
	server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
 //對(duì)于bind,accept之類的函數(shù),里面套接字參數(shù)都是需要強(qiáng)制轉(zhuǎn)換成(struct sockaddr *)
 //bind三個(gè)參數(shù):服務(wù)器端的套接字的文件描述符,
 if(bind(serverSocket, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
	{
		perror("connect");
		return 1;
	}
 //設(shè)置服務(wù)器上的socket為監(jiān)聽(tīng)狀態(tài)
	if(listen(serverSocket, 5) < 0) 
	{
		perror("listen");
		return 1;
	}
 
	while(1)
	{
		printf("Listening on port: %d\n", SERVER_PORT);
 //調(diào)用accept函數(shù)后,會(huì)進(jìn)入阻塞狀態(tài)
 //accept返回一個(gè)套接字的文件描述符,這樣服務(wù)器端便有兩個(gè)套接字的文件描述符,
 //serverSocket和client。
 //serverSocket仍然繼續(xù)在監(jiān)聽(tīng)狀態(tài),client則負(fù)責(zé)接收和發(fā)送數(shù)據(jù)
 //clientAddr是一個(gè)傳出參數(shù),accept返回時(shí),傳出客戶端的地址和端口號(hào)
 //addr_len是一個(gè)傳入-傳出參數(shù),傳入的是調(diào)用者提供的緩沖區(qū)的clientAddr的長(zhǎng)度,以避免緩沖區(qū)溢出。
 //傳出的是客戶端地址結(jié)構(gòu)體的實(shí)際長(zhǎng)度。
 //出錯(cuò)返回-1
		client = accept(serverSocket, (struct sockaddr*)&clientAddr, (socklen_t*)&addr_len);
		if(client < 0)
		{
			perror("accept");
			continue;
		}
		printf("\nrecv client data...n");
 //inet_ntoa ip地址轉(zhuǎn)換函數(shù),將網(wǎng)絡(luò)字節(jié)序IP轉(zhuǎn)換為點(diǎn)分十進(jìn)制IP
 //表達(dá)式:char *inet_ntoa (struct in_addr);
		printf("IP is %s\n", inet_ntoa(clientAddr.sin_addr));
		printf("Port is %d\n", htons(clientAddr.sin_port));
		while(1)
		{
			iDataNum = recv(client, buffer, 1024, 0);
			if(iDataNum < 0)
			{
				perror("recv");
				continue;
			}
			buffer[iDataNum] = '\0';
			if(strcmp(buffer, "quit") == 0)
				break;
			printf("%drecv data is %s\n", iDataNum, buffer);
			send(client, buffer, iDataNum, 0);
		}
	}
	return 0;
}

/*socket tcp客戶端*/
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
 
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
 
#define SERVER_PORT 5555
 
/*
 連接到服務(wù)器后,會(huì)不停循環(huán),等待輸入,
 輸入quit后,斷開(kāi)與服務(wù)器的連接
 */
 
int main()
{
 //客戶端只需要一個(gè)套接字文件描述符,用于和服務(wù)器通信
	int clientSocket;
 //描述服務(wù)器的socket
	struct sockaddr_in serverAddr;
	char sendbuf[200];
	char recvbuf[200];
	int iDataNum;
	if((clientSocket = socket(AF_INET, SOCK_STREAM, 0)) < 0)
	{
		perror("socket");
		return 1;
	}
 
	serverAddr.sin_family = AF_INET;
	serverAddr.sin_port = htons(SERVER_PORT);
 //指定服務(wù)器端的ip,本地測(cè)試:127.0.0.1
 //inet_addr()函數(shù),將點(diǎn)分十進(jìn)制IP轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)序IP
	serverAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
	if(connect(clientSocket, (struct sockaddr *)&serverAddr, sizeof(serverAddr)) < 0)
	{
		perror("connect");
		return 1;
	}
 
	printf("connect with destination host...\n");
 
	while(1)
	{
		printf("Input your world:>");
		scanf("%s", sendbuf);
		printf("\n");
 
		send(clientSocket, sendbuf, strlen(sendbuf), 0);
		if(strcmp(sendbuf, "quit") == 0)
			break;
		iDataNum = recv(clientSocket, recvbuf, 200, 0);
		recvbuf[iDataNum] = '\0';
		printf("recv data of my world is: %s\n", recvbuf);
	}
	close(clientSocket);
	return 0;
}

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Qt利用ffmpeg實(shí)現(xiàn)音視頻同步

    Qt利用ffmpeg實(shí)現(xiàn)音視頻同步

    這篇文章主要為大家詳細(xì)介紹了Qt如何利用ffmpeg實(shí)現(xiàn)音視頻同步的功能,文中的示例代碼講解詳細(xì),對(duì)大家深入了解Qt有一定的幫助,需要的可以參考一下
    2023-01-01
  • Linux下控制(統(tǒng)計(jì))文件的生成的C代碼實(shí)現(xiàn)

    Linux下控制(統(tǒng)計(jì))文件的生成的C代碼實(shí)現(xiàn)

    這篇文章主要介紹了Linux下控制(統(tǒng)計(jì))文件的生成的C代碼實(shí)現(xiàn),感興趣的小伙伴們可以參考一下
    2016-01-01
  • C++利用隨機(jī)策略實(shí)現(xiàn)優(yōu)化二叉樹(shù)操作效率

    C++利用隨機(jī)策略實(shí)現(xiàn)優(yōu)化二叉樹(shù)操作效率

    這篇文章中我們主要來(lái)詳細(xì)探討隨機(jī)化二叉搜索樹(shù)的基本思想、實(shí)現(xiàn)方法,以及如何在C++中應(yīng)用這些策略來(lái)優(yōu)化我們的數(shù)據(jù)結(jié)構(gòu),感興趣的可以了解下
    2024-02-02
  • 深入理解C++11:探索lambda函數(shù)的奧秘

    深入理解C++11:探索lambda函數(shù)的奧秘

    聚焦于C++11,讓我們一起探索lambda函數(shù)的奧秘,本指南將帶您深入了解這個(gè)強(qiáng)大的編程工具,讓您在編程世界中如虎添翼,無(wú)論您是初學(xué)者還是有經(jīng)驗(yàn)的開(kāi)發(fā)者,本指南都將為您帶來(lái)全新的視角和實(shí)用的技巧,需要的朋友可以參考下
    2024-01-01
  • C++采用ring3讀取MBR實(shí)例

    C++采用ring3讀取MBR實(shí)例

    這篇文章主要介紹了C++采用ring3讀取MBR實(shí)例,可實(shí)現(xiàn)對(duì)硬盤的主引導(dǎo)記錄的讀取,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2014-10-10
  • 詳解設(shè)計(jì)模式中的中介者模式在C++編程中的運(yùn)用

    詳解設(shè)計(jì)模式中的中介者模式在C++編程中的運(yùn)用

    這篇文章主要介紹了設(shè)計(jì)模式中的中介者模式在C++編程中的運(yùn)用,中介者模式將對(duì)象間的通信封裝到一個(gè)類中,將多對(duì)多的通信轉(zhuǎn)化為一對(duì)多的通信,降低了系統(tǒng)的復(fù)雜性,需要的朋友可以參考下
    2016-03-03
  • C++中Boost的智能指針shared_ptr

    C++中Boost的智能指針shared_ptr

    這篇文章介紹了C++中Boost的智能指針shared_ptr,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-07-07
  • C++ 智能指針代碼解析

    C++ 智能指針代碼解析

    這篇文章主要介紹了c++ 智能指針基礎(chǔ)的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)使用c++,感興趣的朋友可以了解下,希望能給你帶來(lái)幫助
    2021-10-10
  • C++?OpenCV紅綠燈檢測(cè)Demo實(shí)現(xiàn)詳解

    C++?OpenCV紅綠燈檢測(cè)Demo實(shí)現(xiàn)詳解

    OpenCV(Open Source Computer Vision Library)是開(kāi)源的計(jì)算機(jī)視覺(jué)和機(jī)器學(xué)習(xí)庫(kù),提供了C++、 C、 Python、 Java接口,并支持Windows、 Linux、 Android、 Mac OS平臺(tái),下面這篇文章主要給大家介紹了關(guān)于C++?OpenCV紅綠燈檢測(cè)Demo實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下
    2022-11-11
  • C++實(shí)現(xiàn)智能柜管理系統(tǒng)

    C++實(shí)現(xiàn)智能柜管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)智能柜管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03

最新評(píng)論