网络通讯录服务器

这篇具有很好参考价值的文章主要介绍了网络通讯录服务器。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

六、通讯录4.0实现—⽹络版

简易版本

服务端完整版本

客户端完整版本

Protobuf还常⽤于通讯协议、服务端数据交换场景。那么在这个⽰例中,我们将实现⼀个⽹络版本的
通讯录,模拟实现客⼾端与服务端的交互,通过Protobuf来实现各端之间的协议序列化。
需求如下:

  • 客⼾端可以选择对通讯录进⾏以下操作:

    • 新增⼀个联系⼈
    • 删除⼀个联系⼈
    • 查询通讯录列表
    • 查询⼀个联系⼈的详细信息
  • 服务端提供增删查能⼒,并需要持久化通讯录。

  • 客⼾端、服务端间的交互数据使⽤Protobuf来完成。

如下图:

  • 客户端要有一个菜单,新增一个联系人…
  • 每个功能都有一对请求和响应协议.
  • 例如实现新增一个联系人,首先我们要设计message, 中间是网络传输.
  • 然后安装图形序号执行,客户端完成:1,2,3,7 ; 服务端完成:1,4.5,6

网络通讯录服务器,序列化工具,网络,服务器,运维

1. 环境搭建

1.1 安装Httplib库

Httplib库:cpp-httplib是个开源的库,是⼀个c++封装的http库,使⽤这个库可以在linux、windows平台下完成http客⼾端、http服务端的搭建。

使⽤起来⾮常⽅便,只需要包含头⽂件 httplib.h即可。编译程序时,需要带上-lpthread选项。

源码库地址:https://github.com/yhirose/cpp-httplib
镜像仓库:https://gitcode.net/mirrors/yhirose/cpp-httplib?utm_source=csdn_github_accelerator

1.1升级 gcc

$ gcc -v
Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/libexec/gcc/x86_64-redhat-linux/4.8.5/lto-wrapper
Target: x86_64-redhat-linux
Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --
infodir=/usr/share/info --with-bugurl=http://bugzilla.redhat.com/bugzilla --enable-
bootstrap --enable-shared --enable-threads=posix --enable-checking=release --with-system-
zlib --enable-__cxa_atexit --disable-libunwind-exceptions --enable-gnu-unique-object --
enable-linker-build-id --with-linker-hash-style=gnu --enable-languages=c,c++,objc,obj-
c++,java,fortran,ada,go,lto --enable-plugin --enable-initfini-array --disable-libgcj --
with-isl=/builddir/build/BUILD/gcc-4.8.5-20150702/obj-x86_64-redhat-linux/isl-install --
with-cloog=/builddir/build/BUILD/gcc-4.8.5-20150702/obj-x86_64-redhat-linux/cloog-install -
-enable-gnu-indirect-function --with-tune=generic --with-arch_32=x86-64 --build=x86_64-
redhat-linux
Thread model: posix
gcc version 4.8.5 20150623 (Red Hat 4.8.5-44) (GCC)
cpp-httplib 用老的编译器,要么编译不通过,要么直接运行报错
百度搜索:scl gcc devsettool升级gcc
//安装scl
$ sudo yum install centos-release-scl scl-utils-build
//安装新版本gcc,这里也可以把7换成8或者9,我用的是9,也可以都安装
$ sudo yum install -y devtoolset-7-gcc devtoolset-7-gcc-c++
$ ls /opt/rh/
//启动: 细节,命令行启动只能在本会话有效
$ scl enable devtoolset-7 bash
$ gcc -v
//可选:如果想每次登陆的时候,都是较新的gcc,需要把上面的命令添加到你的~/.bash_profile中
$ cat ~/.bash_profile
# .bash_profile
# Get the aliases and functions
if [ -f ~/.bashrc ]; then
. ~/.bashrc
fi
# User specific environment and startup programs
PATH=$PATH:$HOME/.local/bin:$HOME/bin
export PATH
#添加下面的命令,每次启动的时候,都会执行这个scl命令
scl enable devtoolset-7 bash
or
scl enable devtoolset-8 bash
or
scl enable devtoolset-9 bash

2. 搭建简单的服务器

用来测试httplib

Client.cpp

#include"httplib.h"
#include<iostream>
#include<string>
using namespace std;
using namespace httplib;

const string IP = "127.0.0.1";// 监听所有的ip
//const string IP = "192.139.99.192";

const int PORT = 6666; 
int main()
{
  Client client(IP.c_str(),PORT);
  // 传输./test-Post 资源
  Result res1 = client.Post("./test-Post");
  if(res1->status ==200){
    cout<<"post sucess Post"<<endl;
  }
  // 请求./test-Post 资源
  Result res2 = client.Get("./test-Get");
  if(res2->status ==200){
    cout<<"get sucess Get"<<endl;
  }
}

Server.cpp

#include"httplib.h"
#include<iostream>
#include<string>
using namespace std;
using namespace httplib;
const string IP = "127.0.0.1";

//const string IP = "0.0.0.0";
//const string IP = "192.139.99.192";

const int PORT = 6666; 
int main()
{
  Server svr;
  // 注册post处理方法

  svr.Post("./test-Post",[](const Request& req ,Response& res){
      cout<< "server test Post"<<endl;
      res.status=200;
  });
  svr.Get("./test-Get",[](const Request& req ,Response& res){
      cout<< "server test Get"<<endl;
      res.status=200;
  });

  svr.listen(IP.c_str(),PORT);
  cout<<"sucess"<<endl;
}

3. 约定双端交互接⼝

定制http协议

新增⼀个联系⼈:

[请求]
	Post /contacts/add 
	Content-Type: application/protobuf
	AddContactRequest
[响应]
	Content-Type: application/protobuf
	AddContactResponse

删除⼀个联系⼈:

[请求]
	Post /contacts/del 
	Content-Type: application/protobuf
	DelContactRequest
[响应]
	Content-Type: application/protobuf
	DelContactResponse

查询通讯录列表:

[请求]
	GET /contacts/find-all
[响应]
	Content-Type: application/protobuf
	FindAllContactsResponse

查询⼀个联系⼈的详细信息:

[请求]
	Post /contacts/find-one 
	Content-Type: application/protobuf
	FindOneContactRequest
[响应]
	Content-Type: application/protobuf
	FindOneContactResponse

4. 代码实现客户端

这里我们只实现了新增⼀个联系⼈模块,完整代码在码云.

add_contact.proto

syntax="proto3";
package add_contact;

message AddContactReq{
  string name = 1 ;
  int32 age = 2 ;
  message Phone{
    string number = 1;
    enum PhoneType{
      MP=0;
      TEL=1;
    }
    PhoneType type =2;
  }
  repeated Phone phones = 3;
}

message AddContactResp{
  bool success = 1; // 添加联系人是否成功
  string error_desc =2 ;// 错误信息
  string uid =3 ; // 联系人序号
}

ContactException.h:定义异常类

#include<string>
class ContactException
{
  private:
  std::string message;

  public:
  ContactException(std::string str="A problem"):message(str){}

  std::string what()const {return message; }
};

main.cc

#include "httplib.h"
#include "contactException.h"
#include "add_contact.pb.h"
#include <iostream>
#include <string>
using namespace std;
using namespace httplib;

const string IP = "127.0.0.1"; // 监听所有的ip
// const string IP = "192.139.99.192";
const int PORT = 6666;
void menu();
void addContact();
void buildAddContactReq(add_contact::AddContactReq *req);
int main()
{
  while (true)
  {
    enum OPTION{QUIT = 0,ADD,DEL,FIND_ALL,FIND_ONE};

    menu();
    cout << "--------请选择:";
    int choose;
    cin >> choose;
    cin.ignore(256, '\n');
    try
    {
      switch (choose)
      {
      case ADD:
        addContact();
        break;
      case DEL:
        break;
      case FIND_ALL:
        break;
      case FIND_ONE:
        break;
      case QUIT:
        cout << "程序退出" << endl;
        exit(0);
        break;
      default:
        break;
      }
    }
    catch (ContactException &e)
    {
      cout << "--->操作通讯录时发生异常" << endl
           << "--->异常信息:" << e.what() << endl;
    }
  }
}
void addContact()
{
  Client client(IP, PORT);
  //构造 req
  add_contact::AddContactReq req;
  buildAddContactReq(&req);

  // 序列化 req
  string req_str;
  if (!req.SerializePartialToString(&req_str))
  {
    throw ContactException("req 序列化失败");
  }

  // 发起post调用
  auto ret = client.Post("/contacts/add", req_str, "Content-Type: application/protobuf");
  if (!ret)
  {
    string err_desc;
    err_desc.append("Post /contacts/add 请求失败! 错误信息:")
        .append(/*httplib::to_string(ret.error())当前httplib没有该函数*/
                to_string(ret.error()));
    throw ContactException(err_desc);
  }
  // 方序列号resp
  add_contact::AddContactResp resp;
  bool parse = resp.ParseFromString(ret->body);
  if(!parse){
   throw ContactException("反序列化失败!");
  }
  if (ret->status != 200 && !parse)
  {
    string err_desc;
    err_desc.append("/contacts/add 调用失败")
        .append(std::to_string(ret->status))
        .append("(")
        .append(ret->reason)
        .append(")")
        .append(resp.error_desc());
    throw ContactException(err_desc);
  }
  else if (ret->status != 200)
  {
    string err_desc;
    err_desc.append("/contacts/add 调用失败")
        .append(std::to_string(ret->status))
        .append("(")
        .append(ret->reason)
        .append(")")
        .append("错误信息:")
        .append(resp.error_desc());

    throw ContactException(err_desc);
  }
  else if (!resp.success())
  {
    string err_desc;
    err_desc.append("/contacts/add 结果异常\t异常原因")
        .append(std::to_string(ret->status))
        .append("(")
        .append(resp.error_desc())
        .append(")");
    throw ContactException(err_desc);
  }

  // 结果打印

  cout<<"新添加的联系人"<<resp.uid()<<"成功"<<endl;

}

void buildAddContactReq(add_contact::AddContactReq* req)
{
  cout << "请输入姓名:";
  string name;
  getline(std::cin,name);
  cout << "请输入年龄:";
  int age;
  cin >> age;
  cin.ignore(256, '\n');
  int i = 1;
  req->set_age(age);
  req->set_name(name);
  while (true)
  {
    cout << "请输入手机号码" << i++ << ":";
    string number;
     getline(std::cin,number);
    if (number.empty())
    {
      //cout << "输入联系人完成\n";
      break;
    }
    add_contact::AddContactReq_Phone *phone = req->add_phones();
    phone->set_number(number);
    cout << "请输入手机类型(ML:0,TEL:1):";
    int type;
    cin >> type;
    cin.ignore(256, '\n');
    phone->set_type((add_contact::AddContactReq_Phone_PhoneType)type);
  }
}
void menu()
{
  std::cout << "-----------------------------------------------------" << std::endl
            << "--------------- 请选择对通讯录的操作 ----------------" << std::endl
            << "------------------ 1、新增联系⼈ --------------------" << std::endl
            << "------------------ 2、删除联系⼈ --------------------" << std::endl
            << "------------------ 3、查看联系⼈列表 ----------------" << std::endl
            << "------------------ 4、查看联系⼈详细信息 ------------" << std::endl
            << "------------------ 0、退出 --------------------------" << std::endl
            << "-----------------------------------------------------" << std::endl;
}

5. 代码实现服务端

add_contact.proto

syntax="proto3";
package add_contact;

message AddContactReq{
  string name = 1 ;
  int32 age = 2 ;
  message Phone{
    string number = 1;
    enum PhoneType{
      MP=0;
      TEL=1;
    }
    PhoneType type =2;
  }
  repeated Phone phone = 3;
}

message AddContactResp{
  bool success = 1; // 添加联系人是否成功
  string error_desc =2 ;// 错误信息
  string uid =3 ; // 联系人唯一序号
}

utils.h 工具类


#include <iostream>
#include <sstream>
#include <random>

namespace Contact_Utils
{
  class Utils
{
public:

	/// 生成一个唯一标识符,用于赋值uid
  static std::string generateUUID(size_t len)
  {
    // 使用随机数生成器生成随机数种子
    std::random_device device;
    std::mt19937 generator(device());

    // 使用16进制表示的48位的随机数

    // std::hex是C++中的一个std::ios_base标志,用于指定输出流以十六进制形式输出整数。当使用该标志时,输出流中的整数将以十六进制表示。例如,当输出整数0x123时,使用std::hex标志会将其输出为字符串"123",而不是"291"。在上述示例代码中,std::hex被用于指定std::ostringstream以十六进制形式输出整数,从而生成16进制表示的48位唯一标识符。
    std::ostringstream uuid;
    uuid << std::hex;
    for (size_t i = 0; i < len; ++i)
    {
      uuid << (generator() & 0xf);
    }

    return uuid.str();
  }
};
}

main.cc文章来源地址https://www.toymoban.com/news/detail-518101.html

#include "httplib.h"
#include "add_contact.pb.h"
#include "contactException.h"
#include <iostream>
#include <string>
#include"utils.h"
using namespace std;
using namespace httplib;
const string IP = "127.0.0.1";

// const string IP = "0.0.0.0";
// const string IP = "192.139.99.192";

const int PORT = 6666;

void printContact(add_contact::AddContactReq &request);
int main()
{
  // 接收请求
  Server svr;

  // 处理请求
  // 注册post的回调函数
  svr.Post("/contacts/add", [](const Request &req, Response &resp)
           {
            cout<<"收到Post请求!"<<endl;
            add_contact::AddContactResp response;
            add_contact::AddContactReq  request;
             try
             {
              if(!request.ParseFromString(req.body)){
                throw ContactException("方序序列化失败!");
              }
              // 持久化联系人
              printContact(request);

              // 构造 response : res.body
              response.set_success(true);
              response.set_uid(Contact_Utils::Utils::generateUUID(12));
              // 序列化 response
              string response_str;
              if(!response.SerializePartialToString(&response_str))
              {
                throw ContactException("序列化失败!");
              }

              resp.status=200;
              resp.body=response_str;
              resp.set_header("Content-Type","application/protobuf");

             }
             catch (ContactException &e)
             {
                  resp.status=500;
                  response.set_success(false);
                  response.set_error_desc(e.what());
                  string response_str;
                  if(response.SerializePartialToString(&response_str)){
                    resp.body=response_str;
                    resp.set_header("Content-Type","application/protobuf");
                  }
                  cout<<"/contacts/add 发生异常,异常信息:"<<e.what()<<endl;
             } });
  // 生成resp,

  // 并发送resp,

  svr.listen(IP.c_str(), PORT);
}

void printContact(add_contact::AddContactReq &req)
{
  cout << "添加联系人成功\n-->姓名:" << req.name() << "-->年龄:" << req.age() << endl;
  cout << "-----电话号码-------" << endl;
  for (int i = 0; i < req.phone_size(); i++)
  {
    
    cout << "电话" << i << ":" << req.phone(i).number() << "(type:" <<req.phone(i).PhoneType_Name(req.phone(i).type())<< ")" << endl;
  }
}

到了这里,关于网络通讯录服务器的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C语言】动态内存管理基础知识——动态通讯录,如何实现通讯录容量的动态化

    动态内存管理的函数有:malloc,calloc,ralloc,free,本文讲解动态内存函数和使用,如何进行动态内存管理,实现通讯录联系人容量的动态化,对常见动态内存错误进行总结。                           ✨  猪巴戒 :个人主页✨                 所属专栏 :《C语言进阶》

    2024年02月04日
    浏览(68)
  • 通讯录管理系统

    作者:狮子也疯狂 专栏:《项目集锦》 坚持做好每一步,幸运之神自然会驾凌在你的身上 该项目是用于日常生活中记录联系人信息的一款智能小工具。实现了对联系人的姓名、年龄、性别、电话号码、住址的添加及修改、查找、删除、排序等功能。该项目是以 Windows 控制台

    2024年02月05日
    浏览(54)
  • 【通讯录】--C语言

    💐 🌸 🌷 🍀 🌹 🌻 🌺 🍁 🍃 🍂 🌿 🍄🍝 🍛 🍤 📃 个人主页 :阿然成长日记 👈点击可跳转 📆 个人专栏: 🔹数据结构与算法🔹C语言进阶 🚩 不能则学,不知则问,耻于问人,决无长进 🍭 🍯 🍎 🍏 🍊 🍋 🍒 🍇 🍉 🍓 🍑 🍈 🍌 🍐 🍍 用c语言实现一个通讯

    2024年02月15日
    浏览(59)
  • Java实现通讯录

    前言:通讯录非常适合前期需要练习语法的小伙伴,效果非常好。 水再浑浊,只要长久沉淀,依然会分外清澈。 人再愚钝,只要足够努力,一样能改写命运。 此通讯录的功能包括: 1.删除联系人 2.增加联系人 3.展示已有联系人 4.查找联系人 5.排序联系人 6.退出通讯录 联系人

    2024年02月11日
    浏览(55)
  • python-手机通讯录

    手机通讯录 通讯录是记录了联系人姓名和联系方式的名录,手机通讯录是最常见的通讯录之一,人们可以在通讯录中通过姓名查看相关联系人的联系方式等信息,也可以在其中新增联系人,或修改、删除联系人信息。 本实例要求编写程序,实现具备添加、查看、修改以及删

    2024年02月06日
    浏览(69)
  • C语言通讯录

            在本博客中,我们将介绍如何使用C语言构建一个基本的通讯录。主要涉及C语言的指针、结构体、动态内存管理、文件操作等方面的知识。我们还将学习如何使用C语言的各种功能和技巧来实现通讯录的各种操作,如添加联系人、编辑联系人、删除联系人和搜索联系

    2024年02月16日
    浏览(57)
  • 【C语言】通讯录

    目录 一、关于通讯录 二、代码逻辑 三、通讯录实现 1.菜单设计 2.逻辑主要功能设计 3.增加联系人功能实现 4.显示全部联系人信息   5.删除联系人 6.查找联系人 7.修改联系人信息 8.对联系人进行排序  9.一键清空所有联系人 四、完整源码 test.c contact.c contact.h 在通讯录中,我

    2024年02月08日
    浏览(63)
  • 手机通讯录 python

    person_info = [] print(“=” * 20) print(‘欢迎使用通讯录:’) print(“1.添加联系人”) print(“2.查看通讯录”) print(“3.删除联系人”) print(“4.修改联系人信息”) print(“5.查找联系人”) print(“6.退出”) print(“=” * 20) while True: per_dict = {} fun_num = input(‘请输入功能序号:’) if fun_num

    2024年02月06日
    浏览(38)
  • C语言—通讯录

    通讯录中是存放人的信息的,人的信息包括:姓名、年龄、性别、电话、住址。可以把人的信息定义成结构体,因为每个联系人的信息都有这几个要素。 注:用#define定义的标识符常量,使用修改时可以降低维护成本。如联系人的信息中名字的宽度为20,以便之后需要修改时只

    2024年02月05日
    浏览(72)
  • C语言——通讯录

    相信大家都有过通讯录,今天我来带大家实现以下最简单的通讯录,通过本篇文章,相信可以让大家对C语言有进一步的认识。 话不多说,我们先放函数的实现  是不是看到这里会感到很害怕??不用怕,跟着我的思路,你也可以实现它,我带着你一步一步实现每一个功能  

    2024年02月13日
    浏览(51)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包