TCP/UDP网络编程的基础知识与基本示例(windows和Linux)
一、TCP编程的一般步骤
服务器端:
1、创建一个socket,用函数socket()
2、绑定IP地址、端口等信息到socket上,用函数bind()
3、开启监听,用函数listen()
4、接收客户端上来的连接,用函数accept()
5、收发数据,用函数send()和recv(),或者read()和write()
6、关闭网络连接
7、关闭监听
客户端:
1、创建一个socket,用函数socket()
2、设置要连接的对方IP地址和端口等属性
3、连接服务器,用函数connect()
4、收发数据,用函数send()和recv(),或者read()和write()
5、关闭网络连接
以下是TCP通信的时序图:

二、UDP编程的一般步骤
服务器:
1、创建一个socket,用函数socket()
2、绑定IP地址、端口等信息到socket上,用函数bind()
3、循环接收数据,用函数recvfrom()
4、关闭网络连接
客户端:
1、创建一个socket,用函数socket()
2、设置对方的IP地址、端口等属性
3、发送数据,用函数sendto()
4、关闭网络连接
以下是UDP通信的时序图:

三、Windows socket和Linux socket编程的区别
1、头文件
Windows下winsocket.h/winsocket2.h
Winsocket2.0需要ws2_32.lib和ws2_32.dll
Linux下sys/socket.h
错误处理:errno.h
2、初始化
Windows下需要WSAStartup
Linux不需要
3、关闭socket
Windows下closesocket
Linux下close
4、类型
Windows下SOCKET
Linux下int
#ifdef WIN32
typedef int socklen_t;
typedef int ssize_t;
#endif
#ifdef __Linux__
typedef int SOCKET;
typedef unsigned char BYTE;
typedef unsigned long DWORD;
#define FALSE 0
#define SOCKET_ERROR (-1)
#endif
5、获取错误码
Windows下getlasterror()/WSAGetLastError()
Linux下errno变量
6、设置非阻塞
Windows下ioctlsocket()
Linux下fcntl() <fcntl.h>
7、send函数的最后一个参数
Windows下一般设置为0
Linux下最好设置为MSG_NOSIGNAL,如果不设置,在发送出错后有可 能会导致程序退出。
8、毫秒级时间获取
Windows下GetTickCount()
Linux下gettimeofday()
9、后续用到会再补充!!!
四、网络编程示例
1、Windows下TCP编程
服务器:
#include "stdafx.h"
#include <WinSock2.h>
#include <iostream>
using namespace std;
#include <stdio.h>
int _tmain(int argc, _TCHAR* argv[])
{
WORD wVersionRequested = MAKEWORD(2,2);//确定最高可使用的socket版本
WSADATA wsaData;//用来存储winsocket初始化信息
int err = WSAStartup(wVersionRequested, &wsaData);//初始化使用winsocket dll
if (err != 0)
{
cout<<"WSAStartup failed!"<<endl;
}
//步骤一:创建服务器socket
//第一个参数选择协议族,第二个选择数据流类型,第三个选择协议类型
SOCKET socketSrv = socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
if (socketSrv == INVALID_SOCKET)
{
cout<<"Create socket failed"<<endl;
}
//步骤二:绑定端口、IP信息到socket上
SOCKADDR_IN addrSrv; //这个结构用来存储本地端和远程端用来连接socket的信息
//使用系统指定的IP地址。htonl将一个无符号长整形主机字节序转化为一个TCP/IP网络字节序(大端)
addrSrv.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
addrSrv.sin_family = AF_INET;
addrSrv.sin_port = htons(8888);//htons将一个无符号短整型端口主机字节序转化为一个TCP/IP网络字节序(大端)
if (int n = bind(socketSrv,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR)))
{
cout<<"bind server socket failed"<<endl;
}
//步骤三:监听
if (listen(socketSrv,SOMAXCONN))//第二个参数为可接受的最大连接请求数
{
cout<<"listen failed"<<endl;
}
//步骤四:接收客户端连接
SOCKADDR_IN addrClient;//用来存放客户连接信息
int nLen = sizeof(SOCKADDR);
char buf1[255] = "";//存放连接的客户IP和端口信息字符串
char buf2[255] = "";//接收和发送缓冲数据
char bufRecv[255] = "";
//这个nlen没有设初值也会出错的哦
SOCKET socketCon = accept(socketSrv,(SOCKADDR*)&addrClient,&nLen);//后面改成多线程后就能连接多人对话
while (true)
{
//inet_ntoa将IPv4网络地址转换成标准的点治式字符串(例如:192.168.1.1)
//ntohs将无符号短整型网络字节序转换为无符号短整型主机字节序,与上面的htons相反
sprintf(buf1,"***IP:%s PORT:%d #\n",inet_ntoa(addrClient.sin_addr),ntohs(addrClient.sin_port));
//步骤五:接收和传送数据
int nBytesRecv = recv(socketCon,bufRecv,sizeof(bufRecv),0);
if (strcmp(bufRecv,"exit") == 0)
{
//closesocket(socketCon);
break;
}
strcat(buf1,bufRecv);
cout<<buf1<<endl;
cout<<"***YOU#:"<<endl;
//cin>>buf2;
cin.getline(buf2,sizeof(buf2));
int nBytesSent = send(socketCon,buf2,sizeof(buf1),0);
}
//步骤六:关闭连接
closesocket(socketCon);
closesocket(socketSrv);
WSACleanup();//终止使用winsocket dll
return 0;
}
客户端:
#include "stdafx.h"
#include <WinSock2.h>
#include <iostream>
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{
WORD wVersionRequested = MAKEWORD(2,2);
WSADATA wsaData;
if (WSAStartup(wVersionRequested,&wsaData))
{
cout<<"WSAStartup failed"<<endl;
}
//步骤一:创建socket
SOCKET socktClient = socket(AF_INET, SOCK_STREAM,IPPROTO_TCP);
//步骤二:设置要连接的对方的IP、端口等信息
SOCKADDR_IN addrClient;
//inet_addr将一个点制式IP字符串转换为适合IN_ADDR结构的形式
addrClient.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
addrClient.sin_family = AF_INET;
addrClient.sin_port = htons(8888);
//步骤三:连接服务器
if (connect(socktClient,(SOCKADDR*)&addrClient,sizeof(SOCKADDR)))
{
cout<<"connect failed"<<endl;
}
char buf[255] = "";
while (true)
{
//步骤四:收发数据
cout<<"***YOU#"<<endl;
//cin>>buf;
cin.getline(buf,sizeof(buf));
if (strcmp(buf,"exit") == 0)
{
break;
}
int nBytesSent = send(socktClient,buf,sizeof(buf),0);
int nBytesRecv = recv(socktClient,buf,sizeof(buf),0);
cout<<"***Server#"<<endl;
cout<<buf<<endl;
}
//步骤五:关闭连接
closesocket(socktClient);
return 0;
}
2、WIndows下UDP编程
服务器:
#include "stdafx.h"
#include <WinSock2.h>
#include <iostream>
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{
WORD wVersionRequested = MAKEWORD(2,2);
WSADATA wsaData;
WSAStartup(wVersionRequested,&wsaData);
//步骤一:创建socket
SOCKET socketSrv = socket(AF_INET,SOCK_DGRAM,IPPROTO_UDP);//这里换成了非面向连接的数据报模式
SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
addrSrv.sin_family = AF_INET;
addrSrv.sin_port = htons(8000);
//步骤二:绑定端口、IP等信息到socket上
if (bind(socketSrv,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR)))
{
cout<<"bind failed"<<endl;
}
char buf[255] = "";
while (true)
{
SOCKADDR_IN addrClient;//用来存放客户连接信息
int nLen = sizeof(SOCKADDR);
//步骤三:接收数据
int nBytesRecv = recvfrom(socketSrv,buf,sizeof(buf),0,(SOCKADDR*)&addrClient,&nLen);//这里不用变量存储会报错
if (strcmp(buf,"exit") == 0)
{
//closesocket(addrClient);
continue;
}
cout<<"Client#"<<endl;
cout<<buf<<endl;
}
//步骤四:关闭网络连
closesocket(socketSrv);
WSACleanup();
return 0;
}
客户端:
#include "stdafx.h"
#include <WinSock2.h>
#include <iostream>
using namespace std;
int _tmain(int argc, _TCHAR* argv[])
{
WORD wVersionRequested = MAKEWORD(2,2);
WSADATA wsaData;
WSAStartup(wVersionRequested,&wsaData);
//步骤一:创建socket
SOCKET socketClient = socket(AF_INET,SOCK_DGRAM,IPPROTO_UDP);//这里换成了非面向连接的数据报模式
//步骤二:设置要数据接收者的地址信息
SOCKADDR_IN addrTo;
addrTo.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
addrTo.sin_family = AF_INET;
addrTo.sin_port = htons(8000);
char buf[255] = "";
while (true)
{
cout<<"YOU#"<<endl;
cin.getline(buf,sizeof(buf));
if (strcmp(buf,"exit") == 0)
{
break;
}
//步骤三:发送数据
int nBytesSent = sendto(socketClient, buf,strlen(buf)+1,0,(SOCKADDR*)&addrTo,sizeof(SOCKADDR));
}
//步骤四:关闭网络连
closesocket(socketClient);
WSACleanup();
return 0;
}
3、Linux下TCP编程
服务器:
客户端:
2、Linux下UDP编程
服务器:
客户端:
TCP/UDP网络编程的基础知识与基本示例(windows和Linux)的更多相关文章
- python六十七课——网络编程(基础知识了解)
网络编程: 什么是网络编程? 网络:它是一种隐形的媒介:可以将多台计算机使用(将它们连接到一起) 网络编程:将多台计算机之间可以相互通信了(做数据交互) 一旦涉及到网络编程,划分为两个方向存在,一方我 ...
- java第九节 网络编程的基础知识
/** * * 网络编程的基础知识 * 网络协议与TCP/IP * IP地址和Port(端口号) * 本地回路的IP地址:127.0.0.1 * 端口号的范围为0-65535之间,0-1023之间的端 ...
- 01网络编程(基础知识+OSI七层协议+TCP与UDP)
目录 01 网络编程 一.软件开发架构 1.1 CS架构 1.2 BS架构 二.网络理论前戏 2.1 简介 2.2 常见硬件 三.OSI七层协议(五层) 3.1 七层协议 3.2 五层协议 3.3 知 ...
- TCP/IP网络编程技术基础
零零碎碎记下点→ 不对的欢迎大家批评纠正→ 以免本人及偶尔看到此博客的人继续迷途未返→ >>>>>基础知识→ 1→TCP/IP英文名:Tranmission Contro ...
- 自测之Lesson15:TCP&UDP网络编程
题目:编写一个TCP通信的程序. 实现代码: #include <stdio.h> #include <sys/socket.h> #include <unistd.h& ...
- 《TCP/IP网络编程》
<TCP/IP网络编程> 基本信息 作者: (韩)尹圣雨 译者: 金国哲 丛书名: 图灵程序设计丛书 出版社:人民邮电出版社 ISBN:9787115358851 上架时间:2014-6- ...
- 【RL-TCPnet网络教程】第16章 UDP用户数据报协议基础知识
第16章 UDP用户数据报协议基础知识 本章节为大家讲解UDP(User Datagram Protocol,用户数据报协议),需要大家对UDP有个基础的认识,方便后面章节UDP实战操作. ...
- 三十天学不会TCP,UDP/IP编程--MAC地址和数据链路层
这篇文章主要是来做(da)推(guang)介(gao)的!由于这两年接触到了比较多的这方面的知识,不想忘了,我决定把他们记录下来,所以决定在GitBook用半年时间上面写下来,这是目前写的一节,后面会 ...
- python网络编程——socket基础篇
python的网络编程比c语言简单许多, 封装许多底层的实现细节, 方便程序员使用的同时, 也使程序员比较难了解一些底层的东西. 1 TCP/IP 要想理解socket,首先得熟悉一下TCP/IP协议 ...
随机推荐
- python之pandas模块
一.pandas模块是基于Numpy模块的,pandas的主要数据结构是Series和DadaFrame,下面引入这样的约定: from pandas import Series,DataFrame ...
- android 利用Bitmap获取圆角矩形、圆形图片
1.在很多时候,我们要显示图片资源,需要将他的资源显示为圆角的:示例源码如下: public static Bitmap getRoundedCornerBitmap(Bitmap bitmap,fl ...
- 九章算法系列(#3 Binary Tree & Divide Conquer)-课堂笔记
前言 第一天的算法都还没有缓过来,直接就进入了第二天的算法学习.前一天一直在整理Binary Search的笔记,也没有提前预习一下,好在Binary Tree算是自己最熟的地方了吧(LeetCode ...
- INI文件格式
最近在看git命令,遇到INI文件格式,上网查了一下,把它总结一下: 程序没有任何配置文件,那么它对外是全封闭的,一旦程序需要修改一些参数必须要修改程序代码本身并重新编译,为了让程序出厂后还能根据需要 ...
- 什么是Code Review
Code Review 是一种通过复查代码提高代码质量的过程,在XP方法中占有极为重要的地位,也已经成为软件工程中一个不可缺少的环节. 本文通过对Code Review的一些概念和经验的探讨,就如何进 ...
- Effective Java实作equals() - 就是爱Java
equals()这个方法,定义在Object class中,这个是所有class的base class,因此所有的class都继承这个方法,默认是比较内存地址,不过Mix需要的是商业规则上的比较,所以 ...
- Keil C动态内存管理机制分析及改进
Keil C是常用的嵌入式系统编程工具,它通过init_mempool.mallloe.free等函数,提供了动态存储管理等功能.本文通过对init_mempool.mallloe和free这3个Ke ...
- C++ crash 堆栈信息获取(三篇文章)
最近在做程序异常时堆栈信息获取相关工作,上一篇文章成功的在程序creash时写下了dump文件,而有些情况写dump文件是 不可以的,比如在jni开发时,C++只做底层处理,而整个项目是android ...
- ActiveX in QT
http://doc.qt.io/qt-4.8/activeqt.htmlhttp://doc.qt.io/qt-5/activeqt-index.html
- [科普]MinGW vs MinGW-W64及其它
转载:http://tieba.baidu.com/p/3186234212?pid=54372018139&cid=#54372018139 这里也转一下吧. 部分参照备忘录原文: bitb ...