网络电子词典

news2024/9/28 3:29:00

一、项目要求:

1. 登录注册功能,不能重复登录,重复注册

2. 单词查询功能

3. 历史记录功能,存储单词,意思,以及查询时间

4. 基于TCP,支持多客户端连接

5. 采用数据库保存用户信息与历史记录

格式要求: 1. main函数只跑逻辑,不允许跑功能代码 2. 功能代码封装成函数 

二、流程示意图

服务器:

客户端:

 

 三、实现代码

server.c代码部分:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sqlite3.h>
#include <signal.h>
#include <time.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define DATABASE "my.db"
#define N  16

enum CMD{
    R=1,
    L,
    Q,
    H
};

typedef struct sockaddr SA;

typedef struct
{
    int type;
    char name[N];
    char data[256];   // password or word
} MSG;

static sqlite3 *db;   //数据库句柄指针

void do_register(int connfd, MSG *pbuf)    //用户注册模块
{
    char sqlstr[128];
    char *errmsg;

    sprintf(sqlstr, "insert into usr values ('%s', '%s')", pbuf->name, pbuf->data);
    printf("%s\n", sqlstr);
    if (sqlite3_exec(db, sqlstr, NULL, NULL, &errmsg) != SQLITE_OK)   //执行SQL语句并处理结果
    {
        sqlite3_free(errmsg);
        sprintf(pbuf->data, "user %s already exist!!!", pbuf->name);
    }
    else
    {
        strncpy(pbuf->data, "OK", 256);
    }
    send(connfd, pbuf, sizeof(MSG), 0);
}


void do_login(int connfd, MSG *pbuf)  //用户登录模块
{
    char sqlstr[128];    //拼接后的查询语句
    char *errmsg, **result; //存放错误信息和查询到的结果
    int nrow, ncolumn;   //查询到的行数和列数

    sprintf(sqlstr, "select * from usr where name = '%s' and pass = '%s'", pbuf->name, pbuf->data);
    if (sqlite3_get_table(db, sqlstr, &result, &nrow, &ncolumn, &errmsg) != SQLITE_OK)  
    {
        printf("error : %s\n", errmsg);
        sqlite3_free(errmsg);
    }

    if (nrow == 0)   //查询到的条数为0条
    {
        strncpy(pbuf->data, "name or password is wrong!!!", 256);
    }
    else  //查询到的所需信息
    {
        strncpy(pbuf->data, "OK", 256);
    }
    send(connfd, pbuf, sizeof(MSG), 0);
    sqlite3_free_table(result);

    return;
}

#if 0
int do_searchword(int connfd, MSG *pbuf)   //查询单词模块调用的查询单词函数方式一
{
    FILE *fp;
    char line[300];
    char *p;
    int len, result;

    len = strlen(pbuf->data); //要查询的单词长度
    if ((fp = fopen("dict.txt", "r")) == NULL)  //打开保存单词文件
    {
        strcpy(pbuf->data, "dict on server can't be opened :(");
        send(connfd, pbuf, sizeof(MSG), 0);
        return 0;
    }
    printf("query word is %s\n", pbuf->data); //打印一下客户端要查询的单词
    while (fgets(line, 300, fp) != NULL)  //读文件,每次读300个字符存放到字符数组line中,来查询单词
    {
        result = strncmp(pbuf->data, line, len); //比较一下要查询的单词和line中读取到的文件中每行的单词信息进行比较,比较的长度大小为要查询的单词大小
        if (result > 0) continue; //大于0,说明要查询的单词大小大于读取的单词大小,则继续下一行,比如abandon ,查询abc,则跳入下一行
        if (result < 0 || line[len] != ' ') break;//小于0,则说明要查找的单词小于读取的单词大小,但之前的都已比较完毕,则查询失败。不是空格也不行,比如查询ab,而读取的是abandon.

        p = line + len; //指针指到单词末尾
        while (*p == ' ') p++;  //指针移动到注释之前
        strcpy(pbuf->data, p); //拷贝单词注释
        fclose(fp);     //
        return 1;
    }
    fclose(fp);

    return 0;
}
#else
int do_searchword(int connfd, MSG *pbuf)   //查询单词模块调用的查询单词函数方式二
{
    char buf[128] = {0};

    printf("query word is %s\n", pbuf->data);
    sprintf(buf, "select * from dic where word='%s'", pbuf->data);

    char *errmsg;
	char **resultp;
	int n_row, n_cloum;

	if (sqlite3_get_table(db, buf, &resultp, &n_row, &n_cloum, &errmsg) != SQLITE_OK) 
    {
        printf("err:%s\n", errmsg);
        return -1;
    }

    if(n_row != 1)
    {
        return 0;
    }

    char **p;
    p = resultp + n_cloum;

    // printf("data=%s\n", p[1]);

    strcpy(pbuf->data, p[1]);

    return 1;
}
#endif

void get_date(char date[])  //获取系统时间
{
    time_t t;
    struct tm *tp;

    time(&t);  //获取到一个时间秒数

    tp = localtime(&t);//进行时间格式转换
    /*
	//字符串拼接
    sprintf(date, "%d-%02d-%02d %02d:%02d:%02d", tp->tm_year+1900,
            tp->tm_mon+1, tp->tm_mday, tp->tm_hour, tp->tm_min, tp->tm_sec);
    */
    strftime(date, 64, "%Y-%m-%d %H:%M:%S", tp);

    return;
}

void do_query(int connfd, MSG *pbuf)   //查询单词模块
{
    char sqlstr[128], *errmsg;
    int len, result, found = 0;
    char date[64], word[64];

    strcpy(word, pbuf->data);  //拿出pbuf结构体中要查询的单词
    found = do_searchword(connfd, pbuf);  //调用do_searchword函数模块去查询
    if(found)  //返回值非0,表示找到了该单词,则将用户查询信息(包括用户名,时间和单词)插入到历史记录表中
    {
        get_date(date);//调用时间函数获取时间
        sprintf(sqlstr, "insert into record values ('%s', '%s', '%s')", pbuf->name, date, word);
        if (sqlite3_exec(db, sqlstr, NULL, NULL, &errmsg) != SQLITE_OK)
        {
            printf("error : %s\n", errmsg);
            sqlite3_free(errmsg);
        }
    }
    else   //返回值为0,表示没有找到,发送没到到提示信息
    {
        strcpy(pbuf->data, "not found\n");
    }
    send(connfd, pbuf, sizeof(MSG), 0);   //把查找到的信息发送给客户端
    return;
}

//得到查询结果,并将历史记录发送给客户端
int history_callback(void *arg, int f_num, char **f_value, char **f_name)
{
    int connfd;
    MSG buf;

    connfd = *(int *)arg;
    sprintf(buf.data, "%s : %s", f_value[1], f_value[2]);
    send(connfd, &buf, sizeof(buf), 0);

    return 0;
}

void do_history(int connfd, MSG *pbuf)   //查询历史记录模块
{
    char sqlstr[128], *errmsg;

    sprintf(sqlstr, "select * from record where name = '%s'", pbuf->name);
    if (sqlite3_exec(db, sqlstr, history_callback, (void *)&connfd, &errmsg) != SQLITE_OK) //查询数据库
    {
        printf("error : %s\n", errmsg);
        sqlite3_free(errmsg);
    }
    pbuf->data[0] = '\0';         //所有的记录发送完毕之后,给客户端发送一个结束信息
    send(connfd, pbuf, sizeof(MSG), 0);

    return;
}



void do_client(int connfd)     //处理客户端请求主模块,调用函数完成所需功能
{
    MSG buf;
    while(recv(connfd, &buf, sizeof(buf), 0) > 0){
        switch (buf.type) {
        case R:
            printf("will reg\n");
            do_register(connfd, &buf);
            break;
        case L:
            printf("will login\n");
            do_login(connfd, &buf);
            break;
        case Q:
            printf("will query\n");
            do_query(connfd, &buf);
            break;
        case H:
            printf("will history\n");
            do_history(connfd, &buf);
            break;
        default:
            break;
        }
    }

    exit(0);
}


int main(int argc, char *argv[])    //主函数
{
    int listenfd, connfd;
    struct sockaddr_in myaddr;
    pid_t pid;
    MSG buf;
    

    if (argc < 3)
    {
        printf("Usage : %s <ip> <port>\n", argv[0]);
        exit(-1);
    }

    //打开数据库
    if(sqlite3_open(DATABASE, &db) < 0){
        printf("fail to sqlite3_open : %s\n", sqlite3_errmsg(db));
        return -1;
    }

    //创建服务器socket
    listenfd = socket(PF_INET, SOCK_STREAM, 0);
    if(listenfd < 0){
        perror("fail to socket");
        exit(-1);
    }
    bzero(&myaddr, sizeof(myaddr));
    myaddr.sin_family = PF_INET;
    myaddr.sin_port = htons(atoi(argv[2]));
    myaddr.sin_addr.s_addr = inet_addr(argv[1]);
    if (bind(listenfd, (SA *)&myaddr, sizeof(myaddr)) < 0)
    {
        perror("fail to bind");
        exit(-1);
    }

    // 将套接字设为监听模式
    if (listen(listenfd, 5) < 0)
    {
        perror("fail to listen");
        exit(-1);
    }

    //并发服务器模型
    while(1){
		
        if((connfd = accept(listenfd, NULL, NULL)) < 0){
            perror("fail to accept");
            exit(-1);
        }
		
        pid = fork();
        if(pid == -1){
            perror("fail to fork\n");
            exit(-1);
        }
        else if(pid == 0){ //子进程处理客户端具体的消息
            printf("a user comming\n");
            do_client(connfd);
        }else{  //父进程用于接收客户端的请求
            close(connfd);
        }
    }
}

client.c代码部分:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sqlite3.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#define N  16
typedef struct sockaddr SA;

enum CMD{
    R=1,
    L,
    Q,
    H
};


typedef struct
{
    int type;
    char name[N];
    char data[256];   // password or word
} MSG;

void do_register(int sockfd, MSG *pbuf)   //一级菜单用户注册模块
{
    pbuf->type = R;
    printf("input name : ");
    scanf("%s", pbuf->name);
    printf("input password : ");
    scanf("%s", pbuf->data);
    send(sockfd, pbuf, sizeof(MSG), 0);
    recv(sockfd, pbuf, sizeof(MSG), 0);
    printf("register : %s\n\n", pbuf->data);
}

int do_login(int sockfd, MSG *pbuf)     //一级菜单用户登录模块
{
    pbuf->type = L;
    printf("input name : ");
    scanf("%s", pbuf->name);
    printf("input password : ");
    scanf("%s", pbuf->data);
    send(sockfd, pbuf, sizeof(MSG), 0);
    recv(sockfd, pbuf, sizeof(MSG), 0);
    if(strncmp(pbuf->data, "OK", 3) == 0){    //3个字节:O K /0
        printf("login sucess\n\n");
        return 1;
    }else{
        printf("login : %s\n\n", pbuf->data);
        return 0;
    }
}

void do_history(int sockfd, MSG *pbuf)   //二级菜单查询单词历史记录模块
{
    pbuf->type = H;
    send(sockfd, pbuf, sizeof(MSG), 0);
    while ( 1 )
    {
        recv(sockfd, pbuf, sizeof(MSG), 0);
        if (pbuf->data[0] == '\0') break;
        printf("%s\n", pbuf->data);
    }

    return;
}

void do_query(int sockfd, MSG *pbuf)  //二级菜单查询单词模块
{
    pbuf->type = Q;
    while ( 1 )
    {
        printf("input word(# to quit) : ");
        scanf("%s", pbuf->data);
        if (strcmp(pbuf->data, "#") == 0) break; //输入#返回上一级菜单
        send(sockfd, pbuf, sizeof(MSG), 0); //将要查询的单词发送给服务器
        recv(sockfd, pbuf, sizeof(MSG), 0); //等待接收服务器传递回来的单词信息
        printf("%s\n", pbuf->data);
        printf("\n");
    }

    return;
}

void enter_query(int sockfd, MSG *buf)    //二级菜单主页面
{
    int input;
    char cleanbuf[64];

    while (1) {
        printf("***********************************************\n");
        printf("* 1: query_word   2: history_record   3: quit *\n");
        printf("***********************************************\n");
        printf("please choose : ");
        //获取用户输入
        if(scanf("%d", &input) == 0){
            fgets(cleanbuf, 64, stdin);  //类型错误需要重新清除输入缓冲区
            continue;
        }

        switch (input) {
        case 1:
            printf("\n");
            do_query(sockfd, buf);
            printf("\n");
            break;
        case 2:
            printf("\n");
            do_history(sockfd, buf);
            printf("\n");
            break;
        case 3:
            return;
        default:
            break;
        }
    }
}

int main(int argc, char *argv[])      //主函数
{
	int sockfd, login = 0;
	struct sockaddr_in servaddr;
	MSG buf;
	char clean[64];

	if (argc < 3)
	{
		printf("Usage : %s <serv_ip> <serv_port>\n", argv[0]);
		exit(-1);
	}

    //创建客户端socket
	if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
	{
		perror("fail to socket");
		exit(-1);
	}

	bzero(&servaddr, sizeof(servaddr));
	servaddr.sin_family = PF_INET;
	servaddr.sin_port = htons(atoi(argv[2]));
	servaddr.sin_addr.s_addr = inet_addr(argv[1]);

    //连接服务器
	if (connect(sockfd, (SA *)&servaddr, sizeof(servaddr)) < 0)
	{
		perror("fail to connect");
		exit(-1);
	}

    int input;
    char cleanbuf[64];
    while(1){
        printf("************************************\n");
        printf("* 1: register   2: login   3: quit *\n");
        printf("************************************\n");
        printf("please choose : ");
        //获取用户输入
        if(scanf("%d", &input) == 0){
            fgets(cleanbuf, 64, stdin);  //类型错误需要重新清除输入缓冲区
            continue;
        }

        switch (input) {
        case 1:
            do_register(sockfd, &buf);
            break;
        case 2:
            if(do_login(sockfd, &buf) == 1){  //成功
                enter_query(sockfd, &buf);
            }
            break;
        case 3:
            close(sockfd);
            exit(0);
            break;
        default:
            break;
        }
    }
}

Makefile代码部分:

all:server client

server:server.c
	gcc $< -o $@ -l sqlite3

client:client.c
	gcc $< -o $@
	
clean:
	rm server client	

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

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

相关文章

002 编程是什么?

魔法师:在这个充满魔法和奇迹的数字时代,你是否好奇过计算机是如何运作的?当你用手机玩游戏、在电脑上浏览网页、看动画电影,你是否想过这背后的秘密是什么?别担心,今天我们将揭开这神秘的面纱,一起来探索编程的神奇世界! 编程,简单地说,就是一种让计算机执行任务的…

第五章:认证和动态菜单功能【基于Servlet+JSP的图书管理系统】

一、登录功能 1.认证实现 53-图书管理系统-登录功能-认证处理 首先完成最基础的登录功能&#xff0c;也就是在登录页面通过表单提交账号和密码到Servlet中。做相关的校验。给出不同的反应。 然后对应的Servlet中的处理逻辑 WebServlet(name "loginServlet",urlPatt…

软件测试的常用概念

目录 需求 需求和软件测试人员的关系 需求是侧式人员进行软件测试工作的依据,需要通过软件需求,来设计测试用例 软件的生命周期 在每个阶段,测试人员需要做什么事? 软件测试的生命周期 BUG 什么是bug? 如何描述一个bug? bug的级别 bug的生命周期: 调试和测试的区…

Linux虚拟机安装(Ubuntu 20)

最近这段时间使用VMWare安装了一下Ubuntu版本的Linux虚拟机&#xff0c;在这里记录一下安装时参考的文章以及需要注意的细节 参考链接&#xff1a; 虚拟机&#xff08;VMware&#xff09;安装Linux&#xff08;Ubuntu&#xff09;安装教程 VMware虚拟机下安装Ubuntu20.04&…

这所985重大科目变更!新增专硕可考信号!

一、学校及专业介绍 重庆大学&#xff08;ChongqingUniversity&#xff0c;CQU&#xff09;&#xff0c;简称“重大”&#xff0c;是教育部直属的全国重点大学&#xff0c;是国家“211工程”和“985工程”重点建设的高水平研究型综合性大学、国家“世界一流大学建设高校&#…

跨境电商系统开发:打破国界壁垒,拓展全球市场

拓展全球市场的必然选择 随着国际贸易水平的不断提升和全球市场的日益开放&#xff0c;跨境电商作为一种高效的贸易模式&#xff0c;受到了越来越多电商企业的关注和青睐。跨境电商不仅可以打破国界壁垒和时差限制&#xff0c;还可以在全球市场上实现销售的拓展&#xff0c;带来…

SVN 项目管理笔记

SVN 项目管理笔记 主要是介绍 SVN 管理项目的常用操作&#xff0c;方便以后查阅&#xff01;&#xff01;&#xff01; 一、本地项目提交到SVN流程 在SVN仓库下创建和项目名同样的文件夹目录&#xff1b;选中本地项目文件&#xff0c;选择SVN->checkout,第一个是远程仓库项…

FOC之SVPWM学习笔记

一、参考资料 【自制FOC驱动器】深入浅出讲解FOC算法与SVPWM技术 - 知乎FOC入门教程_zheng是在下的博客-CSDN博客DengFOC官方文档技术干货 |【自制】FOC驱动板 二、FOC控制算法流程框图 在FOC控制中主要用到三个PID环&#xff0c;从内到外依次是&#xff1a;电流环、速度环、位…

DevExpress WinForms数据编辑器组件,提供丰富的数据输入样式!(二)

DevExpress WinForms超过80个高影响力的WinForms编辑器和多用途控件&#xff0c;从屏蔽数据输入和内置数据验证到HTML格式化&#xff0c;DevExpress数据编辑库提供了无与伦比的数据编辑选项&#xff0c;包括用于独立数据编辑或用于容器控件(如Grid, TreeList和Ribbon)的单元格。…

第60步 深度学习图像识别:误判病例分析(Pytorch)

基于WIN10的64位系统演示 一、写在前面 上期内容基于Tensorflow环境做了误判病例分析&#xff08;传送门&#xff09;&#xff0c;考虑到不少模型在Tensorflow环境没有迁移学习的预训练模型&#xff0c;因此有必要在Pytorch环境也搞搞误判病例分析。 本期以SqueezeNet模型为…

Java不用加减乘除做加法(图文详解)

目录 1.题目描述 2.题解 分析 具体实现 1.题目描述 写一个函数&#xff0c;求两个整数之和&#xff0c;要求在函数体内不得使用、-、*、/四则运算符号。 示例 输入&#xff1a;1 2 输出&#xff1a;3 2.题解 分析 不能使用加减乘除四则运算符&#xff0c;那我们只能考虑…

论文笔记:从不平衡数据流中学习的综述: 分类、挑战、实证研究和可重复的实验框架

0 摘要 论文&#xff1a;A survey on learning from imbalanced data streams: taxonomy, challenges, empirical study, and reproducible experimental framework 发表&#xff1a;2023年发表在Machine Learning上。 源代码&#xff1a;https://github.com/canoalberto/imba…

多环境开发

多环境 1、多环境开发&#xff08;YAML版&#xff09; 小结&#xff1a; 多环境开发需要设置若干种常用环境&#xff0c;例如开发、生产、测试环境yaml格式中设置多环境使用—区分环境设置边界每种环境的区别在于加载的配置属性不同启用某种环境时需要指定启动时使用该环境 …

跨境电商儿童沙画办理EN71测试标准

儿童沙画就是小孩子玩的那种用彩色沙子或者彩色墨水&#xff0c;在有图形轮廓的纸片上去绘画&#xff0c;可以按照儿童沙画底板上的人物轮廓线条&#xff0c;动物线条&#xff0c;风景线条&#xff0c;动漫线条&#xff0c;去添加自己喜欢的颜色&#xff0c;让单调的线条变成自…

请不要将数据中台做成BI!

近年来&#xff0c;随着数据驱动决策的重要性日益凸显&#xff0c;数据管理的概念也不断演进。数据中台作为一种新兴的数据管理模式&#xff0c;旨在集中组织和整合数据资源&#xff0c;为企业提供更高效、更智能的数据支持。然而&#xff0c;有时候在构建数据中台的过程中&…

C语言入门 Day_9 条件判断

目录 前言&#xff1a; 1.if判断 2.else判断 3.易错点 4.思维导图 前言&#xff1a; 我们知道比较运算和逻辑运算都会得到一个布尔型的数据&#xff0c;要么为真&#xff08;true&#xff09;&#xff0c;要么为假&#xff08;false&#xff09;。 今天我们来学习真和假在…

【性能优化】使用Perfetto定位应用启动性能的瓶颈

Android应用启动优化相关的文章已经有很多人都写过了&#xff0c;但是主要都是聚焦在&#xff0c;为了启动性能都做了哪些改动上&#xff0c;少见有文章会说应该如何分析、识别应用的启动性能。 本篇文章将会结合我个人对Perfetto的实际使用经历&#xff0c;讲解车载应用的启动…

Python“牵手”当当网商品详情API接口运用场景及功能介绍,当当网API接口申请指南

当当网是全球知名的综合性网上购物商城&#xff0c;由国内著名出版机构科文公司、美国老虎基金、美国IDG集团、卢森堡剑桥集团、亚洲创业投资基金&#xff08;原名软银中国创业基金&#xff09;共同投资成立。当当网是北京当当网信息技术有限公司营运的一家中文购物网站&#x…

【校招VIP】产品行测考点之图的推理和分析

考点介绍&#xff1a; 大厂产品校招笔试里经常会出现行测的考察&#xff0c;而图的推理是行测里面稍微有难度的一部分。因为时间有限&#xff0c;很多同学因为没有解题思路而丢分。 『产品行测考点之图的推理和分析』相关题目及解析内容可点击文章末尾链接查看&#xff01; 一…

【Unity小技巧】最简单的UI设置适配方案,萌新必看

文章目录 前言导入素材开始一、页面适配方案二、侧边栏适配方法一方法二 参考完结 前言 这期来讲一个简单的UI设计方案&#xff0c;很多同学可能搞不懂锚点、轴心这些概念&#xff0c;导致做好的UI在别人的设备上&#xff0c;乱跑或者是重叠&#xff0c;或者是参加游戏老发时间…