引用自 http://blog.sina.com.cn/s/blog_6c9c71840101brdz.html


在线英英词典(sqlite3)

项目功能描述:

1.用户注册和登录验证:服务器端将用户信息和历史记录保存在数据库中,客户端输入用户名和密码,服务器端在数据库中查找、匹配,返回结果。

2.单词在线翻译:根据客户端输入的单词在字典文件中搜索。

3.历史记录查询

include.h

#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"

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

server.c

#include "include.h"
void do_register(int connectfd, MSG *msg, sqlite3 *db)  //注册用户
{
    char sqlstr[128];
    char *errmsg;
    
    sprintf(sqlstr, "insert into usr values ('%s', '%s')", msg->name, msg->data);//封装SQL操作语句
    printf("%s\n", sqlstr);
    if (sqlite3_exec(db, sqlstr, NULL, NULL, &errmsg) != SQLITE_OK) //执行SQL操作语句 
    {
        sqlite3_free(errmsg);
        sprintf(msg->data, "user %s already exist!!!", msg->name);
    }
    else
    {
        strncpy(msg->data, "OK", 256);    // 操作成功
    }
    send(connectfd, msg, sizeof(MSG), 0);
    return;
}

void do_login(int connectfd, MSG *msg, sqlite3 *db)  //登录用户
{
    char sqlstr[128];
    char *errmsg, **result;
    int nrow, ncolumn;
    sprintf(sqlstr, "select * from usr where name = '%s' and pass = '%s'", msg->name, msg->data);
    if (sqlite3_get_table(db, sqlstr, &result, &nrow, &ncolumn, &errmsg) != SQLITE_OK)
    {
        printf("error : %s\n", errmsg);
        sqlite3_free(errmsg);
    }
    if (nrow == 0)        //没有找到登录的用户,报错
    {
        strncpy(msg->data, "name or password is wrong!!!", 256);
    }
    else                  //登录成功,发OK
    {
        strncpy(msg->data, "OK", 256);
    }
    send(connectfd, msg, sizeof(MSG), 0);
    sqlite3_free_table(result);
    return;
}
void get_date(char date[])  //获得历史记录的时间
{
    time_t t;
    struct tm *tp;
    time(&t);
    tp = localtime(&t);
    strftime(date, 64, "%Y-%m-%d %H:%M:%S", tp);
    return;
}

int do_searchword(int connectfd, MSG *msg)  //搜索单词
{
    FILE *fp;
    char s[300];

char *p;

if ((fp = fopen("dict.txt", "r")) == NULL)
    {
        strcpy(msg->data, "dict on server can't be opened :(");
        send(connectfd, msg, sizeof(MSG), 0);
    }
    printf("query word is %s\n", msg->data);

while (fgets(s, 300, fp) != NULL)    
    {
        if (0 == strncmp(msg->data,s,strlen(msg->data)) )

{

p = s + strlen(msg->data);

while(0 == strncmp(p," ",1) ) p++;    //去掉这一行前面的单词和空格

strcpy(msg->data,p);                   //只发送解释的部分

send(connectfd,msg,sizeof(MSG),0);

fclose(fp);

return 1;     //返回1说明查找成功

}

}
    fclose(fp);
    return 0;                //返回0说明查找失败
}

void do_query(int connectfd, MSG *msg, sqlite3 *db)  //将历史记录插入到record表中
{
    char sqlstr[128], *errmsg;
    int found = 0;
    char date[64], word[64];
    strcpy(word, msg->data);
    found = do_searchword(connectfd, msg);
    if ( found )          //判断找到单词是否成功
    {
        get_date(date);   //如果成功,记录时间,插入历史表
        sprintf(sqlstr, "insert into record values ('%s', '%s', '%s')", msg->name, date, word);
        if (sqlite3_exec(db, sqlstr, NULL, NULL, &errmsg) != SQLITE_OK)
        {
            printf("error : %s\n", errmsg);
            sqlite3_free(errmsg);
        }
    }
    else                 //如果失败,报错
    {
        strcpy(msg->data, "not found\n");
    }
    send(connectfd, msg, sizeof(MSG), 0);
    return;
}

int history_callback(void *arg, int f_num, char **f_value, char **f_name)//回调表中msg->name的记录
{
    int connectfd;
    MSG msg;
    connectfd = *(int *)arg;
    sprintf(msg.data, "%s : %s", f_value[1], f_value[2]);
    send(connectfd, &msg, sizeof(msg), 0);
    return 0;
}

void do_history(int connectfd, MSG *msg, sqlite3 *db)  //查找当前用户的历史记录
{
    char sqlstr[128], *errmsg;
    sprintf(sqlstr, "select * from record where name = '%s'", msg->name);
    if (sqlite3_exec(db, sqlstr, history_callback, (void *)&connectfd, &errmsg) != SQLITE_OK)
    {
        printf("error : %s\n", errmsg);
        sqlite3_free(errmsg);
    }
    msg->data[0] = '\0';               //表示回调结束
    send(connectfd, msg, sizeof(MSG), 0);
    
    return;
}

void do_client(int connectfd, sqlite3 *db)  //处理客户机发来的信息
{
    MSG msg;
    while (recv(connectfd, &msg, sizeof(msg), 0) > 0)  
    {
        switch ( msg.type )
        {
        case 1 :
            do_register(connectfd, &msg, db);    //注册
            break;
        case 2 :
            do_login(connectfd, &msg, db);    //登录
            break;
        case 3 :
            do_query(connectfd, &msg, db);    //查询单词
            break;
        case 4 :
            do_history(connectfd, &msg, db);    //历史记录
            break;
        }
    }
    printf("client quit\n");
    exit(0);
    return;
}

main_server.c

#include "include.h"

int main(int argc, char *argv[])

{
    int listenfd, connectfd;
    struct sockaddr_in server_addr;
    pid_t pid;
    sqlite3 *db;
 
    if (sqlite3_open(DATABASE, &db) != SQLITE_OK)
    {
        printf("error : %s\n", sqlite3_errmsg(db));
        exit(-1);
    }

//create table record (name text,date,text,word text);

//create table usr (name text primary key,pass text);

//这边没有创建数据库的表,请在my.db里自行创建以上两个表
    if ((listenfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror("fail to socket");
        exit(-1);
    }
    bzero(&server_addr, sizeof(server_addr));
    server_addr.sin_family = PF_INET;
    server_addr.sin_addr.s_addr = inet_addr("192.168.1.106");
    server_addr.sin_port = htons(8888);
    if (bind(listenfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
    {
        perror("fail to bind");
        exit(-1);
    }
    if (listen(listenfd, 5) < 0)
    {
        perror("fail to listen");
        exit(-1);
    }
    signal(SIGCHLD, SIG_IGN);  // 为了不生成僵尸进程
    while (1)
    {
        if ((connectfd = accept(listenfd, NULL, NULL)) < 0)
        {
            perror("fail to accept");
            exit(-1);
        }
        if ((pid = fork()) < 0)         //多进程以便多用户登录
        {
            perror("fail to fork");
            exit(-1);
        }
        if (pid == 0)             //子进程执行客户机函数
        {
            do_client(connectfd, db);
        }
        close(connectfd);
    }
    return 0;
}

client.c

#include "include.h"
void do_register(int socketfd, MSG *msg)  //注册用户
{
    msg->type = 1;
    printf("input name : ");
    scanf("%s", msg->name);
    printf("input password : ");
    scanf("%s", msg->data);
    send(socketfd, msg, sizeof(MSG), 0);
    recv(socketfd, msg, sizeof(MSG), 0);
    printf("register : %s\n", msg->data);
    return;
}
 
int do_login(int socketfd, MSG *msg)  //登录用户
{
    msg->type = 2;
    printf("input name : ");
    scanf("%s", msg->name);
    printf("input password : ");
    scanf("%s", msg->data);
    send(socketfd, msg, sizeof(MSG), 0);
    recv(socketfd, msg, sizeof(MSG), 0);
    if (strncmp(msg->data, "OK", 3) == 0)     //判断接受的是不是OK
    {
        printf("login : OK\n");
        return 1;                            //返回1说明登录成功
    }
    printf("login : %s\n", msg->data);       //打印报错的信息
    return 0;                                //返回0说明登录失败
}

void do_query(int socketfd, MSG *msg)  //把输入的单词发给服务器
{
    msg->type = 3;
    while (1)
    {
        scanf("%s", msg->data);
        if (strcmp(msg->data, "#") == 0) break;  //输入“#”退出
        send(socketfd, msg, sizeof(MSG), 0);
        recv(socketfd, msg, sizeof(MSG), 0);
        printf("%s\n",msg->data);          //打印单词解释,或是没有找到
    }
    return;
}
 
void do_history(int socketfd, MSG *msg)  //保存记录到历史记录里
{
    msg->type = 4;
    send(socketfd, msg, sizeof(MSG), 0);
    while (1)
    {
        recv(socketfd, msg, sizeof(MSG), 0);  //执行一次接收一条历史记录
        if (msg->data[0] == '\0') break;      //如果遇到'\0'表示回调历史记录结束
        printf("%s\n", msg->data);       //打印一条历史记录
    }
    return;
}

main_client.c

#include "include.h"

int main(int argc, char *argv[])

{
    int socketfd ;
    struct sockaddr_in server_addr;
    MSG msg;

if ((socketfd = socket(PF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror("fail to socket");
    }
    bzero(&server_addr, sizeof(server_addr));
    server_addr.sin_family = PF_INET;
    server_addr.sin_addr.s_addr = inet_addr("192.168.1.106");
    server_addr.sin_port = htons(8888);
    if (connect(socketfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
    {
        perror("fail to connect");
        exit(-1);
    }
    int n;
    while (1) //用户注册登录界面
    {
        printf("************************************\n");
        printf("* 1: register   2: login   3: quit *\n");
        printf("************************************\n");
        printf("please choose : ");
        if (scanf("%d", &n) <= 0)
        {
            perror("scanf");
            exit(-1);
        }
        switch (n)
        {
        case 1 :
            do_register(socketfd, &msg);
            break;
        case 2 :
            if (do_login(socketfd, &msg) == 1)
            {
                printf("\n");
                goto next; //登录成功则跳至查询单词的界面
            }
            break;
        case 3 :
            close(socketfd);
            exit(0);
        }
    }
next:
    while (1)
    {
        printf("***********************************************\n");
        printf("* 1: query_word   2: history_record   3: quit *\n");
        printf("***********************************************\n");
        printf("please choose : ");
        if (scanf("%d", &n) < 0)
        {
            perror("scanf");
            exit(-1);
        }
        switch (n)
        {
        case 1 :
            do_query(socketfd, &msg);
            break;
        case 2 :
            do_history(socketfd, &msg);
            break;
        case 3 :
            close(socketfd);
            exit(0);
        }
    }
    return 0;
}

gcc -o server server.c main_server.c -lsqlite3    //sqlite3的库一定要在最后加上

gcc -o server server.c main_server.c -lsqlite3

注意:SQLite数据库要安装好(sudo apt-get install sqlite3 libsqlite3-dev),一定要有dict.txt的词典,可以上网下个。

Linux 在线词典相关推荐

  1. Linux多进程编程之在线词典

    在线词典是基于Linux 多进程并发服务器编程,由服务器端和客户端构成,客户端可以运行在多个不同的主机上连接服务器,服务器对员工信息的操作结果通过数据库sqlite来保存.当用户登录后,根据用户名判断 ...

  2. 词汇小助手V1.1——引入自动翻译和在线词典功能

    词汇小助手V1.1--引入自动翻译和在线词典功能 下载地址: http://files.cnblogs.com/pythonlover/词汇小助手V1.1.zip 版本更新信息: 1.词频统计部分引入 ...

  3. golang在线词典

    golang在线词典 前言 抓包 解析成go代码 json转go结构体 输入看看效果 重点 前言   实现功能:输入一个英文单词,返回翻译及其英标.   这个项目非常简单,本文的主要目的是记录两个网站 ...

  4. Linux在线求助 man page

    Linux在线求助 man page man page: 带中括号的都是可选的. whatis 命令:样例, whatis ls 描述命令的主要功能 命令 –help 告诉你命令怎么使用 ls –he ...

  5. linux怎么在线安装gcc,Linux 在线安装软件 gcc在线安装的操作方法

    Linux在线安装相关命令: yum install:全部安装 yum install :安装指定的程序包 yum groupinsall :安装程序组 yum -y update:升级所有包,改变软 ...

  6. linux在线扩展文件系统空间ext2online

    linux在线扩展文件系统空间 ext2resize是一款可以增大/减小ext2/ext3文件系统的工具,安装后有2个命令ext2resize和ext2online,ext2resize命令 必须要在 ...

  7. Linux在线升级ruby

    Linux在线升级ruby 标签: y / u / u 一.使用RVM也就是Ruby Version Manager,Ruby版本管理器来升级ruby,RVM包含了Ruby的版本管理和Gem库管理(g ...

  8. 【基于TCP的在线词典】

    一.项目名称 基于TCP的在线词典 二.功能 1.新用户使用时需要注册,用户名不可以和已有用户重复. 2.登录时需要校验用户名及密码. 3.输入要查询的单词后返回查询结果,可循环查询,在输入#后退出查 ...

  9. go语言实现简单的在线词典

    go语言实现简单的在线词典 这里使用火山翻译 1. Chome浏览器打开火山翻译,右键检查 输入一个单词进行翻译,发起请求,查看Network中的Headers是post请求的,并且Response里 ...

  10. 嵌入式学习项目实战 --- 在线词典

    目录 一.前言 二.项目功能 三.程序流程 1.客户端 2.服务器 四.代码实现 1.客户端代码 2.服务器代码 3.Makefile 一.前言 本文学习自[华清远见]的一个开源嵌入式项目在线词典综合 ...

最新文章

  1. idea自动生成get set_CTO:不要在Java代码中写set/get方法了,逮一次罚款
  2. php函数的实现原理及性能分析
  3. java keygenerator_spring-cache -KeyGenerator自定义
  4. spring框架学习(二)依赖注入
  5. VS2008中对.Net 3.5 sp1程序打安装包的前提系统环境的配置
  6. linux 网络管理
  7. linux mingling
  8. 第三方验收测试报告怎么做?
  9. link标签的real属性理解
  10. 全球公认最搞笑的15则冷笑话
  11. CF632E Thief in a Shop 题解
  12. 一名大二废柴今后目标
  13. Incompatible pointer types assigning to ‘id<PHPickerViewControllerDelegate> _Nullable‘ from ‘Class‘
  14. 用html代码画玫瑰花
  15. APK获取包名的办法
  16. 网络监控系统安装的六种传输方式
  17. vs2017打开项目后项目是空的
  18. HTML5~问卷调查页面的设计与实现
  19. 成都Java培训完什么水平?
  20. 极客日报:苹果承认从2019年开始扫描用户邮件寻找虐童资料;新浪回应“花钱买热搜”传闻;李沐斯坦福《机器学习》课程上线

热门文章

  1. 开心消消乐简单的逆向破解过程
  2. Word解析之Word内部结构
  3. JSP计算机社团管理系统源码,大学生社团管理系统JSP
  4. vlan 的三种模式
  5. Excel表格中的删除操作,看看你会几种?
  6. 7-15 计算圆周率 (C语言)
  7. 离散数学及其应用 前言
  8. 教你如何制作网页木马
  9. 将ClearCase的客户端编码设置为UTF-8
  10. 汉语词典 mdd mdx 下载_三款你必须拥有的英文词典软件