网络电子词典

一、项目要求:

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  16enum 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;
}
#endifvoid 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;}//创建服务器socketlistenfd = 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 /0printf("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);}//创建客户端socketif ((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 clientserver:server.cgcc $< -o $@ -l sqlite3client:client.cgcc $< -o $@clean:rm server client	

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

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

相关文章

java: 无法访问org.springframework.boot.SpringApplication 错误的类文件

项目场景&#xff1a; 提示&#xff1a;这里简述项目相关背景&#xff1a; 错误1&#xff1a; java: 无法访问org.springframework.boot.SpringApplication 错误的类文件: /D:/Software/env-java/apache-maven-3.6.1/repository/org/springframework/boot/spring-boot/3.1.2/sp…

C#,《小白学程序》第四课:数学计算

1 文本格式 /// <summary> /// 《小白学程序》第四课&#xff1a;数学计算 /// 这节课超级简单&#xff0c;就是计算成绩的平均值&#xff08;平均分&#xff09; /// 这个是老师们经常做的一件事。 /// </summary> /// <param name"sender"></…

图片换脸-->>视频换脸-->>直播换脸

资源网站&#xff1a;https://tianfeng.space/ 个人娱乐&#xff0c;切勿作恶 下载 ​ 网盘&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1DHMY1mCXpT0OtpmlvIoMKA 提取码&#xff1a;nf57 使用 下载解压后&#xff0c;打开 第一个就是你要替换的人脸&#xff0c;…

MyCAT命令行监控

9066端口 &#xff0c;用mysql命令行连接 Mysql –utest –ptest –P9066 show help 可显示所有相关管理命令 显示后端物理库连接信息&#xff0c;包括当前连接数&#xff0c;端口 Show backend Show connection 显示当前前端客户端连接情况&#xff0c;已经网络流量信息、…

再见 Xshell替代工具Tabby

替代Xshell 之前经常使用Xshell来操作Linux虚拟机&#xff0c;基本上是够用了。但是Xshell免费使用只供非商业用途&#xff0c;而且如果你想用FTP来进行文件传输的话&#xff0c;还需单独下载Xftp。 无意中发现了另一款开源的终端工具Tabby&#xff0c;它直接集成了SFTP功能&…

Qt XML文件解析 QDomDocument

QtXml模块提供了一个读写XML文件的流&#xff0c;解析方法包含DOM和SAX,两者的区别是什么呢&#xff1f; DOM&#xff08;Document Object Model&#xff09;&#xff1a;将XML文件保存为树的形式&#xff0c;操作简单&#xff0c;便于访问。 SAX&#xff08;Simple API for …

渗透测试工具ZAP入门教程(1)-安装和快速开始

介绍 ZAP Zed Attack Proxy&#xff08;ZAP&#xff09;是一个免费的开源渗透测试工具&#xff0c;在 软件安全项目 &#xff08;SSP&#xff09;。ZAP 专为测试 Web 应用程序而设计&#xff0c;既灵活又可扩展。 ZAP的核心是所谓的“中间人代理”。它位于测试人员的浏览器和…

go学习-指针 标识符

指针&#xff0c;以及标识符 1.指针 &#xff08;1&#xff09;.基本介绍 1&#xff09;基本数据类型&#xff0c;变量存的值&#xff0c;也叫值类型 2&#xff09;获取变量的地址用&&#xff0c;比如 var num int ,获取num的地址&#xff1a;&num 3)指针类型&…

Yolo系列-yolov2

YOLO-V2 更快&#xff01;更强&#xff01; YOLO-V2-BatchNormalization BatchNormalization&#xff08;批归一化&#xff09;是一个常用的深度神经网络优化技术&#xff0c;它可以将输入数据进行归一化处理&#xff0c;使得神经网络更容易进行学习。在YOLOv2中&#xff0c;B…

promise

promise 属于事件循环的微任务&#xff0c;具体详见&#xff1a;事件循环 Promise 语法: const p1 new Promise((reslove,reject)>{console.log(2);reslove(1) }).then((data)>{console.log(3);console.log(data) }).catch((data)>{console.log(3); }) promise.th…

基于Milvus Cloud的相似穿搭推荐系统教程——图像分割

作为一个大众眼中的“非典型程序员”,我喜欢拥抱时尚和潮流,比如我经常在演讲时穿粉色裤子,这甚至已经成为一个标志性打扮。某天又逢主题演讲日,我站在衣柜前挑选上衣的时候,忽然灵光乍现:有没有可能借助 Milvus Cloud找到和我穿搭风格最为相似的明星呢? 这个想法在我脑…

Oracle给表空间添加容量

假如给SYSTEM表空间添加 查看文件位置和容量&#xff1a;Select * FROM DBA_DATA_FILES; FILE_NAME就是要修改的文件 查看每一个表空间的容量&#xff0c;单位MB&#xff1a; SELECT t.tablespace_name, round(SUM(bytes / (1024 * 1024)), 0) ts_size FROM dba_tablespaces…

部分调试记录

Ubuntu16.04纯命令行安装VMwareTools hudahuahudahua-virtual-machine:~$ sudo apt-get install open-vm-tools -yhudahuahudahua-virtual-machine:~$ sudo apt-get install open-vm-tools-desktop无法加载so文件&#xff0c;版本问题 [rootdragonboard /]# ./Qserial -qws .…

sql server删除历史数据

1 函数 datediff函数: DATEDIFF ( datepart , startdate , enddate )datepart的取值可以是year,quarter,Month,dayofyear,Day,Week,Hour,minute,second,millisecond startdate 是从 enddate 减去。如果 startdate 比 enddate 晚&#xff0c;返回负值。 2 例子 删除2023年以…

2023前端面试笔记 —— CSS3

系列文章目录 内容链接2023前端面试笔记HTML52023前端面试笔记CSS3 文章目录 系列文章目录前言一、CSS选择器的优先级二、通过 CSS 的哪些方式可以实现隐藏页面上的元素三、px、em、rem之间有什么区别&#xff1f;四、让元素水平居中的方法有哪些五、在 CSS 中有哪些定位方式六…

DETRs with Collaborative Hybrid Assignments Training论文笔记

Title&#xff1a;[DETRs with Collaborative Hybrid Assignments Training Code 文章目录 1. Motivation2. one to one VS one to many3. Method&#xff08;1&#xff09;Encoder feature learning&#xff08;2&#xff09;Decoder attention learning 1. Motivation 当前…

正确进行自动化测试

前言&#xff1a; &#x1f4d5;作者简介&#xff1a;热爱编程的小七&#xff0c;致力于C、Java、Python等多编程语言&#xff0c;热爱编程和长板的运动少年&#xff01; &#x1f4d8;相关专栏Java基础语法&#xff0c;JavaEE初阶&#xff0c;数据库&#xff0c;数据结构和算法…

论文阅读及复现——《CT_ICP: Real-time Elastic LiDAR Odometry with Loop Closure》

论文阅读之——《CT_ICP: Real-time Elastic LiDAR Odometry with Loop Closure》带闭环的实时弹性激光雷达里程计 1. 主要贡献2. 相关说明3. 激光里程计3.1 里程计公式构建3.2 局部地图与健壮性 4. 回环检测与后端5. 实验结果5.1 里程计实验结果5.2 回环检测实验结果 6. 总结…

opencv-答题卡识别判卷

#导入工具包 import numpy as np import argparse import imutils import cv2# 设置参数 ap argparse.ArgumentParser() ap.add_argument("-i", "--image", requiredTrue,help"path to the input image") args vars(ap.parse_args())# 正确答案…

【附安装包】Vred2023安装教程

软件下载 软件&#xff1a;Vred版本&#xff1a;2023语言&#xff1a;简体中文大小&#xff1a;2.39G安装环境&#xff1a;Win11/Win10/Win8/Win7硬件要求&#xff1a;CPU2.0GHz 内存4G(或更高&#xff09;下载通道①百度网盘丨64位下载链接&#xff1a;https://pan.baidu.com…