socket 网络编程快速入门(一)教你编写基于UDP/TCP的服务(客户端)通信

因为UNIX和Win的socket大同小异,为了方便和大众化,这里先介绍Winsock编程。

socket 网络编程的难点在入门的时候就是对基本函数的了解和使用,因为这些函数的结构往往比较复杂,参数大部分都是结构体,令人难以记忆和理解。

但是一旦我们知道这些函数包括其参数的具体含义,socket网络编程也就变得不是那么复杂。这里不赘述 具体函数的详细含义,网络上有很多的文章,同时笔者建议大家参考

MSDN,对返回值,参数等会有更好的理解。

以下均为单线程的简单实例,多线程的请关注下一篇文章。

(一)

UDP:中文名是用户数据报协议,是OSI参考模型中的一种无连接的传输层协议,提供面向事物的简单不可靠信息传送服务,在网络良好的情况下也可能会出现丢包现象。

UDP通讯流程:

(Winsock的初始化:WSAStartup) //windows特有的

Server端:

socket:建立套接字

bind:发布端口(netstat -an可见已经发布的端口)

sendto/recv/recvfrom:收发数据

Client端:

socket:建立套接字

bind:发布端口(netstat -an可见已经发布的端口)

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 };   //随机分配一个端口 (仅初始化一个值,让客户端自己分配端口)
	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和端口号

	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绑定端口失败!" << endl;
		cout << "错误码是:" << WSAGetLastError() << endl;
		return -1;
	}
	else
	{
		cout << "端口发布成功:" << 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:发布端口

listen:开始侦听

accept:接纳客户端连接(如同公司的前台)

send/recv/recvfrom:收发数据(如同公司的客户经理)

Client端:

socket:建立套接字

bind:发布端口(客户端可以不绑定或绑定0端口)

connect:连接到服务器

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绑定端口失败!" << endl;
		cout << "错误码是:" << WSAGetLastError() << endl;
		return -1;
	}
	else
	{
		cout << "端口发布成功:" << 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和端口

	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 };   //随机分配一个端口
	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)); //指定端口发送数据
	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;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-26 12:16:17

socket 网络编程快速入门(一)教你编写基于UDP/TCP的服务(客户端)通信的相关文章

socket 网络编程快速入门(二)教你编写基于UDP/TCP的服务端多线程通信

在上一篇博文中,我们介绍了利用socket进行简单的UDP/TCP的服务端和客户端的通信. (一) 在基于UDP的程序中,你有没有想过,如果我的这台主机在通讯的时候要求既能够收到别的主机发来的数据,又能够自己向目的主机发出数据,该怎样实现?也就是说需要两个while循环同时进行.答案是使用多线程,一个线程用于接受数据,另一个线程用来发送数据.接下来我们介绍WinSock的多线程编程. 多线程的实现我们使用_beginthread()函数: uintptr_t _beginthread( void

socket 网络编程高速入门(一)教你编写基于UDP/TCP的服务(client)通信

由于UNIX和Win的socket大同小异,为了方便和大众化,这里先介绍Winsock编程. socket 网络编程的难点在入门的时候就是对基本函数的了解和使用,由于这些函数的结构往往比較复杂,參数大部分都是结构体,令人难以记忆和理解. 可是一旦我们知道这些函数包含其參数的详细含义,socket网络编程也就变得不是那么复杂. 这里不赘述 详细函数的详细含义.网络上有非常多的文章.同一时候笔者建议大家參考 MSDN.对返回值,參数等会有更好的理解. 下面均为单线程的简单实例,多线程的请关注下一篇文

Unix网络编程学习笔记之第8章 基于UDP套接字编程

一. UDP C/S的典型函数调用 UDP没有像TCP那样的连接,客户端直接sendto向某服务器发送数据,服务器端一直recvfrom阻塞,以接收任何客户端发送的数据. 二. sendto和recvfrom函数 int sendto(int sockfd, const void* buff, size_t nbytes, int flag, const struct sockaddr* to, socklen_taddrlen); int recvfrom(int sockfd, void*

Socket网络编程进阶与实战

第1章 课程导学(Java语言教学)[说明:课程案例部分以Java语言实现]本章节首先会对课程进行导学讲解,包括为什么应该学习本课程,课程目标与收获,课程内容安排,适合人群和学习建议等,接着会讲解代码规范与开发注意事项,目的全在于希望极大的方便同学进行本课程的学习.... 第2章 Socket网络编程快速入门本章首先整体介绍什么是Socket网络编程:让大家对Socket有个大概的概念与方向.之后通过几个小Case引出课程的主角,轻松愉快的让你体验到Socket编程的快感.本章节主要收获:1.

[8421论坛]2019 Socket网络编程入门到进阶与实战完整视频教程

第一章 课程介绍及java语言简介第二章 快速入门Socket网络编程第三章 快速入门Socket UDP第四章 快速入门Socket TCP第五章 UDP辅助TCP实现点对点传输案例第六章 简易聊天室案例第七章 服务器传输优化-NIO第八章 数据传输稳定性优化第九章 局域网文件快传技术实战第十章 聊天室升级版实战第十一章 语音数据即时通信实战第十二章 整体代码结构梳理与升华资料 下载地址? 原文地址:https://www.cnblogs.com/yellowvase/p/10720644.h

Linux Socket 网络编程

Linux下的网络编程指的是socket套接字编程,入门比较简单.在学校里学过一些皮毛,平时就是自学玩,没有见识过真正的socket编程大程序,比较遗憾.总感觉每次看的时候都有收获,但是每次看完了之后,过段时间不看,重新拾起这些知识的时候又要从头开始,所以,在这里做个笔记也算是做个模板,以后可以直接从某一个阶段开始接着玩... 1. socket套接字介绍 socket机制其实就是包括socket, bind, listen, connect, accept等函数的方法,其通过指定的函数实现不同

Java基础篇Socket网络编程中的应用实例

说到java网络通讯章节的内容,刚入门的学员可能会感到比较头疼,应为Socket通信中一定会伴随有IO流的操作,当然对IO流比较熟练的哥们会觉得这是比较好玩的一章,因为一切都在他们的掌握之中,这样操作起来就显得非常得心应手,但是对于IO本来就不是多熟悉的哥们来说就有一定的困难了,在搞清楚IO流操作机制的同时还必须会应用到Socket通信中去,否则会对得到的结果感到非常郁闷和懊恼,下面就和大家一起分享一下自己遇到一点小麻烦后的感触以及给出的解决办法. 要求:客户端通过Socket通信技术上传本地一

C# Socket网络编程实例

本文实例讲述了C# Socket网络编程技巧.分享给大家供大家参考.具体分析如下: 客户端要连接服务器:首先要知道服务器的IP地址.而服务器里有很多的应用程序,每一个应用程序对应一个端口号 所以客户端想要与服务器中的某个应用程序进行通信就必须要知道那个应用程序的所在服务器的IP地址,及应用程序所对应的端口号 TCP协议: 安全稳定,一般不会发生数据丢失,但是效率低.利用TCP发生数据一般经过3次握手(所有效率低,自己百度三次握手) UDP协议: 快速,效率高,但是不稳定,容易发生数据丢失(没有经

bash编程快速入门

首先,我们简单的介绍一下bash,bash是GNU计划编写的Unixshell,它是许多Linux平台上的内定shell,它提供了用户与系统的很好的交互,对于系统运维人员,bash的地位是举足轻重的,bash编程能很快处理日常的任务 bash入门,一个最简单的bash例子 #vim hello.sh #!/bin/bash #This is the first example of the bash #echo "Hello world" 下面,我们就这个简单的bash 脚本来介绍一下