由于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)通信的更多相关文章

  1. 网络编程基础:粘包现象、基于UDP协议的套接字

    粘包现象: 如上篇博客中最后的示例,客户端有个 phone.recv(2014) , 当服务端发送给客户端的数据大于1024个字节时, 多于1024的数据就会残留在管道中,下次客户端再给服务端发命令时 ...

  2. Day09: socket网络编程-OSI七层协议,tcp/udp套接字,tcp粘包问题,socketserver

    今日内容:socket网络编程    1.OSI七层协议    2.基于tcp协议的套接字通信    3.模拟ssh远程执行命令    4.tcp的粘包问题及解决方案    5.基于udp协议的套接字 ...

  3. Socket网络编程系列教程序

    C语言的用途相当多,可以用在数据结构.数据库.网络.嵌入式等方面,历经40多年不衰,真是厉害!最近一直想从某一应用方面写一个系列教程,好好地把某一方面讲深讲透.         正好博主对网络方面的编 ...

  4. Linux Socket 网络编程

    Linux下的网络编程指的是socket套接字编程,入门比较简单.在学校里学过一些皮毛,平时就是自学玩,没有见识过真正的socket编程大程序,比较遗憾.总感觉每次看的时候都有收获,但是每次看完了之后 ...

  5. Socket网络编程--Libev库学习(1)

    这一节是安装篇. Socket网络编程不知不觉已经学了快两个月了.现在是时候找个网络库学学了.搜索了很多关于如何学网络编程的博客和问答.大致都是推荐学一个网络库,至于C++网络库有那么几个,各有各的好 ...

  6. Python Socket 网络编程

    Socket 是进程间通信的一种方式,它与其他进程间通信的一个主要不同是:它能实现不同主机间的进程间通信,我们网络上各种各样的服务大多都是基于 Socket 来完成通信的,例如我们每天浏览网页.QQ ...

  7. Python全栈【Socket网络编程】

    Python全栈[socket网络编程] 本章内容: Socket 基于TCP的套接字 基于UDP的套接字 TCP粘包 SocketServer 模块(ThreadingTCPServer源码剖析) ...

  8. python之Socket网络编程

    什么是网络? 网络是由节点和连线构成,表示诸多对象及其相互联系.在数学上,网络是一种图,一般认为专指加权图.网络除了数学定义外,还有具体的物理含义,即网络是从某种相同类型的实际问题中抽象出来的模型.在 ...

  9. Python之路【第七篇】python基础 之socket网络编程

    本篇文章大部分借鉴 http://www.cnblogs.com/nulige/p/6235531.html python socket  网络编程 一.服务端和客户端 BS架构 (腾讯通软件:ser ...

随机推荐

  1. Educational Codeforces Round 31- D. Boxes And Balls

    D. Boxes And Balls time limit per test2 seconds memory limit per test256 megabytes 题目链接:http://codef ...

  2. PAT Basic 1027

    1027 打印沙漏 本题要求你写个程序把给定的符号打印成沙漏的形状.例如给定17个“*”,要求按下列格式打印 ***** *** * *** ***** 所谓“沙漏形状”,是指每行输出奇数个符号:各行 ...

  3. Uiautomator简介及其环境搭建、测试执行

    UiAutomator框架使用指南 UiAutomator是Google开发的自动化测试工具,通过UI创建自动化测试代码,来测试界面(UI)的有效功能,可以针对应用程序运行在一个或更多的设备上.我们并 ...

  4. JSTL 配置

    pache Tomcat安装JSTL 库步骤如下: 从Apache的标准标签库中下载的二进包(jakarta-taglibs-standard-current.zip). 官方下载地址:http:// ...

  5. <题解>洛谷P3385 【模板】负环

    题目链接 判断一张图中是否存在关于顶点1的负环: 可以用SPFA跑一遍,存在负环的情况就是点进队大于n次 因为在存在负环的情况下,SPFA会越跑越小,跑进死循环 在最差的情况下,存在的负环长度是“n+ ...

  6. luogu3834 【模板】可持久化线段树 1(主席树)

    关于空间,第零棵树是 \(4n\),其后每棵树都要多来 \(\log(n)\) 的空间,所以我是开 \(n(4+\log(n))\) 的空间. 关于借用节点: 图片来自这里 #include < ...

  7. PL/SQL配置访问多个不同IP的oracle

     第一步:打开Oracle的tnsnames.ora文件.添加 test = (DESCRIPTION =    (ADDRESS_LIST =      (ADDRESS = (PROTOCOL = ...

  8. ajax 下拉加载更多效果

    1.生成HTML <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...

  9. vue v-dialogDrag: 弹窗拖拽

    Vue.directive('dialogDrag', { inserted:function(el) { const dragDom = el.querySelector('.jsPropupLay ...

  10. POJ——2449Remmarguts' Date(A*+SPFA)

    Remmarguts' Date Time Limit: 4000MS   Memory Limit: 65536K Total Submissions: 26504   Accepted: 7203 ...