windows環境下C語言socket編程

最近由于實驗需要,要求寫一個c程序與java程序通信的軟件,為了測試首先寫了一個windows環境下c語言的socket(tcp)通信程序。

首先socket通信的步驟:


?? 圖一???? socket通信步驟(轉載)


??????????????????????????????????????????? 圖二 三次握手協議(轉載)


? 圖三??? 三次握手協議(轉載)

代碼? 服務器端

??? #include <winsock2.h>
??? #include <windows.h>
??? #pragma comment(lib,"ws2_32.lib")
??? ?
??? int main(int argc, char* argv[])
??? {
?? ??? ?//初始化WSA
?? ??? ?WORD sockVersion = MAKEWORD(2, 2);
?? ??? ?WSADATA wsaData;
?? ??? ?if (WSAStartup(sockVersion, &wsaData) != 0)
?? ??? ?{
?? ??? ??? ?return 0;
?? ??? ?}
??? ?
?? ??? ?//創建套接字
?? ??? ?SOCKET slisten = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
?? ??? ?if (slisten == INVALID_SOCKET)
?? ??? ?{
?? ??? ??? ?printf("socket error !");
?? ??? ??? ?return 0;
?? ??? ?}
??? ?
?? ??? ?//綁定IP和端口
?? ??? ?struct sockaddr_in sin;
?? ??? ?sin.sin_family = AF_INET;
?? ??? ?sin.sin_port = htons(8888);
?? ??? ?sin.sin_addr.S_un.S_addr = INADDR_ANY;
?? ??? ?if (bind(slisten, (LPSOCKADDR)&sin, sizeof(sin)) == SOCKET_ERROR)
?? ??? ?{
?? ??? ??? ?printf("bind error !");
?? ??? ?}
??? ?
?? ??? ?//開始監聽
?? ??? ?if (listen(slisten, 5) == SOCKET_ERROR)
?? ??? ?{
?? ??? ??? ?printf("listen error !");
?? ??? ??? ?return 0;
?? ??? ?}
??? ?
?? ??? ?//循環接收數據
?? ??? ?SOCKET sClient;
?? ??? ?struct sockaddr_in remoteAddr;
?? ??? ?int nAddrlen = sizeof(remoteAddr);
?? ??? ?//char revData[255];
?? ??? ?printf("等待連接...\n");
?? ??? ?sClient = accept(slisten, (SOCKADDR *)&remoteAddr, &nAddrlen);
?? ??? ?while (1)
?? ??? ?{
?? ??? ??? ?char revData[255];
?? ??? ??? ?//printf("等待連接...\n");
?? ??? ??? ?//sClient = accept(slisten, (SOCKADDR *)&remoteAddr, &nAddrlen);
?? ??? ??? ?if (sClient == INVALID_SOCKET)
?? ??? ??? ?{
?? ??? ??? ??? ?printf("accept error !");
?? ??? ??? ??? ?continue;
?? ??? ??? ?}
?? ??? ??? ?printf("接受到一個連接:%s \r\n", inet_ntoa(remoteAddr.sin_addr));
??? ?
?? ??? ??? ?//接收數據
?? ??? ??? ?int ret = recv(sClient, revData, 255, 0);
?? ??? ??? ?//printf(ret);
?? ??? ??? ?if (ret > 0)
?? ??? ??? ?{
?? ??? ??? ??? ?revData[ret] = 0x00;
?? ??? ??? ??? ?printf(revData);
?? ??? ??? ?}
?? ??? ??? ?//發送數據
?? ??? ??? ?char * sendData = "你好,TCP客戶端!\n";
?? ??? ??? ?send(sClient, sendData, strlen(sendData), 0);
?? ??? ?}
?? ??? ??? ?closesocket(sClient);
?? ??? ?closesocket(slisten);
?? ??? ?WSACleanup();
?? ??? ?return 0;
??? }

代碼?? 客戶端

??? #include <winsock2.h>
??? #include <windows.h>
??? #pragma comment(lib,"ws2_32.lib")
??? #include <STDIO.H>
??? int main(int argc, char* argv[])
??? {
?? ??? ?WORD sockVersion = MAKEWORD(2, 2);
?? ??? ?WSADATA data;
?? ??? ?if (WSAStartup(sockVersion, &data) != 0)
?? ??? ?{
?? ??? ??? ?return 0;
?? ??? ?}
??? ?
?? ??? ?SOCKET sclient = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
?? ??? ?if (sclient == INVALID_SOCKET)
?? ??? ?{
?? ??? ??? ?printf("invalid socket !");
?? ??? ??? ?return 0;
?? ??? ?}
??? ?
?? ??? ?struct sockaddr_in serAddr;
?? ??? ?serAddr.sin_family = AF_INET;
?? ??? ?serAddr.sin_port = htons(8888);
?? ??? ?serAddr.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
?? ??? ?if (connect(sclient, (struct sockaddr *)&serAddr, sizeof(serAddr)) == SOCKET_ERROR)
?? ??? ?{
?? ??? ??? ?printf("connect error !");
?? ??? ??? ?closesocket(sclient);
?? ??? ??? ?return 0;
?? ??? ?}
?? ??? ?for (int i = 0; i < 1000; i++)
?? ??? ?{
??? ?
?? ??? ??? ?char * sendData = "csi接收? 二進制文件未解析\n";
?? ??? ??? ?send(sclient, sendData, strlen(sendData), 0);
?? ??? ??? ?char recData[255];
?? ??? ??? ?/*int ret = recv(sclient, recData, 255, 0);
?? ??? ??? ?if (ret > 0)
?? ??? ??? ?{
?? ??? ?
?? ??? ??? ??? ?recData[ret] = 0x00;
?? ??? ??? ??? ?//printf(recData);
?? ??? ??? ??? ?printf("%d%s",i,recData);
?? ??? ??? ?}
?? ??? ??? ?*/
?? ??? ?}
?? ??? ?closesocket(sclient);
?? ??? ?WSACleanup();
?? ??? ?return 0;
??? }

后續會給出c程序與java程序通信的例子。


?

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

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

相關文章

進程控制塊(PCB)

進程控制塊PCB 我們知道&#xff0c;每個進程在內核中都有一個進程控制塊&#xff08;PCB&#xff09;來維護進程相關的信息&#xff0c;Linux內核的進程控制塊是task_struct結構體。 /usr/src/linux-headers-3.16.0-30/include/linux/sched.h文件中可以查看struct task_struct…

網絡層攻擊防御

網絡層攻擊防御 網絡層攻擊防御主要分為以下三類&#xff1a; TCP類報文攻擊防御 UDP類報文攻擊防御 ICMP類報文攻擊防御 TCP類報文攻擊防御 TCP正常的交互過程&#xff1a; 圖&#xff1a;TCP正常交互過程 在TCP/IP協議中&#xff0c;TCP協議提供可靠的連接服務&#xff0c…

Linux之環境變量

常見環境變量 按照慣例&#xff0c;環境變量字符串都是namevalue這樣的形式&#xff0c;大多數name由大寫字母加下劃線組成&#xff0c;一般把name的部分叫做環境變量&#xff0c;value的部分則是環境變量的值。環境變量定義了進程的運行環境&#xff0c;一些比較重要的環境變量…

環境變量操作函數

getenv獲取環境變量值的函數&#xff1b; setenv改變或者添加環境變量函數&#xff1b; unsetenv 取消環境變量&#xff1b; &#xff08;可以在終端上man 函數名來獲取詳細的函數信息&#xff09; #include<stdio.h> #include<stdlib.h> #include<string.h>…

Makefile(三)

在平時使用中&#xff0c;可以使用以下的makefile來編譯單獨的代碼 src $(wildcard *.c) obj $(patsubst %.c, %.o, $(src))CC gcc CFLAGS -Wall -gall:$(target)$(target):%:%.c$(CC) $< -o $ $(CFLAGS).PHONY: clean all clean:-rm -rf $(target) 使用方法就是make 后…

位運算(C++)

C輸出十六進制 #include<iostream> #include<iomanip> using namespace std;int main() {int a 60;int b 13;int c a &b;cout << "a : hex "<<hex << a << endl;cout << "b : hex "<<hex <<…

數學函數(C/C++)

C中包含頭文件<math.h> C包含頭文件<cmath> 函數 double cos&#xff08;double&#xff09; 該函數返回弧度角&#xff08;double型&#xff09;的余弦 double tan&#xff08;double&#xff09; 該函數返回弧度角&#xff08;double型&#xff09;的正切…

數據類型(C++)

不同系統會有不同差異&#xff1a; 類型 位(byte) 范圍 char 1 -128—127 or 0 – 255 unsigned char 1 0 – 255 signed int 1 -128—127 int 4 -2^31 – 2^32-1 unsigned int 4 0 – 2^32 signed int 4 -2^31 – 2^32-1 short int 2 2^15 – 2^15-1 …

日期與時間(C/C++)

C繼承了C語言用于日期和時間操作的結構和函數&#xff0c;使用之前程序要引用<ctime>頭文件 有四個與時間相關的類型:clock_t、time_t、size_t、和tm。類型clock_t、size_t、和time_t能夠把系統時間和日期表示為某種整數。 結構體tm把時間和日期以C結構的形式保存&#x…

標準輸入輸出(C++)

輸入輸出流函數&#xff08;模板&#xff09; #include<iostream> #include<iomanip> using namespace std; int main() {cout << setiosflags(ios::left|ios::showpoint); //設左對齊cout.precision(5); //設置除小數…

拷貝函數和構造函數

類的構造函數&#xff1a; 類的構造函數是類的一種特殊的成員函數&#xff0c;它會在每次創建類的新對象時執行。 構造函數的名稱與類的名稱是完全相同的&#xff0c;并且不會返回任何類型&#xff0c;也不會返回void。 構造函數可用于為某些成員變量初始值。 默認的構造函數是…

拷貝構造函數

拷貝構造函數是一種特殊的構造函數&#xff0c;它在創建對象時&#xff0c;使用的是同一類中之前創建的對象來初始化新創建的對象。拷貝構造函數通常用于&#xff1a; l 通過使用另一個同類型的對象來初始化新創建的對象&#xff1b; l 復制對象把它作為參數傳遞給函數&#…

Linux進程通信之管道

進程間完成數據傳遞需要借助操作系統提供的特殊的方法&#xff0c;比如&#xff1a;文件、管道、信號、共享內存、消息隊列、套接字、命名管道等。但現在常用的進程間通信方式有&#xff1a; 管道 – 使用最簡單 pipe 管道一般讀寫行為 FIFO&#xff08;有名管道&#xff09;&a…

Linux進程通信之文件

父子進程共享打開的文件描述符------使用文件完成進程間通信. /*** fork_share_fd.c***/ #include <stdio.h> #include <unistd.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/wait.h>int main(void) {in…

dup2函數

將當前系統中的進程信息打印到文件中 命令行&#xff1a;ps aux > out 將ps得到的信息重定向到out文件中 使用dup2文件在程序中完成。 int dup2(int oldfd,int newfd); /*** dup2.c ***/ #include<stdio.h> #include<fcntl.h> #include<unistd.h> #includ…

wait()函數

wait()函數&#xff1a;回收僵尸進程 父進程調用wait函數可以回收子進程終止信息。該函數有三個功能&#xff1a; 1&#xff09; 阻塞等待子進程退出 2&#xff09; 回收子進程殘留資源 3&#xff09; 獲取子進程結束狀態&#xff08;退出原因&#xff09; /*** zoom_test.c **…

waitpid()函數

waitpid函數 作用同于wait&#xff0c;但可指定pid進程清理&#xff0c;可以不阻塞。 pid_t waitpid(pid_t pid,int *status,int options);成功&#xff1a;返回清理掉的子進程ID&#xff1b;失敗&#xff1a;-1&#xff08;無子進程&#xff09; 特殊參數和返回情況&#xff1…

孤兒進程、僵尸進程

孤兒進程&#xff1a;父進程先于子進程結束&#xff0c;則子進程成為孤兒進程&#xff0c;子進程的父進程成為init進程&#xff0c;稱為init進程領養孤兒進程。 /*** orphan.c ***/ #include <stdio.h> #include <unistd.h> #include <sys/wait.h>int main(v…

友元函數

類的友元函數是定義在類外部&#xff0c;但有權訪問類的所有私有成員和保護成員。盡管友元函數的原型有在類的定義中出現過&#xff0c;但友元函數并不是成員函數。 友元可以是一個函數&#xff0c;該函數被稱為友元函數&#xff1b;友元也可以是一個類&#xff0c;該類被稱為友…

this指針

在C中&#xff0c;每一個對象都能夠通過this指針來訪問自己的地址。this指針是所有成員函數的隱含參數。因此&#xff0c;在成員函數內部&#xff0c;它可以用來指向調用對象。 友元函數是沒有this指針的&#xff0c;因為友元不是類的成員&#xff0c;只有成員函數才有this指針…