建立基于TCP的客户端和服务端

函数介绍: 

1.socket()

作用:创建套接字

domain:

  • AF_INET:IPv4 Internet 协议族。
  • AF_INET6:IPv6 Internet 协议族。
  • AF_UNIX:Unix 域协议族,用于在同一台主机上的进程间通信。

type:

  • SOCK_STREAM:提供序列化的、可靠的、双向连接的字节流服务,通常用于TCP连接。
  • SOCK_DGRAM:提供无连接的、不可靠的数据报服务,通常用于UDP通信。

protocol:

        这个参数指定了使用的特定协议。对于很多常见的套接字类,如 SOCK_STREAM 和 SOCK_DGRAM,这个参数可以设置为0,系统会自动选择一个默认协议。

return value:

socket() 函数调用成功时,返回一个非负整数,即新创建的套接字的文件描述符(socket descriptor)。如果调用失败,函数返回-1,并且 errno 被设置为描述错误的值。

服务端:

2.bind()

作用:将套接字(fd)和ip和port绑定起来

sockfd:

  • 要绑定的文件描述符,也就是socket的返回值

addr:

 指向 sockaddr 结构体的指针,该结构体包含了套接字的地址信息。

  • sin_family:协议家族
  • sin_port:要绑定的端口。
  • sin_addr.s_addr:要绑定的ip

 介绍以下里面的函数:

h表示host,n表示network

htons 通常用于端口号,htonl 通常用于IP地址

作用:将IPv4地址的点分十进制字符串表示转换为网络字节序的二进制值

INADDR_ANY:用于指定一个特殊的IPv4地址0.0.0.0,表示“接受任何可用的网络接口”,通常用在服务器监听上。

addrlen: 大小

return value:

成功返回0,失败返回-1并设置错误码。

3.listen()

作用:将套接字设置为监听状态可以监听链接请求

sockfd:socket()的返回值

backlog:一个大于0的整数(后续补充)

return value:成功返回0失败返回-1并设置错误码

4.accept()

 作用:接受链接请求

sockfd:socket()的返回值

addr和addrlen:输出型参数,用于存储连接客户端的地址信息。

return value:

成功返回一个新的文件描述符用于与连接的客户端进行通信。之后的read和write传入的都是accept的返回值。

失败返回-1,设置错误码。

 客户端:

2.connect()

 

作用:向服务端发起链接

sockfd:socket()的返回值

addr:要连接的服务端的地址信息

addrlen:addr的大小

return value:

成功返回0,失败返回-1并设置错误码。

代码: 

tcpServer.hpp

#pragma once
#include <iostream>
#include <string>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <cstring>
#include <cstdlib>
#include <sys/wait.h>
#include <signal.h>
#include <pthread.h>
#include "log.hpp"
#include "threadpool.hpp"namespace server
{enum{USAGE_ERR = 1,SOCKET_ERR,BIND_ERR,LISTEN_ERR,OPEN_ERR};static const uint16_t gport = 8080;static const int gbacklog = 5; // 底层链接队列的长度class TcpServer;class ThreadData{public:ThreadData(TcpServer *self, int sock) : _self(self), _sock(sock){}public:TcpServer *_self;int _sock;};class TcpServer{public:TcpServer(const uint16_t &port = gport) : _listensock(-1), _port(port){}void initServer(){// 1.创建socket文件套接字对象_listensock = socket(AF_INET, SOCK_STREAM, 0);if (_listensock < 0){logMessage(FATAL, "create socket error");exit(SOCKET_ERR);}logMessage(NORMAL, "create socket success:%d", _listensock);// 2.bind绑定自己的网络信息struct sockaddr_in local;memset(&local, 0, sizeof(local));local.sin_family = AF_INET;local.sin_port = htons(_port);local.sin_addr.s_addr = INADDR_ANY;if (bind(_listensock, (struct sockaddr *)&local, sizeof(local)) < 0){logMessage(FATAL, "bind socket error");exit(BIND_ERR);}logMessage(NORMAL, "bind socket success");// 3.设置socket为监听状态,获取新的客户端链接if (listen(_listensock, gbacklog) < 0){logMessage(FATAL, "listen socket error");exit(LISTEN_ERR);}logMessage(NORMAL, "listen socket success");}void start(){// 4.线程池初始化ThreadPool<Task>::getInstance()->run();logMessage(NORMAL, "Thread init success");// signal(SIGCHLD,SIG_IGN);//忽略这个信号for (;;){// 4.server获取新链接// sock,和client进行通信的fdstruct sockaddr_in peer;socklen_t len = sizeof(peer);int sock = accept(_listensock, (struct sockaddr *)&peer, &len);if (sock < 0){logMessage(ERROR, "accept error");continue;}logMessage(NORMAL, "accept a new link success,get a new sock:%d", sock);serviceIO(sock);close(sock);//对一个已经使用完毕的sock,我们必须要关闭这个sock,要不然会导致文件描述符泄漏}}// static void *threadRoutine(void *args)// {//     pthread_detach(pthread_self());//     ThreadData *td = static_cast<ThreadData *>(args);//     td->_self->serviceIO(td->_sock);//     delete td;//     close(td->_sock);//     return nullptr;// }~TcpServer() {}private:int _listensock;uint16_t _port;};
}
// 5.这里就是一个sock,未来通信我们就用这个sock,面向字节流的,后续全部都是文件操作!
//  serviceIO(sock);
//  close(sock);//对一个已经使用完毕的sock,我们必须要关闭这个sock,要不然会导致文件描述符泄漏// version2,多进程版,多线程版,线程池版
// version1,多进程版
//  pid_t id = fork();
//  if(id == 0)//child
//  {
//      close(_listensock);
//      if(fork()>0) exit(0);//关闭child,创建了一个孙子。因为child退了,孙子进程成了孤儿进程,由操作系统回收
//      serviceIO(sock);
//      close(sock);//对一个已经使用完毕的sock,我们必须要关闭这个sock,要不然会导致文件描述符泄漏
//      exit(0);
//  }
// father
//  pid_t ret = waitpid(id,nullptr,0);
//  if(ret>0)
//  {
//      std::cout<<"wait success:"<<ret<<std::endl;
//  }// version2,多进程版
//  pid_t id = fork();
//  if(id == 0)//child
//  {
//      close(_listensock);
//      serviceIO(sock);
//      close(sock);//对一个已经使用完毕的sock,我们必须要关闭这个sock,要不然会导致文件描述符泄漏
//      exit(0);
//  }
//  close(sock);// version3:多线程版
//  pthread_t tid;
//  ThreadData* td = new ThreadData(this,sock);
//  pthread_create(&tid,nullptr,threadRoutine,td);// version4:线程池
//ThreadPool<Task>::getInstance()->push(Task(sock, serviceIO));

 tcpServer.cc

#include"tcpServer.hpp"
#include<memory>
using namespace server;
using namespace std;
static void Usage(string proc)
{cout<<".\nUsage:\n\t"<<proc<<"local_port\n\n";
}
//tcp服务器,启动上和udp Server一模一样
//./tcpServer local_port
int main(int argc,char* argv[])
{if(argc!=2){Usage(argv[0]);exit(USAGE_ERR);}uint16_t port = atoi(argv[1]);unique_ptr<TcpServer> tsvr(new TcpServer(port));tsvr->initServer();tsvr->start();return 0;}

tcpClient.hpp 

#pragma once
#include <iostream>
#include<string>
#include<sys/socket.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#include<cstring>
#define NUM 1024
class TcpClient
{public:TcpClient(const std::string &serverip,const uint16_t &serverport):_sock(-1),_serverip(serverip),_serverport(serverport){}void initClient(){//1.创建socket_sock = socket(AF_INET,SOCK_STREAM,0);if(_sock < 0){std::cerr<<"socket create error"<<std::endl;exit(2);}//2.客户端不用显示的bind//3.要不要listen?没人连我啊//4.要不要accept?no//5.要什么呢?要发起链接!}void start(){struct sockaddr_in server;memset(&server,0,sizeof(server));server.sin_family = AF_INET;server.sin_port = htons(_serverport);server.sin_addr.s_addr = inet_addr(_serverip.c_str());if(connect(_sock,(struct sockaddr*)&server,sizeof(server)) != 0){std::cerr<<"socket connect error"<<std::endl;}else{std::string msg;while(1){std::cout<<"Enter#";std::getline(std::cin,msg);write(_sock,msg.c_str(),msg.size());char buffer[NUM];int n = read(_sock,buffer,sizeof(buffer)-1);if(n>0){//目前把读到的数据当字符串buffer[n] = 0;std::cout<<"Server回显#"<<buffer<<std::endl;}else{break;}}}}~TcpClient(){if(_sock >= 0)close(_sock);//文件描述符的生命周期随进程,所以这里不写也行}
private:int _sock;std::string  _serverip;uint16_t _serverport;};

tcpClient.cc 

#include"tcpClient.hpp"
#include<memory>
using namespace std;
static void Usage(string proc)
{cout<<".\nUsage:\n\t"<<proc<<"serverip serverport\n\n";
}
//./tcpClient serverip serverport
int main(int argc,char* argv[])
{if(argc!=3){Usage(argv[0]);exit(1);}string serverip = argv[1];uint16_t serverport = atoi(argv[2]);unique_ptr<TcpClient> tcli(new TcpClient(serverip,serverport));tcli->initClient();tcli->start();return 0;
}

Task.hpp 

#pragma once#include <iostream>
#include <string>
#include <cstdio>
#include <functional>
#include <unistd.h>
#include <string.h>
#include"log.hpp"
void serviceIO(int sock)
{char buffer[1024];while (true){ssize_t n = read(sock, buffer, sizeof(buffer) - 1);if (n > 0){// 目前我们把读到的数据当成字符串,截止目前// 为什么UDP并不符合文件特性,文件也是字节流的,UDP是数据报的,所以要用特殊函数来读buffer[n] = 0;std::cout << "recv messages:" << buffer << std::endl;write(sock, buffer, strlen(buffer)); // 多路转接}else if (n == 0){// 代表client退出logMessage(NORMAL, "client quit,me too!");break;}}close(sock);
}class Task
{using func_t = std::function<void(int)>;public:Task() {}Task(int sock, func_t func): _sock(sock), _callback(func){}void operator()(){_callback(_sock);}private:int _sock;func_t _callback;
};

log.hpp 

#pragma once
#include<iostream>
#include<string>
#include<cstdarg>
#include<ctime>
#include<unistd.h>
//cat /var/log/messages系统日志
//日志等级
#define DEBUG   0
#define NORMAL  1
#define WARNING 2
#define ERROR   3
#define FATAL   4//致命错误const char* to_levelstr(int level)
{switch(level){case DEBUG:return "DEBUG";case NORMAL:return "NORMAL";case WARNING:return "WARNING";case ERROR:return "ERROR";case FATAL:return "FATAL";}
}
// void logMessage(int level,const char* messages)
// {
//     std::cout<<messages<<std::endl;
// }
// void logMessage(DEBUG,"hello %f ,%d,%c",3.14,10,'C');
void logMessage(int level,const char* format,...)
{//[日志等级][时间戳/时间][pid][message]//暂定// va_list start;// va_start(start);// while(*p)// {//     switch(*p)//     {//         case '%'://             p++;//             if(*p == 'f') arg = va_arg(start,float);//         ...//     }// }// va_end(start);#define NUM 1024char logprefix[NUM];snprintf(logprefix,sizeof(logprefix),"[%s][%ld][pid:%d]",to_levelstr(level),(long int)time(nullptr),getpid());char logcontent[NUM];va_list arg;va_start(arg,format);vsnprintf(logcontent,sizeof(logcontent),format,arg);std::cout<<logprefix<<logcontent<<std::endl;
}

makefile 

cc=g++
.PHONY:all
all:tcpClient tcpServertcpClient:tcpClient.cc$(cc) -o $@ $^ -std=c++11 -lpthread
tcpServer:tcpServer.cc$(cc) -o $@ $^ -std=c++11 -lpthread
.PHONY:clean
clean:rm -f tcpClient tcpServer

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/62854.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CNCF云原生生态版图-分类指南(三)- 运行时

CNCF云原生生态版图-分类指南&#xff08;三&#xff09;- 运行时 CNCF云原生生态版图-分类指南三、运行时&#xff08;Runtime&#xff09;&#xff08;一&#xff09;云原生存储&#xff08;Cloud Native Storage&#xff09;1. 是什么&#xff1f;2. 解决什么问题&#xff1…

MVC基础——市场管理系统(三)Clean Architecture

文章目录 项目地址五、Clean Architecture5.1 user cage driven5.1.1创建CoreBusiness 5.2 创建UseCases5.2.1 创建CategoriesUseCases1. 创建VeiwCategoriesUseCase获取所有Cagegory 5.2.2. 实现ICategoryRepository接口3. 实现获取所有Category的方法4. 实现获取一个Cagegory…

手机上和电脑上都能观看的翻页电子书是如何制作的?

想知道手机上和电脑上都能观看的翻页电子书是都是如何制作的&#xff1f; 想知道这样的电子书是怎样呈现出来的&#xff1f; 那收藏这篇文章&#xff0c;我来跟大家说说该如何实现。 操作方法 一、登录FLBOOK 二、开始制作&#xff0c;有多种创建方式&#xff0c;分别是&…

Java 实现给pdf文件指定位置盖章功能

Java 实现给pdf文件指定位置盖章功能 开发中遇到一个需求, 需要给用户上传的的pdf文件, 指定位置上盖公章的功能, 经过调研和对比, 最终确定实现思路. 这里是使用pdf文件中的关键字进行章子的定位, 之所以这样考虑是因为如果直接写死坐标的话, 可能会出现因pdf大小, 缩放, 盖章…

ASP.NET Core API + MySql

环境 数据库&#xff1a; mysql8.0 后端&#xff1a; vs2022 ASP.NET Core API .net 8 前端&#xff1a; Hbuilderx bootstrap 5.3.0 jquery v3.7.1 bootstrap-table 1.23.5 创建项目 添加资源包 AutoMapper Microsoft.EntityFrameworkCore.Tools 8.0.0 Pomelo.EntityFramew…

Bananna Pi开源社区联合矽昌通信打造开源的低成本Wifi5路由器

香蕉派 BPI-Wifi5 路由器采用矽昌SF19A2890S2芯片方案设计。它是一款高性能无线路由器&#xff0c;适用于小微企业、家庭和其他网络环境。Banana Pi开源社区提供整体解决方案。所有代码开源&#xff0c;用户可以在上面自由开发自己的应用。 Banana Pi wifi5 路由器github代码: …

item2 for macos

安装Item2 brew install iterm2 查看终端类型 cat /etc/shells Mac OS X 10.15 已经将默认的shell从Bash换成了zsh&#xff0c;所以不用安装&#xff0c;10.15以前的可以使用下面的命令进行安装 brew install zsh 安装Oh My ZSH # curl sh -c "$(curl -fsSL https://ra…

Pytest-Bdd-Playwright 系列教程(14):Docstring 参数

Pytest-Bdd-Playwright 系列教程&#xff08;14&#xff09;&#xff1a;Docstring 参数 前言一、什么是docstring?二、基本语法三、主要特点四、实际例子五、注意事项六、使用建议总结 前言 在自动化测试的过程中&#xff0c;我们经常需要处理复杂的测试数据或需要输入多行文…

手机租赁系统开发指南一站式服务流程解析

内容概要 手机租赁系统的开发是一个复杂但有趣的过程&#xff0c;像搭建乐高一样&#xff0c;只要找到合适的模块&#xff0c;就能打造出一个宾至如归的租赁平台。在这部分&#xff0c;我们将对开发流程的整体结构进行简要概述&#xff0c;并指出每个环节的重要性。 首先&…

OpenAI 正式赋予 ChatGPT 通过视频实时与用户互动的能力

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

EasyExcel设置表头上面的那种大标题(前端传递来的大标题)

1、首先得先引用easyExcel的版本依赖&#xff0c;我那 <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>2.2.6</version> </dependency> 2、然后得弄直接的实体类&#xff0c;&…

纯血鸿蒙崛起,原生Android挑战?两大操作系统巅峰对决,智能设备未来谁主沉浮?

鸿蒙HarmonyOS和原生Android系统虽然在一些方面相似&#xff0c;但在架构、设计理念、API、开发工具等方面存在一些差异。鸿蒙系统的目标是跨设备、分布式的操作系统&#xff0c;强调多设备协同和资源共享&#xff0c;而Android则主要集中在智能手机和移动设备领域。 下面将从…

计算机网络:传输层、应用层、网络安全、视频/音频/无线网络、下一代因特网

目录 &#xff08;五&#xff09;传输层 1&#xff0e;传输层寻址与端口 2&#xff0e;无连接服务与面向连接服务 3. 传输连接的建立与释放 4. UDP 的优点 5. UDP 和 TCP 报文段报头格式 6. TCP 的流量控制 7&#xff0e;TCP 的拥塞控制 8. TCP 传送连接的管理 &#…

【前端开发】HTML+CSS网页,可以拿来当作业(免费开源)

HTML代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content_lizhongyu"widthdevice-width, initial-scale1.0"><title>小兔鲜儿-新鲜、惠民、快捷<…

CV(4)--边缘提取和相机模型

前言 仅记录学习过程&#xff0c;有问题欢迎讨论 边缘提取&#xff08;涉及语义分割&#xff09;&#xff1a; 图象的边缘是指图象局部区域亮度变化显著的部分,也有正负之分&#xff0c;暗到亮为正 求边缘的幅度&#xff1a;sobel&#xff0c;Canny算子 图像分高频分量和低…

智能技术引领未来:自动图像标注的创新应用与发展

&#x1f351;个人主页&#xff1a;Jupiter. &#x1f680; 所属专栏&#xff1a;传知代码 欢迎大家点赞收藏评论&#x1f60a; 目录 概述算法原理核心逻辑效果演示使用方式参考文献 参考文献&#xff1a;需要本文的详细复现过程的项目源码、数据和预训练好的模型可从该地址处获…

C语言-排序

常见的排序算法分为以下四种&#xff0c;插入排序&#xff0c;选择排序&#xff0c;交换排序&#xff0c;归并排序。 一、插入排序 (一)直接插入排序 直接插入排序&#xff0c;将一段数组看做被分成已排序序列和未排序序列&#xff0c;排序过程是从未排序序列的元素开始&…

【Java笔记】LinkedList 底层结构

一、LinkedList 的全面说明 LinkedList底层实现了双向链表和双端队列特点可以添加任意元素(元素可以重复)&#xff0c;包括null线程不安全&#xff0c;没有实现同步 二、LinkedList 的底层操作机制 三、LinkedList的增删改查案例 public class LinkedListCRUD { public stati…

网管平台(基础篇):路由器的介绍与管理

路由器简介 路由器&#xff08;Router&#xff09;是一种计算机网络设备&#xff0c;它的主要作用是将数据通过打包&#xff0c;并按照一定的路径选择算法&#xff0c;将网络传送至目的地。路由器能够连接两个或更多个网络&#xff0c;并根据信道的情况自动选择和设定路由&…

排序算法(2):选择排序

问题 排序 [30, 24, 5, 58, 18, 36, 12, 42, 39] 选择排序 选择排序每次从待排序序列中选出最小&#xff08;或最大&#xff09;的元素&#xff0c;将其放到序列的起始位置&#xff0c;然后&#xff0c;再从剩余未排序元素中继续寻找最小&#xff08;或最大&#xff09;元素…