Linux下 Socket服务器和客户端文件互传
目录
1.项目描述
2.函数准备
2.1 gets函数
2.2 popen函数、fread函数
2.3 access 函数
2.4 exit 函数
2.5 strtok 函数
2.6 chdir函数
3.项目代码
3.1服务器代码
3.2客户端代码
4.问题总结
1.项目描述
基于Soket聊天服务器,实现服务器和客户端的文件传输。
Linux系统下建立Socket聊天服务器_趣知boy的博客-CSDN博客
ls 获取服务器文件列表
pwd 获取服务器当前路径
cd 对服务器目录的操作 +dir
quit 退出连接 put 上传文件到服务器 +file_name
get 获取服务器数据 +file_namelcd 对客户端目录的操作 +dir
lls 列出客户端所有文件 dofile 创建文件
项目结构
2.函数准备
2.1 gets函数
gets 从标准输入流(通常是键盘)读取一行字符串,并将其存储在指定的字符数组中。
其比较重要的一个功能是阻塞
printf("请输入您的姓名:");gets(name);printf("您的姓名是:%s\n", name);
2.2 popen函数、fread函数
size_t fread(void *ptr, size_t size, size_t count, FILE *stream);
该函数可以读取指定数量(count = 1)的元素,每个元素的大小为size个字节,从指定文件(stream)中读取到内存指针(ptr)所指的位置。
2.3 access 函数
用于判断文件或目录是否具有某种权限。
int access(const char *pathname, int mode);
F_OK:检查文件是否存在
R_OK:检查读权限是否存在
W_OK:检查写权限是否存在
X_OK:检查执行权限是否存在
返回值:
-
如果路径名指定的文件或目录具有所需的权限,则返回0。
-
如果权限不足,则返回-1,并且errno设置为适当的错误码
2.4 exit 函数
exit函数是一个用于终止程序运行的函数。当调用exit函数时,程序将立即退出并返回到操作系统。
2.5 strtok 函数
char *strtok(char *str, const char *delim);
其中str是要进行分割的字符串,delim是作为分隔符的字符串。函数返回一个指向被分割出的子字符串的指针。
strtok函数陷阱:会改变指针指向的字符串,所以想保留原来str需要用strcpy复制出来处理。
token = strtok(str, ",");
while (token != NULL) {printf("%s\n", token);token = strtok(NULL, ",");
}return 0;
执行上述代码会输出以下结果:
Hello
World
This
is
Strtok
2.6 chdir函数
更改当前工作目录
int chdir(const char *path);
参 数:Path 目标目录,可以是绝对目录或相对目录。
返回值:成功返回0 ,失败返回-1
3.项目代码
3.1服务器代码
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <ctype.h>
#include <arpa/inet.h>
#include "config.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>struct Msg msg;
int ss_fd;void handler_cmd(char *cmd){int cmd_int,fd;char *p;FILE *file;char *cmd_pre=cmd;//handle the cmd to numberif(!strcmp(cmd_pre,"ls")) cmd_int=ls;if(!strcmp(cmd_pre,"pwd")) cmd_int=pwd; if(!strcmp(cmd_pre,"quit")) cmd_int=quit;if(!strcmp(cmd,"lls")) cmd_int=lls;if(strstr(cmd,"lcd")!=NULL) cmd_int=lcd;if(strstr(cmd_pre,"cd")!=NULL) cmd_int=cd;if(strstr(cmd_pre,"put")!=NULL) cmd_int=put;if(strstr(cmd_pre,"get")!=NULL) cmd_int=get;//handle cmd switch(cmd_int){case ls:case pwd:file=popen(cmd,"r");fread(msg.data,sizeof(msg.data),1,file);printf("%s",msg.data);write(ss_fd,&msg,sizeof(msg));break;case cd:p=strtok(cmd," ");p=strtok(NULL," ");printf("dir:%s \r\n",p);strcpy(msg.data,"change dir over \r\n");write(ss_fd,&msg,sizeof(msg)); //must give a msg backchdir(p);break;case get:p=strtok(cmd," ");p=strtok(NULL," ");printf("dir:%s \r\n",p); if(!access(p,F_OK)){ //if have this filemsg.flag=dofile;fd=open(p,O_RDONLY);read(fd,&msg.data,sizeof(msg.data));send(ss_fd,&msg,sizeof(msg),0);close(fd);}else{ //if no this filesend(ss_fd,"no this file \r\n",16,0);}break;case put:p=strtok(cmd," ");p=strtok(NULL," ");printf("dir:%s \r\n",p);fd=open(p,O_RDWR|O_CREAT,0666);write(fd,&msg.data,strlen(msg.data));close(fd);strcpy(msg.data,"i have got a file.\r\n");write(ss_fd,&msg,sizeof(msg)); //must give a msg backbreak;case quit:msg.flag=quit;printf("quit==========\r\n");exit(-1);}}int main(void)
{int s_fd,nread,len;struct sockaddr_in s_ddr; //build server msgstruct sockaddr_in c_ddr; //save clinet msgs_fd= socket(AF_INET, SOCK_STREAM, 0);//1.build a soket specifiedif(s_fd==-1){perror("error is");}//2.build all binds_ddr.sin_family=AF_INET;s_ddr.sin_port=htons(8880);s_ddr.sin_addr.s_addr=htonl(INADDR_ANY);//give the bindbind(s_fd,(struct sockaddr *)&s_ddr,sizeof(s_ddr));//3.waite for clientlisten(s_fd,8);//4.accept come and connect for oncelen=sizeof(c_ddr);while(1){ss_fd=accept(s_fd,(struct sockaddr *)&c_ddr,&len);if(ss_fd == -1){perror("accept:");}printf("conect succese!==========\r\n");//5.read from connect ss_fdif(fork()==0){ //creat kid pid for handler msg and cmd //5.2 read while(1){memset(&msg,'\0',sizeof(msg)); //clear msgprintf("====================\r\n");nread=read(ss_fd,&msg,sizeof(msg));if(nread==0){ // cannot recv cmd printf("connect is cutdon! \r\n");break;}else{ //can recive cmd printf("server receved cmd:%s \r\n",msg.cmd); printf("====================\r\n");handler_cmd(msg.cmd); //hander the cmd give msg data }}}}close(ss_fd);close(s_fd);return 0;
}
3.2客户端代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include "config.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>int cmd_to_number(char *cmd)
{if(!strcmp(cmd,"ls")) return ls;if(!strcmp(cmd,"pwd")) return pwd; if(!strcmp(cmd,"quit")) return quit;if(!strcmp(cmd,"lls")) return lls;if(strstr(cmd,"lcd")!=NULL) return lcd;if(strstr(cmd,"cd")!=NULL) return cd;if(strstr(cmd,"put")!=NULL) return put;if(strstr(cmd,"get")!=NULL) return get;return -1;}int handler_cmd(struct Msg msg, int s_fd)
{//handle cmd int fd,n_read; char p_pre[12];char *p=NULL;int ret=cmd_to_number(msg.cmd);switch(ret){case ls:case pwd:case cd:send(s_fd,&msg,sizeof(msg),0);break;case get:send(s_fd,&msg,sizeof(msg),0);break;case put:strcpy(p_pre,msg.cmd);printf("dir:%s \r\n",p_pre);p=strtok(p_pre," ");p=strtok(NULL," "); //get file_name printf("dir:%s \r\n",p);if(!access(p,F_OK)){ //if have this filefd=open(p,O_RDONLY);n_read=read(fd,&msg.data,1024);if(n_read>=1024){printf("file over 1024! faild! \r\n");} send(s_fd,&msg,sizeof(msg),0);memset(&msg.data,0,1024);printf("client put cmd is: %s \r\n",msg.cmd);close(fd);}else{ //if no this filesend(s_fd,"no this file",16,0);}break; case lls:system("ls");break;case lcd:strcpy(p_pre,msg.cmd);p=strtok(p_pre," ");p=strtok(NULL," "); //get file_printf("dir:%s \r\n",p);chdir(p);break;case quit:send(s_fd,&msg,sizeof(msg),0);close(s_fd); exit(-1);}return ret;
} void handle_server_msg(struct Msg msg,int s_fd)
{int n_read,fd;char *p;struct Msg recv_msg; //this is new msg use it becase need pre msg cmdn_read=read(s_fd,&recv_msg,sizeof(recv_msg)); //while blockif(n_read==0){printf("\r\n server is outline! ");exit(-1);}if(recv_msg.flag == dofile){p=strtok(msg.cmd," ");p=strtok(NULL," "); //get file_name fd=open(p,O_RDWR|O_CREAT,0666);printf("creat the file\r\n");write(fd,&recv_msg.data,strlen(recv_msg.data));close(fd);putchar('>');fflush(stdout);}else{ //normal direct cmdprintf("-------------------->\r\n");printf("%s >",recv_msg.data);printf("--------------------");putchar('>');fflush(stdout);}}int main(int argc,char *argv[])
{int flag,s_fd,n_read,ret;struct sockaddr_in c_ddr;struct Msg msg;//1.build sockets_fd=socket(AF_INET,SOCK_STREAM,0);//2.0 prepare server addrmemset(&c_ddr,0,sizeof(c_ddr)); //clear c_ddrc_ddr.sin_family=AF_INET;c_ddr.sin_port=htons(8880);inet_aton("127.0.0.1",&c_ddr.sin_addr);//2.connect server get s_fdif(connect(s_fd,(struct sockaddr *)&c_ddr,sizeof(c_ddr))==-1){perror("error");exit(-1);}printf("connect success==============\r\n");while(1){//1.get cmd form keyborad and handle cmd to servermemset(&msg,0,sizeof(msg));printf("\r\n >");gets(msg.cmd);int ret=handler_cmd(msg,s_fd);//2.handle the msg form serverif(ret==-1){printf("no this cmd!\r\n ");fflush(stdout);continue;}if(ret==5 || ret==6){printf("client cmd!\r\n ");fflush(stdout);continue;} handle_server_msg(msg,s_fd);}close(s_fd);return 0;
}
4.问题总结
- 在put 发送文件中,当文件大小超过给定字节就会溢出,导致整个结构体崩坏,破坏cmd。
思考:除了加大给定字节大小,还有其它办法吗?strcpy是怎么实现的
read函数陷阱:read函数读的字节小于fd文件字节,会崩坏buf里的数据。 - strtok函数会破坏使用的字符串,要保留原来字符串需要strcpy复制出来处理。
- 本地命令不用服务器处理的命令,ret=5,ret=6不要进入hanle_server_msg用recv函数阻塞
相关文章:

Linux下 Socket服务器和客户端文件互传
目录 1.项目描述 2.函数准备 2.1 gets函数 2.2 popen函数、fread函数 2.3 access 函数 2.4 exit 函数 2.5 strtok 函数 2.6 chdir函数 3.项目代码 3.1服务器代码 3.2客户端代码 4.问题总结 1.项目描述 基于Soket聊天服务器,实现服务器和客户端的文件传输。…...

Nginx详解 第五部分:Ngnix反向代理(负载均衡 动静分离 缓存 透传 )
Part 5 一、正向代理与反向代理1.1 正向代理简介1.2 反向代理简介 二、配置反向代理2.1 反向代理配置参数2.1.1 proxy_pass2.1.2 其余参数 2.2 配置实例:反向代理单台web服务器2.3 代理转发 三、反向代理实现动静分离四、缓存功能五、反向代理客户端的IP透传5.1 原理概述5.2 一…...

中国行政区域带坐标经纬度sql文件及地点获取经纬度方法
文章目录 前言一、如何获取某地的经纬度?1.1 搜索百度地图1.2 在下方找到地图开放平台1.3 下滑找到坐标拾取器1.4 使用 二、sql文件2.1 创建表2.2 插入数据 前言 当工作业务上需要涉及地图,给前端返回经纬度等场景,需要掌握区域经纬度的获取…...
[国产MCU]-W801开发实例-WiFi网络扫描
WiFi网络扫描 文章目录 WiFi网络扫描1、WiFi模块介绍2、WiFi扫描API介绍3、WiFi扫描实例本文将演示如何使用WiFi模块进行WiFi网络扫描。 1、WiFi模块介绍 W801的WiFi具有如下特性: 支持 GB15629.11-2006 IEEE802.11 b/g/n支持 Wi-Fi WMM/WMM-PS/WPA/WPA2/WPS支持 EDCA信道接入…...
SpringBoot使用kafka事务-消费者方
前言 在上一篇文章中,写到了如何在springboot中生产者如何使用kafka的事务,详情链接:Springboot使用kafka事务-生产者方 那么,这一篇就接着上篇所写的内容,讲解一下再springboot中消费者如何使用kafka的事务。 实现…...
C# 实现PictureBox从指定的文件夹内进行翻页操作
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System...
Eureka 注册中心的使用
环境 springboot springcloud Eureka-Server注册中心服务端 pom.xml导入依赖 <dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-eureka-server</artifactId><version>2.2.7.RELEAS…...
vue3 组件通信方式
文章目录 组件通信方式props自定义事件全局事件总线v-modeluseAttrsref与$parentprovide与injectpiniaslot 组件通信方式 props 实现父子组件通信,在vue3中可以通过defineProps获取父组件传递的数据。且在组件内部不需要引入defineProps方法可以直接使用! 父组…...

淘宝商品API使用示例:如何通过调用外部API来获取淘宝商品价格销量主图详情数据
淘宝上的商品信息量非常之大,商品的详情信息也很齐全。如何通过调用外部API来实现批量获取商品价格销量主图详情等信息呢?上周刚好完成了一个完整的淘宝商品采集项目,今天特来分享一下。 接口名称:item_get 请求地址:…...

RK3568-android11-适配ov13850摄像头
硬件连接 主要分为两部分: mipi接口:传输摄像头数据 i2c接口:配置摄像头和对焦马达芯片寄存器相关驱动 |-- arch/arm64/boot/dts/rockchip DTS配置文件 |-- drivers/phy/rockchip/|-- phy-rockchip-mipi-rx.c mipi dphy 驱动 |-- drivers/media||-- platform/rockchip/isp1…...
基于Sider-chatgpt3.5-编写一个使用springboot2.5连接elasticsearch7的demo程序,包括基本的功能,用模板方法
下面是一个使用Spring Boot 2.5连接Elasticsearch 7的示例程序,包括基本的功能,使用模板方法: 首先,确保你的项目中添加了以下依赖: <dependency> <groupId>org.springframework.boot</groupId> &l…...
nodejs中如何使用Redis
Redis介绍: Redis 是一个开源的内存数据结构存储器,一般可以用于数据库、缓存、消息代理等,我们常在项目中用redis解决高并发、高可用、高可扩展、大数据存储等问题; 它本质上是一个NoSql(非关系型数据库)…...
golang append坑
查看如下代码输出 package mainimport "fmt"func main() {a : make([][]int, 0)b : make([]int, 0)b append(b, 1)a append(a, b)fmt.Println(a)b[0] 2fmt.Println(a) }输出: [[1]] [[2]]可以看出b改变之后,在a中也发生了改变,…...

PaddleNLP使用Vicuna
LLaMA 模型 LLaMa 是一个大型语言模型,由 Meta 开源。它的全称是 Large Language Model Meta AI,参数量从 70 亿到 650 亿不等。例如,130 亿参数的 LLaMA 模型在大多数基准上可以胜过参数量达 1750 亿的 GPT-3,而且可以在单块 V1…...
jackson常用操作
#jackson常用操作 jackson序列化框架,一些常用的操作 依赖 <!--Jackson包--><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-core</artifactId><version>2.15.2</version></de…...

ios ipa包上传需要什么工具
目录 ios ipa包上传需要什么工具 前言 一、IPA包的原理 二、IPA包上传的步骤 1.注册开发者账号 2.apk软件制作工具创建应用程序 3.构建应用程序 4.生成证书和配置文件 5.打包IPA包 6.上传IPA包 三、总结 前言 iOS IPA包是iOS应用程序的安装包,可以通过iT…...

科目1基础知识快速入门精简
科目1-4 科目一,又称科目一理论考试、驾驶员理论考试。》学习道路交通安全法律、法规和相关知识学习 考试内容包括驾车理论基础、道路安全法律法规、地方性法规等相关知识,再加地方性法规。考试形式为上机考试,100道题,90分及以…...

安卓逆向 - 某东app加密参数还原
本文仅供学习交流,只提供关键思路不会给出完整代码,严禁用于非法用途,拒绝转载,若有侵权请联系我删除! 目标app:5Lqs5LicYXBwMTEuMy4y 目标接口:aHR0cHM6Ly9hcGkubS5qZC5jb20vY2xpZW50LmFjdGl…...

Visual Studio(2022)生成链接过程的.map映射文件以及.map映射文件的内容说明
微软的官方说明 /MAP(生成映射文件) | Microsoft Learn 设置步骤 1. 右键项目属性, 连接器 -> 常规 -> 启用增量链接,设置为否。如下图: 2. 连接器 -> 调试 生成调试信息 设置为 生成调试信息 (/DEBUG) 生成程序数据库…...
A. Gift Carpet
time limit per test 1 second memory limit per test 256 megabytes input standard input output standard output Recently, Tema and Vika celebrated Family Day. Their friend Arina gave them a carpet, which can be represented as an n⋅m�⋅&…...

19c补丁后oracle属主变化,导致不能识别磁盘组
补丁后服务器重启,数据库再次无法启动 ORA01017: invalid username/password; logon denied Oracle 19c 在打上 19.23 或以上补丁版本后,存在与用户组权限相关的问题。具体表现为,Oracle 实例的运行用户(oracle)和集…...
React hook之useRef
React useRef 详解 useRef 是 React 提供的一个 Hook,用于在函数组件中创建可变的引用对象。它在 React 开发中有多种重要用途,下面我将全面详细地介绍它的特性和用法。 基本概念 1. 创建 ref const refContainer useRef(initialValue);initialValu…...

CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...

ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...

Cloudflare 从 Nginx 到 Pingora:性能、效率与安全的全面升级
在互联网的快速发展中,高性能、高效率和高安全性的网络服务成为了各大互联网基础设施提供商的核心追求。Cloudflare 作为全球领先的互联网安全和基础设施公司,近期做出了一个重大技术决策:弃用长期使用的 Nginx,转而采用其内部开发…...

关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...

【C++】纯虚函数类外可以写实现吗?
1. 答案 先说答案,可以。 2.代码测试 .h头文件 #include <iostream> #include <string>// 抽象基类 class AbstractBase { public:AbstractBase() default;virtual ~AbstractBase() default; // 默认析构函数public:virtual int PureVirtualFunct…...

Elastic 获得 AWS 教育 ISV 合作伙伴资质,进一步增强教育解决方案产品组合
作者:来自 Elastic Udayasimha Theepireddy (Uday), Brian Bergholm, Marianna Jonsdottir 通过搜索 AI 和云创新推动教育领域的数字化转型。 我们非常高兴地宣布,Elastic 已获得 AWS 教育 ISV 合作伙伴资质。这一重要认证表明,Elastic 作为 …...

Linux操作系统共享Windows操作系统的文件
目录 一、共享文件 二、挂载 一、共享文件 点击虚拟机选项-设置 点击选项,设置文件夹共享为总是启用,点击添加,可添加需要共享的文件夹 查询是否共享成功 ls /mnt/hgfs 如果显示Download(这是我共享的文件夹)&…...
接口 RESTful 中的超媒体:REST 架构的灵魂驱动
在 RESTful 架构中,** 超媒体(Hypermedia)** 是一个核心概念,它体现了 REST 的 “表述性状态转移(Representational State Transfer)” 的本质,也是区分 “真 RESTful API” 与 “伪 RESTful AP…...