TCP服务器实现数据通信
前言
今天我们要介绍的是使用TCP协议实现数据通信,相比于之前写的UDP服务器实现数据信,在主体逻辑上并没有差别。客户端向服务器发送信息,服务器接受信息并回显,因为UDP是面向数据报,而TCP是面向连接的,所以在实现的时候接口上会有一些差别,下面,我们具体来看看UDP和TCP在编码的实现上有什么不同。
1.接口介绍
因为TCP是面向连接的,所以服务器创建完套接字,然后绑定成功后,将套接字设置为监听套接字
服务器启动之后,首先需要根据监听套接字建立连接,建立连接成功后返回一个新的文件描述符,后续的通信都是按照这个新的文件描述符按照读写文件的形式进行读写数据。
对于客户端来说创建完套接字之后,客户端启动之后首先需要建立连接
listen():设置sock为监听状态
#include <sys/types.h> #include <sys/socket.h> int listen(int sockfd, int backlog);
sockfd:创建套接字的返回值
backlog:底层全连接队列的长度
accept():服务端建立连接
#include <sys/types.h> #include <sys/socket.h> int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
sockfd:监听套接字
struct sockaddr* addr:输出型参数,可以获取服务端的IP地址和port端口号
socklen_t* addrlen:结构体的大小
返回值:返回一个新打开的文件描述符
connect():客户端建立连接
#include <sys/types.h> #include <sys/socket.h> int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);
sockfd:创建套接字返回值
struct sockaddr* addr:输出型参数,用来填写需要访问的服务端的IP地址和port端口号
socklen_t addrlen:结构体的大小
2.编写服务器
tcpServer.hpp
#pragma once #include <iostream> #include <string> #include <cstring> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #include "log.hpp" namespace server { using namespace std; enum { USAGE_ERR = 1, SOCKET_ERR, BIND_ERR, LISTEN_ERR }; static const uint16_t gport = 8080; static const int gback = 5; class TcpServer { public: TcpServer(const uint16_t &port = gport) : _port(gport), _sock(-1) {} void InitServer() { _sock = socket(AF_INET, SOCK_STREAM, 0); if (_sock < 0) { logMessage(FATAL, "create socket error"); exit(SOCKET_ERR); } logMessage(NORMAL, "create socket success"); // 绑定: struct sockaddr_in local; local.sin_family = AF_INET; local.sin_port = htons(_port); local.sin_addr.s_addr = INADDR_ANY; if (bind(_sock, (struct sockaddr *)&local, sizeof(local)) < 0) { logMessage(FATAL, "bind socket error"); exit(BIND_ERR); } logMessage(NORMAL, "bind socket success"); // 设置sock为监听状态: if (listen(_sock, gback) < 0) { logMessage(FATAL, "listen socket error"); exit(LISTEN_ERR); } logMessage(NORMAL, "listen socket success"); } void start() { for (;;) { // 建立连接: struct sockaddr_in peer; socklen_t len = sizeof(peer); int sock = accept(_sock, (struct sockaddr *)&peer, &len); if (sock < 0) { logMessage(ERROR, "accept error, next"); continue; } logMessage(NORMAL, "accept a new link success"); std::cout << "sock: " << sock << std::endl; //未来通信全部用sock,面向字节流的,后续全部都是文件操作: serviceIO(sock); close(sock); } } void serviceIO(int sock) { char buffer[1024]; while(true) { ssize_t n = read(sock,buffer,sizeof(buffer)-1); if(n > 0) { buffer[n] = 0; cout << "recvice message: " << buffer << endl; string outbuffer = buffer; outbuffer += "[server echo]"; write(sock,outbuffer.c_str(),outbuffer.size()); } else if(n == 0) { // 代表client退出 logMessage(NORMAL, "client quit, me too!"); break; } } } ~TcpServer() {} private: int _sock; uint16_t _port; }; }
tcpServer.cc:启动服务器
#include"tcpServer.hpp" #include<memory> using namespace server; static void Usage(string proc) { cout << "\nUsage:\n\t" << proc << " local_port\n\n"; } int main(int argc,char* argv[]) { if(argc != 2) { Usage(argv[0]); exit(USAGE_ERR); } uint16_t port = atoi(argv[1]); unique_ptr<TcpServer> tcs(new TcpServer(port)); tcs->InitServer(); tcs->start(); return 0; }
3.编写客户端
tcpClient.hpp
#pragma once #include <iostream> #include <string> #include <cstring> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> namespace client { using namespace std; class TcpClient { public: TcpClient(const string& serverip,const uint16_t port) :_serverip(serverip),_port(port),_sock(-1) {} void InitClient() { _sock = socket(AF_INET,SOCK_STREAM,0); if(_sock < 0) { cerr << "create sock fail" << endl; exit(-1); } } void start() { //建立连接: struct sockaddr_in server; server.sin_family = AF_INET; server.sin_port = htons(_port); server.sin_addr.s_addr = inet_addr(_serverip.c_str()); if(connect(_sock,(struct sockaddr*)&server,sizeof(server)) != 0) { cerr << "connect fail" << endl; } else { string message; while(true) { cout << "Please Enter: "; getline(cin,message); write(_sock,message.c_str(),message.size()); char buffer[1024]; int n = read(_sock,buffer,sizeof(buffer)-1); if(n > 0) { buffer[n] = 0; cout << "Server回复: " << buffer << endl; } else { break; } } } } ~TcpClient() { if(_sock >= 0) close(_sock); } private: string _serverip; uint16_t _port; int _sock; }; } // namespace client
tcpClient.cc:启动客户端
#include"tcpClient.hpp" #include<memory> using namespace client; static void Usage(string proc) { cout << "\nUsage:\n\t" << proc << " serverip serverport\n\n"; } int main(int argc,char* argv[]) { if(argc != 3) { Usage(argv[0]); exit(-1); } uint16_t port = atoi(argv[2]); string ip = argv[1]; unique_ptr<TcpClient> tcc(new TcpClient(ip,port)); tcc->InitClient(); tcc->start(); return 0; }
4.编译链接
makefile:
.PHONY:all all:tcpServer tcpClient tcpServer:tcpServer.cc g++ -o $@ $^ -std=c++11 tcpClient:tcpClient.cc g++ -o $@ $^ -std=c++11 .PHONY:clean clean: rm tcpServer tcpClient
5.测试
如图所示,服务端和客户端可以完成正常的数据通信了。
6.总结
TCP协议和UDP协议在数据通信的实现中,除了一些接口使用的不同之外,其实并没有太大的不同,在之前说的UDP是面向数据报的而TCP是面向字节流的,这些特性又是如何体现的呢?关于这个问题,博主将在后面的文章中会为大家继续进行介绍。不要错过哦!
到此这篇关于TCP服务器实现数据通信的文章就介绍到这了,更多相关TCP服务器数据通信内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
网络线路科普之CN2,GIA,CIA,BGP以及IPLC都是什么意思
购买海外vps或者物理服务器或者海外实体服务器托管的时候,在中国IDC服务器商中的有关网络线路带宽的术语有很多,今天专门做了一个专题,有关IPLC专线、CN2、BGP、CIA和普通线路知识普及2021-06-06通过IBM 3650 M2服务器的ServerGuide工具配置RAID图文教程
这篇文章主要介绍了通过IBM 3650 M2服务器的ServerGuide工具配置RAID图文教程,需要的朋友可以参考下2018-05-05阿里云主机不能用IP访问网站的解决方法(配置安全组规则搞定)
刚买了一台阿里云主机,迫不待及的试试速度,怎知网站访问不了,用IP或绑定域名都无法访问,后来提交工单才知道,需要配置安全组规则才行。针对同样像我一样的新手,本文就介绍一下如何在开通阿里云主机后配置安全组规则,让网站能够外网访问,需要的朋友可以参考下2020-07-07
最新评论