这一节将不会介绍太多的技术的问题,这节主要是搭建一个小小的框架,为了方便接下来的继续编写扩展程序。本次会在上一小节的基础上加上一个身份验证的功能。

  因为网盘程序不像聊天程序,网盘是属于主动向服务器拉取信息,而聊天程序有可能要被动获取信息,所以为了减轻服务器压力,每次要向服务器获取服务就建立一个短连接,而不像聊天程序一样的长连接,微信的公众平台,输入指令获取服务,就是这个样子了。具体看一下代码就知道了。

  还有为了方便处理,我增加了一个控制信号,这个控制信号在以前的聊天程序中讲到过,但是当时为了简单就没有实现,现在就简单实现一些,方便以后可以参考。

  还有就是这次的网盘程序我将尽量实现下面的这些功能:

  (1) file push filename           //用户上传文件到服务器上

  (2) file pull filename            //用户下载文件到本地上

  (3) file list              //列出用户在服务器中的所有文件

  (4) file sendto  filename username //共享文件给其他用户

  (5) file delete filename        //删除服务器中的文件

  (6) 用户登录,自动完成

  本次数据库名为filetranslate,目前有表user

 create table user
(
uid int;
username varchar(64);
password varchar(64);
);

  修改后的client.cpp

 #include <netinet/in.h> // sockaddr_in
#include <sys/types.h> //socket
#include <sys/socket.h> //socket
#include <netdb.h> //gethostbyname
#include <unistd.h> //close
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <arpa/inet.h> //inet_addr #define SERVVER_PORT 12138
#define LISTEN_QUEUE 20
#define BUFFER_SIZE 1024 //传输控制信号宏定义
//struct Control中control的取值
#define USER_CHECK_LOGIN 1
#define FILE_PUSH 2
#define FILE_PULL 3
#define FILE_LIST 4
#define FILE_SENDTO 5
#define FILE_DELECT 6 struct Addr
{
char host[];
int port;
};
struct User
{
int uid;
char username[];
char password[];
};
struct Control
{
int uid;
int control;
}; void print_time(char *ch);//打印时间
int file_push(struct Addr addr,struct User user,char *filenames);
int check_login(struct Addr addr,struct User * user); int main(int argc,char *argv[])
{
char orderbuf[BUFFER_SIZE];
struct Addr addr;
char arg1[],arg2[],arg3[],arg4[];
struct User user; if(argc!=)
{
perror("usage: ./client [serverhost] [serverport] [username] [password]");
exit(-);
}
strcpy(addr.host,argv[]);
addr.port=atoi(argv[]);
strcpy(user.username,argv[]);
strcpy(user.password,argv[]);
int uid=check_login(addr,&user);
if(uid<=)
{
perror("用户验证失败");
exit(-);
}
printf("验证登陆成功\n");
while()
{
printf("\n请输入操作指令:");
fgets(orderbuf,BUFFER_SIZE,stdin);
memset(arg1,,sizeof(arg1));
memset(arg2,,sizeof(arg2));
memset(arg3,,sizeof(arg3));
memset(arg4,,sizeof(arg4));
sscanf(orderbuf,"%s%s%s%s",arg1,arg2,arg3,arg4);
if(strcmp(arg1,"file")==)
{
if(strcmp(arg2,"push")==)
{
strcpy(orderbuf,arg3);
file_push(addr,user,orderbuf);
}
else if(strcmp(arg2,"pull")==)
{
;
}
else if(strcmp(arg2,"list")==)
{
;
}
else if(strcmp(arg2,"sendto")==)
{
;
}
else if(strcmp(arg2,"delect")==)
{
;
}
else
{
printf("该命令不支持\n");
}
}
else
{
printf("该命令不支持\n");
}
} return ;
} //验证成功时返回大于0的uid号码,错误返回-1
int check_login(struct Addr addr,struct User * user)
{
struct sockaddr_in servAddr;
struct hostent * host;
struct Control control;
int sockfd; host=gethostbyname(addr.host);
servAddr.sin_family=AF_INET;
servAddr.sin_addr=*((struct in_addr *)host->h_addr);
//servAddr.sin_addr.s_addr=inet_addr("127.0.0.1");
servAddr.sin_port=htons(addr.port);
if(host==NULL)
{
perror("获取IP地址失败");
exit(-);
}
if((sockfd=socket(AF_INET,SOCK_STREAM,))==-)
{
perror("socket创建失败");
exit(-);
}
if(connect(sockfd,(struct sockaddr *)&servAddr,sizeof(struct sockaddr_in))==-)
{
perror("connect 失败");
exit(-);
} control.control=USER_CHECK_LOGIN;
control.uid=;
if(send(sockfd,(char *)&control,sizeof(struct Control),)<)
{
perror("发送数据失败");
exit(-);
} if(send(sockfd,(char *)user,sizeof(struct User),)<)
{
perror("发送数据失败");
exit(-);
}
bzero(user,sizeof(struct User));
if(recv(sockfd,(char *)user,sizeof(struct User),)<)
{
perror("接收数据失败");
exit(-);
}
printf("获取后用户名:%s 密码:%s ID号:%d\n",user->username,user->password,user->uid); close(sockfd);//关闭socket连接
return user->uid;
} int file_push(struct Addr addr,struct User user,char *filenames)
{
struct sockaddr_in servAddr;
struct hostent * host;
struct Control control;
int sockfd;
FILE *fp; host=gethostbyname(addr.host);
servAddr.sin_family=AF_INET;
servAddr.sin_addr=*((struct in_addr *)host->h_addr);
//servAddr.sin_addr.s_addr=inet_addr("127.0.0.1");
servAddr.sin_port=htons(addr.port);
if(host==NULL)
{
perror("获取IP地址失败");
exit(-);
}
if((sockfd=socket(AF_INET,SOCK_STREAM,))==-)
{
perror("socket创建失败");
exit(-);
} if(connect(sockfd,(struct sockaddr *)&servAddr,sizeof(struct sockaddr_in))==-)
{
perror("connect 失败");
exit(-);
} //打开文件
if((fp=fopen(filenames,"rb"))==NULL)
{
perror("文件打开失败");
exit(-);
}
//这里传输控制信号
control.control=FILE_PUSH;
control.uid=user.uid;
if(send(sockfd,(char *)&control,sizeof(struct Control),)<)
{
perror("控制信号发送失败");
exit(-);
}
char buffer[BUFFER_SIZE];
bzero(buffer,BUFFER_SIZE);
printf("正在传输文件");
int len=;
//不断的读取文件直到文件结束
while((len=fread(buffer,,BUFFER_SIZE,fp))>)
{
if(send(sockfd,buffer,len,)<)
{
perror("发送数据失败");
exit(-);
}
bzero(buffer,BUFFER_SIZE);
printf(".");//1K打印一个点//如果要实现百分比,就要计算文件大小,然后再处理即可
} printf("传输完毕\n");
fclose(fp);//关闭文件流
close(sockfd);//关闭socket连接 return ;
} void print_time(char *ch)
{
time_t now;
struct tm * stm;
time(&now);
stm=localtime(&now);
sprintf(ch,"%02d:%02d:%02d\n",stm->tm_hour,stm->tm_min,stm->tm_sec);
return ;
}

  下面这个是server.cpp

 #include <netinet/in.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/select.h>
#include <netdb.h>
#include <stdlib.h>
#include <time.h>
#include <string.h>
#include <unistd.h>
#include <stdio.h>
#include <arpa/inet.h> //inet_ntoa
#include <mysql.h> #define SERVER_PORT 12138
#define LISTEN_QUEUE 20
#define BACKLOG 200
#define BUFFER_SIZE 1024 //传输控制信号宏定义
//struct Control中control的取值
#define USER_CHECK_LOGIN 1
#define FILE_PUSH 2
#define FILE_PULL 3
#define FILE_LIST 4
#define FILE_SENDTO 5
#define FILE_DELECT 6 struct User
{
int uid;
char username[];
char password[];
}; struct Control
{
int uid;
int control;
}; void print_time(char *ch);//打印时间
int MAX(int a,int b);
int mysql_check_login(struct User user); int main(int argc,char *argv[])
{
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
struct User user;
struct Control control;
char ch[];
int clientfd;
pid_t pid;
socklen_t length;
bzero(&server_addr,sizeof(server_addr));
server_addr.sin_family=AF_INET;
server_addr.sin_addr.s_addr=htons(INADDR_ANY);
server_addr.sin_port=htons(SERVER_PORT); //创建套接字
int sockfd=socket(AF_INET,SOCK_STREAM,);
if(sockfd<)
{
perror("创建套接字失败");
exit(-);
} if(bind(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr))==-)
{
perror("bind 失败");
exit(-);
} if(listen(sockfd,LISTEN_QUEUE))
{
perror("listen 失败");
exit(-);
} length=sizeof(struct sockaddr); while()
{
clientfd=accept(sockfd,(struct sockaddr *)&client_addr,&length);
if(clientfd==-)
{
perror("accept 失败");
continue;
}
printf(">>>>>%s:%d 连接成功,当前所在的ID(fd)号: %d \n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port),clientfd);
print_time(ch);
printf("加入的时间是:%s\n",ch); //来一个连接就创建一个进程进行处理
pid=fork();
if(pid<)
{
perror("fork error");
}
else if(pid==)
{
recv(clientfd,(char *)&control,sizeof(struct Control),);
printf("用户 %d 使用命令 %d\n",control.uid,control.control);
switch(control.control)
{
case USER_CHECK_LOGIN:
{
//身份验证处理
recv(clientfd,(char *)&user,sizeof(struct User),);
printf("客户端发送过来的用户名是:%s,密码:%s\n",user.username,user.password);
if((user.uid=mysql_check_login(user))>)
{
printf("验证成功\n");
}
else
{
printf("验证失败\n");
}
send(clientfd,(char *)&user,sizeof(struct User),);
break;
}
case FILE_PUSH:
{
char buffer[BUFFER_SIZE];
int data_len;
FILE * fp=NULL;
bzero(buffer,BUFFER_SIZE);
if((fp=fopen("data","wb"))==NULL)
{
perror("文件打开失败");
exit(-);
}
//循环接收数据
int size=;//表示有多少个块
while(data_len=recv(clientfd,buffer,BUFFER_SIZE,))
{
if(data_len<)
{
perror("接收数据错误");
exit(-);
}
size++;
if(size==)
printf("正在接收来自%s:%d的文件\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
else
printf(".");
//向文件中写入
int write_len=fwrite(buffer,sizeof(char),data_len,fp);
if(write_len>data_len)
{
perror("写入数据错误");
exit(-);
}
bzero(buffer,BUFFER_SIZE);
}
if(size>)
printf("\n%s:%d的文件传送完毕\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
else
printf("\n%s:%d的文件传送失败\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
fclose(fp);
//rename("data","asdf");//这里可以修改文件的名字
exit();
break;
}
case FILE_PULL:
{
break;
}
case FILE_LIST:
{
break;
}
case FILE_DELECT:
{
break;
}
default:
{
break;
}
}
close(clientfd);//短连接结束
exit();//退出子进程
}
} return ;
} //函数定义
int mysql_check_login(struct User user)
{
MYSQL conn;
MYSQL_RES *res_ptr;
MYSQL_ROW result_row;
int res;
int row;
int column;
int uid;
char sql[]={};
strcpy(sql,"select uid from users where username=\"");
strcat(sql,user.username);
strcat(sql,"\" and password=\"");
strcat(sql,user.password);
strcat(sql,"\";");
printf("查询的sql:%s\n",sql);
uid=-;
mysql_init(&conn);
if(mysql_real_connect(&conn,"localhost","root","","filetranslate",,NULL,CLIENT_FOUND_ROWS))
{
res=mysql_query(&conn,sql);
if(res)
{
perror("Select SQL ERROR!");
exit(-);
}
else
{
res_ptr=mysql_store_result(&conn);
if(res_ptr)
{
column=mysql_num_fields(res_ptr);//获取列数
row=mysql_num_rows(res_ptr)+;//获取行数,加1表示还有第一行字段名
if(row<=)
{
;//验证失败
}
else
{
//这里先假定username是唯一
result_row=mysql_fetch_row(res_ptr);
printf("获取到的uid是:%s\n",result_row[]);
uid=atoi(result_row[]);
}
}
else
{
printf("没有查询到匹配的数据\n");
}
}
}
else
{
perror("Connect Failed!\n");
exit(-);
}
mysql_close(&conn);
return uid;
} void print_time(char *ch)
{
time_t now;
struct tm * stm;
time(&now);
stm=localtime(&now);
sprintf(ch,"%02d:%02d:%02d\n",stm->tm_hour,stm->tm_min,stm->tm_sec);
return ;
} int MAX(int a,int b)
{
if(a>b)
return a;
return b;
}

  下面给张运行时的截图

  关于c语言调用mysql: http://www.cnblogs.com/wunaozai/p/3876134.html

  本文地址: http://www.cnblogs.com/wunaozai/p/3887728.html

Socket网络编程--小小网盘程序(2)的更多相关文章

  1. Socket网络编程--小小网盘程序(5)

    各位好呀!这一小节应该就是这个小小网盘程序的最后一小节了,这一节将实现最后的三个功能,即列出用户在服务器中的文件列表,还有删除用户在服务器中的文件,最后的可以共享文件给好友. 列出用户在服务器中的文件 ...

  2. Socket网络编程--小小网盘程序(4)

    在这一小节中实现了文件的下载,具体的思路是根据用户的uid和用户提供的文件名filename联合两张表,取得md5唯一标识符,然后操作这个标识符对应的文件发送给客户端. 实现下载的小小网盘程序 cli ...

  3. Socket网络编程--小小网盘程序(3)

    接上一小节,这次增加另外的两张表,用于记录用户是保存那些文件.增加传上来的文件的文件指纹,使用MD5表示. 两张表如下定义: create table files( fid int, filename ...

  4. Socket网络编程--小小网盘程序(1)

    这个系列是准备讲基于Linux Socket进行文件传输.简单的文件传输就是客户端可以上传文件,可以从服务器端下载文件.就这么两个功能如果再加上身份验证,就成了FTP服务器了,如果对用户的操作再加上一 ...

  5. linux下C语言socket网络编程简例

    原创文章,转载请注明转载字样和出处,谢谢! 这里给出在linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到client的连接后,发送数据给client:clie ...

  6. 5.3linux下C语言socket网络编程简例

    原创文章,转载请注明转载字样和出处,谢谢! 这里给出在Linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到客户端的连接后,发送数据给客户端:客户端在接受到数据后 ...

  7. Socket网络编程--聊天程序(9)

    这一节应该是聊天程序的最后一节了,现在回顾我们的聊天程序,看起来还有很多功能没有实现,但是不管怎么说,都还是不错的.这一节我们将讲多服务器问题(高大上的说法就是负载问题了.)至于聊天程序的文件发送(也 ...

  8. Socket网络编程系列教程序

    C语言的用途相当多,可以用在数据结构.数据库.网络.嵌入式等方面,历经40多年不衰,真是厉害!最近一直想从某一应用方面写一个系列教程,好好地把某一方面讲深讲透.         正好博主对网络方面的编 ...

  9. Socket网络编程--FTP客户端

    Socket网络编程--FTP客户端(1)(Windows) 已经好久没有写过博客进行分享了.具体原因,在以后说. 这几天在了解FTP协议,准备任务是写一个FTP客户端程序.直接上干货了. 0.了解F ...

随机推荐

  1. 洛谷 p1434 滑雪【记忆化搜索】

    <题目链接> Michael喜欢滑雪.这并不奇怪,因为滑雪的确很刺激.可是为了获得速度,滑的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待升降机来载你.Michael想知道 ...

  2. sql注入总结(一)--2018自我整理

    SQL注入总结 前言: 本文和之后的总结都是进行总结,详细实现过程细节可能不会写出来~ 所有sql语句均是mysql数据库的,其他数据库可能有些函数不同,但是方法大致相同 0x00 SQL注入原理: ...

  3. struts1 标签引入

    1 tld文件导入 目录结构如下 2 jsp 文件头部标签引入 <%@ page pageEncoding="gbk" contentType="text/html ...

  4. 线程安全地获取插入mysql的条目的id

    在往mysql中插入条目时有时会希望能得到该插入条目的id,一种方式是再执行一个select语句条件为max(id)来获取,但这种形式在并发环境里并不是线程安全的,因为在你完成插入到再执行一个sele ...

  5. C++中的智能指针

    一.动态内存管理 通常我们创建动态内存的时候,需要自己管理好内存,也就是说,new出来的对象一定要注意释放掉.下面通过例子可以看到这个问题所在: struct BBE{ int X; int Y; v ...

  6. jQueryPrint 的简单使用

    jQueryPrint 的简单使用 一.为什么要使用 jQueryPrint?  1.当然是方便的要死尼,相比于其他的方法. 2.打印整个页面或者局部页面都是非常的可以的,使用很方便. 3.如果要导出 ...

  7. CF 633 F. The Chocolate Spree 树形dp

    题目链接 CF 633 F. The Chocolate Spree 题解 维护子数答案 子数直径 子数最远点 单子数最长直径 (最长的 最远点+一条链) 讨论转移 代码 #include<ve ...

  8. 洛谷.3374.[模板]树状数组1(CDQ分治)

    题目链接 简易CDQ分治教程 //每个操作分解为一个有序数对(t,p),即(时间,操作位置),时间默认有序,用CDQ分治处理第二维 //对于位置相同的操作 修改优先于查询 //时间是默认有序的 所以可 ...

  9. PHP中的字符串 — 表示方法

    Strings 的字符集,因此本质上不支持Unicode编码,关于Unicode阅读 utf8_encode() 和 utf8_decode() . 注意: 一个字符串的大小决定与计算机内存的大小,理 ...

  10. Android MediaPlayer架构 -- 前言小知识点(一)

    在Android中可以使用MediaPlayer+SurfaceView来实现一个简单的多媒体播放器. 一  构造函数 java MediaPlayer class 的源码位置:frameworks\ ...