socket 网络编程高速入门(一)教你编写基于UDP/TCP的服务(client)通信
由于UNIX和Win的socket大同小异,为了方便和大众化,这里先介绍Winsock编程。
socket 网络编程的难点在入门的时候就是对基本函数的了解和使用,由于这些函数的结构往往比較复杂,參数大部分都是结构体,令人难以记忆和理解。
可是一旦我们知道这些函数包含其參数的详细含义,socket网络编程也就变得不是那么复杂。
这里不赘述 详细函数的详细含义。网络上有非常多的文章。同一时候笔者建议大家參考
MSDN。对返回值,參数等会有更好的理解。
下面均为单线程的简单实例,多线程的请关注下一篇文章。
(一)
UDP:中文名是用户数据报协议。是OSI參考模型中的一种无连接的传输层协议。提供面向事物的简单不可靠信息传送服务,在网络良好的情况下也可能会出现丢包现象。
UDP通讯流程:
(Winsock的初始化:WSAStartup) //windows特有的
Server端:
socket:建立套接字
bind:公布port(netstat -an可见已经公布的port)
sendto/recv/recvfrom:收发数据
Client端:
socket:建立套接字
bind:公布port(netstat -an可见已经公布的port)
sendto/recv/recvfrom:收发数据
其它函数详见凝视。
UDP--Client
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <iostream>
#include <string>
#include <winsock2.h>
#pragma comment(lib,"ws2_32.lib") //编译器设置--链接库
using namespace std;
const int PORT = 8009; int main()
{
int n;
WSADATA wd;
n = WSAStartup(MAKEWORD(2, 2), &wd);
if (n)
{
cout << "WSAStartup函数错误!" << endl;
return -1;
}
SOCKET sock = socket(AF_INET, SOCK_DGRAM, 0); //SOCK_DGRAM代表UDP
if (INVALID_SOCKET == sock)
{
cout << "socket建立失败。" << endl;
return -1;
}
sockaddr_in sa = { AF_INET }; //随机分配一个port (仅初始化一个值,让client自己分配port)
n = bind(sock, (sockaddr*)&sa, sizeof(sa));
if (n == SOCKET_ERROR)
{
cout << "bind函数失败。" << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
sockaddr_in sa1 = { AF_INET, htons(8009) }; //必须使用htons。由于网络字节序的缘故
sa1.sin_addr.S_un.S_addr = inet_addr("192.168.253.1");//指定要发往的IP和port号 char s[256];
while (true)
{
fflush(stdin);
gets(s);
sendto(sock, s, strlen(s), 0,(sockaddr*)&sa1,sizeof(sa1)); //发送数据
}
return 0;
}
UDP-Server
#include <cstdio>
#include <iostream>
#include <string>
#include <winsock2.h>
#pragma comment(lib,"ws2_32.lib")
using namespace std;
const int PORT = 8009;
int main()
{
int n;
WSADATA wd;
n=WSAStartup(MAKEWORD(2, 2), &wd);
if (n)
{
cout << "WSAStartup函数错误! " << endl;
return -1;
}
SOCKET sock = socket(AF_INET, SOCK_DGRAM,0);
if (INVALID_SOCKET == sock)
{
cout << "socket建立失败!" << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
sockaddr_in sa = { AF_INET, htons(PORT) };
n=bind(sock,(sockaddr*)&sa, sizeof(sa));
if (n == SOCKET_ERROR)
{
cout << "bind绑定port失败。" << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
else
{
cout << "port公布成功:" << PORT << endl;
}
char s[256];
while (true)
{
n = recv(sock, s, sizeof(s), 0); //recv返回的是实际copy的字节数
s[n] = '\0';
cout << s << endl;
} return 0; }
(二)
TCP: 是一种面向连接(连接导向)的、可靠的、基于字节流的运输层(Transport layer)通信协议。
TCP通讯流程
(Winsock的初始化:WSAStartup)
Server端:
socket:建立套接字
bind:公布port
listen:開始侦听
accept:接纳client连接(如同公司的前台)
send/recv/recvfrom:收发数据(如同公司的客户经理)
Client端:
socket:建立套接字
bind:公布port(client能够不绑定或绑定0port)
connect:连接到server
send/recv/recvfrom:收发数据(如同公司的客户经理)
一般通讯流程是按:CS两方1发1收对称的。一旦收发顺讯混乱软件就失控了。
(请求应答模式:Request/Reply)
TCP---Server
#include <cstdio>
#include <iostream>
#include <string>
#include <winsock2.h>
#pragma comment(lib,"ws2_32.lib")
using namespace std;
const int PORT = 8009;
int main()
{
int n;
WSADATA wd;
n=WSAStartup(MAKEWORD(2, 2), &wd);
if (n)
{
cout << "WSAStartup函数错误!" << endl;
return -1;
}
SOCKET sock = socket(AF_INET, SOCK_STREAM,0);
if (INVALID_SOCKET == sock)
{
cout << "socket建立失败!" << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
sockaddr_in sa = { AF_INET, htons(PORT) };
n=bind(sock,(sockaddr*)&sa, sizeof(sa));
if (n == SOCKET_ERROR)
{
cout << "bind绑定port失败。" << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
else
{
cout << "port公布成功:" << PORT << endl;
}
listen(sock, 5); //第二个參数一般设置5
int nlen = sizeof(sa);
SOCKET socka = accept(sock, (sockaddr*)&sa, &nlen); //会返回一个新的连接套接字
if (socka == INVALID_SOCKET)
{
cout << "accepth函数失败! " << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
cout << "有人连接进来:" << inet_ntoa(sa.sin_addr) << "--" << htons(sa.sin_port) << endl;//输出连接者的ip和port char s[256];
while ((n = recv(socka, s, sizeof(s)-1, 0)) > 0) //-1是留给‘\0’一个位置
{
s[n] = '\0';
cout << s << endl;
/* char k[256] = "你好啊";
if (strcmp(s,k)==0)
cout << "你好你好。大家都好哈。" << endl;*/
} return 0; }
TCP---Client
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <iostream>
#include <string>
#include <winsock2.h>
#pragma comment(lib,"ws2_32.lib")
using namespace std;
const int PORT = 8009; int main()
{
int n;
WSADATA wd;
n = WSAStartup(MAKEWORD(2, 2), &wd);
if (n)
{
cout << "WSAStartup函数错误!" << endl;
return -1;
}
SOCKET sock = socket(AF_INET, SOCK_STREAM, 0);
if (INVALID_SOCKET == sock)
{
cout << "socket建立失败!" << endl;
return -1;
}
sockaddr_in sa = { AF_INET }; //随机分配一个port
n = bind(sock, (sockaddr*)&sa, sizeof(sa));
if (n == SOCKET_ERROR)
{
cout << "bind函数失败! " << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
sa.sin_addr.S_un.S_addr = inet_addr("192.168.253.1");//115.200.33.112
sa.sin_port = htons(PORT);
n = connect(sock, (sockaddr*)&sa, sizeof(sa)); //指定port发送数据
if (n == SOCKET_ERROR)
{
cout << "connect函数失败!" << endl;
cout << "错误码是:" << WSAGetLastError() << endl;
return -1;
}
char s[256];
while (true)
{
fflush(stdin);
gets(s);
send(sock, s, strlen(s), 0);
}
return 0;
}
socket 网络编程高速入门(一)教你编写基于UDP/TCP的服务(client)通信的更多相关文章
- 网络编程基础:粘包现象、基于UDP协议的套接字
粘包现象: 如上篇博客中最后的示例,客户端有个 phone.recv(2014) , 当服务端发送给客户端的数据大于1024个字节时, 多于1024的数据就会残留在管道中,下次客户端再给服务端发命令时 ...
- Day09: socket网络编程-OSI七层协议,tcp/udp套接字,tcp粘包问题,socketserver
今日内容:socket网络编程 1.OSI七层协议 2.基于tcp协议的套接字通信 3.模拟ssh远程执行命令 4.tcp的粘包问题及解决方案 5.基于udp协议的套接字 ...
- Socket网络编程系列教程序
C语言的用途相当多,可以用在数据结构.数据库.网络.嵌入式等方面,历经40多年不衰,真是厉害!最近一直想从某一应用方面写一个系列教程,好好地把某一方面讲深讲透. 正好博主对网络方面的编 ...
- Linux Socket 网络编程
Linux下的网络编程指的是socket套接字编程,入门比较简单.在学校里学过一些皮毛,平时就是自学玩,没有见识过真正的socket编程大程序,比较遗憾.总感觉每次看的时候都有收获,但是每次看完了之后 ...
- Socket网络编程--Libev库学习(1)
这一节是安装篇. Socket网络编程不知不觉已经学了快两个月了.现在是时候找个网络库学学了.搜索了很多关于如何学网络编程的博客和问答.大致都是推荐学一个网络库,至于C++网络库有那么几个,各有各的好 ...
- Python Socket 网络编程
Socket 是进程间通信的一种方式,它与其他进程间通信的一个主要不同是:它能实现不同主机间的进程间通信,我们网络上各种各样的服务大多都是基于 Socket 来完成通信的,例如我们每天浏览网页.QQ ...
- Python全栈【Socket网络编程】
Python全栈[socket网络编程] 本章内容: Socket 基于TCP的套接字 基于UDP的套接字 TCP粘包 SocketServer 模块(ThreadingTCPServer源码剖析) ...
- python之Socket网络编程
什么是网络? 网络是由节点和连线构成,表示诸多对象及其相互联系.在数学上,网络是一种图,一般认为专指加权图.网络除了数学定义外,还有具体的物理含义,即网络是从某种相同类型的实际问题中抽象出来的模型.在 ...
- Python之路【第七篇】python基础 之socket网络编程
本篇文章大部分借鉴 http://www.cnblogs.com/nulige/p/6235531.html python socket 网络编程 一.服务端和客户端 BS架构 (腾讯通软件:ser ...
随机推荐
- 数据结构( Pyhon 语言描述 ) — —第9章:列表
概念 列表是一个线性的集合,允许用户在任意位置插入.删除.访问和替换元素 使用列表 基于索引的操作 基本操作 数组与列表的区别 数组是一种具体的数据结构,拥有基于单个的物理内存块的一种特定的,不变的实 ...
- python基础——10(三元运算符、匿名函数)
一.三元运算符 本质是if--else--的语法糖 前提:简化if--else--的结构,且两个分支有且只有一条语句 案例: a = 20 b = 30 res = a if a > b els ...
- vmware 装 puppy
说做就做了,但是开始怎么试都没装成功,后来在网上查了些资料,再结合自己的实践,终于成功了.下面我就把如何让Puppy Linux安装到VMware 硬盘上一步一步奉献给大家. 一.准备 1.1 安装好 ...
- Linux下二进制文件安装MySQL
MySQL 下载地址:https://dev.mysql.com/downloads/mysql/ 并按如下方式选择来下载安装包. 1. 设置配置文件/etc/my.cnmore /etc/my.cn ...
- 【03】图解原型和原型链by魔芋
[03]图解原型和原型链 一图胜前言 请先结合图解原型和原型链这张图. 可以分为4种情况. 情况1: Object有: constructor:是Function. __pro ...
- 【Luogu】P2953牛的数字游戏(博弈论)
题目链接 自己乱搞……然后一遍AC啦! 思路从基本的必胜态和必败态开始分析.我们把减去最大数得到的数叫作Max,减去最小数得到的数叫作Min. 那么开始分析. 一.0是必败态. 这个没法解释.题目就这 ...
- PAT天梯赛练习题——L3-008. 喊山(邻接表+BFS)
L3-008. 喊山 时间限制 150 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 喊山,是人双手围在嘴边成喇叭状,对着远方高山发出“喂—喂喂 ...
- NOI2015 荷马史诗 【k-哈夫曼树】
题目 追逐影子的人,自己就是影子 --荷马 Allison 最近迷上了文学.她喜欢在一个慵懒的午后,细细地品上一杯卡布奇诺,静静地阅读她爱不释手的<荷马史诗>.但是由<奥德赛> ...
- 【CCF】交通规划 Dijstra变形 优先级队列重载
[题意] 给定一个无向图,求这个图满足所有点到顶点的最短路径不变的最小生成树 [AC] 注意双向边要开2*maxm 注意优先级队列 参考https://www.cnblogs.com/cielosun ...
- Aspose.Words使用代码插入表格
Aspose.Words是一款功能强大的word文档处理控件,在不需要安装word的条件下,可进行word的创建,修改,转换等操作. Aspose.Words可以简单使用该产品提供的DocumentB ...