分布式文件系统之FastDFS
环境引入:
在一个大型的教育官网,会拥有大量优质的视频教程,并且免费提供给用户去下载,文件太多如果高效存储?用户访问量大如何保证下载速度?分布式文件系统是解决这些问题的有效方法之一
一、什么是文件系统
分布式文件系统(distributed file system)是指文件系统管理的物理存储资源不一定直接连接在本地节点上,而是通过计算机网络与节点相连。分布式文件系统的设计基于客户机/服务器模式。一个典型的网络可能包括多个供多用户访问的服务器。另外,对等特性允许一些系统扮演客户机和服务器的双重角色。例如,用户可以“发表”一个允许其他客户机访问的目录,一旦被访问,这个目录对客户机来说就像使用本地驱动器一样。
-----------------百度百科
为什么需要:以前,我们将上传的视屏文件等放在一台宿主机内,如果一个盘符内存不够,就增加硬盘个数,但是单纯的增加硬盘个数已经无法满足当代的需求,毕竟硬盘访问速度有限
解决方式:增加计算机个数,将视屏分别放在不同计算机内,通过网络将一个一个计算机的文件系统连接起来组成一个网络文件系统,形成一个分布式网络
优点:用于扩容、高并发场景
1、一台计算机的文件系统处理能力扩充到多台计算机内同时处理
2、一台计算机挂了还有另外副本计算提供数据
3、每台计算机可以放在不同的地域,这样用户就可以就近访问,提高访问速度
二、分布式文件服务提供商
1、阿里的oss
2、七牛云存储
3、百度云储存
三、什么是fastdfs
-----百度百科
图片来源于百度百科
四、fastdfs安装:
1、安装编译环境
[root@bogon ~]# docker exec -it centos /bin/bash [root@0b5933e7fd96 /]# yum -y groupinstall 'development tools' [root@0b5933e7fd96 /]# yum -y install wget
2、下载安装libfastcommon
[root@0b5933e7fd96 /]#git clone https://github.com/happyfish100/libfastcommon.git [root@0b5933e7fd96 /]#cd libfastcommon/ [root@0b5933e7fd96 /]#./make.sh [root@0b5933e7fd96 /]#./make.sh install
3、下载安装fastdfs
[root@0b5933e7fd96 /]#wget https://github.com/happyfish100/fastdfs/archive/v5.05.tar.gz [root@0b5933e7fd96 /]#tar -zxvf v5.05.tar.gz [root@0b5933e7fd96 /]#cd fastdfs-5.05/ [root@0b5933e7fd96 /]#./make.sh [root@0b5933e7fd96 /]#./make.sh install
4、修改tracker与storage配置文件命名
备份tracker与storage配置文件,为了防止数据文件的破坏
[root@0b5933e7fd96 /]# find / -name fdfs /etc/fdfs [root@0b5933e7fd96 fdfs]# ll total 40 -rw-r--r--. 1 root root 1461 jan 13 02:08 client.conf.sample -rw-r--r--. 1 root root 7829 jan 13 02:08 storage.conf.sample -rw-r--r--. 1 root root 7102 jan 13 02:08 tracker.conf.sample [root@0b5933e7fd96 fdfs]# cp client.conf.sample client.conf [root@0b5933e7fd96 fdfs]#cp storage.conf.sample storage.conf [root@0b5933e7fd96 fdfs]#cp tracker.conf.sample tracker.conf [root@0b5933e7fd96 fdfs]# ll total 40 -rw-r--r--. 1 root root 1461 jan 13 02:14 client.conf -rw-r--r--. 1 root root 1461 jan 13 02:08 client.conf.sample -rw-r--r--. 1 root root 7836 jan 13 03:30 storage.conf -rw-r--r--. 1 root root 7829 jan 13 02:08 storage.conf.sample -rw-r--r--. 1 root root 7103 jan 13 03:28 tracker.conf -rw-r--r--. 1 root root 7102 jan 13 02:08 tracker.conf.sample
5、配置tracker
[root@0b5933e7fd96 fdfs]# vi tracker.conf
disabled=false port=22122 base_path=/home/fastdfs/tracker http.server_port=8080
7、配置storage
disabled=false group_name=group1 port=23000 base_path=/home/fastdfs/storage
store_path0=/home/fastdfs/storage-data tracker_server=192.168.174.128:22122 http.server_port=8888
8、配置路径
[root@0b5933e7fd96 home]# ll total 0 drwxr-xr-x. 5 root root 53 jan 13 03:26 fastdfs [root@0b5933e7fd96 home]# cd fastdfs/ [root@0b5933e7fd96 fastdfs]# ll total 0 drwxr-xr-x. 3 root root 17 jan 13 03:31 storage drwxr-xr-x. 2 root root 6 jan 13 03:26 storage-data drwxr-xr-x. 4 root root 28 jan 13 03:31 tracker
8、启动tracker和storage
[root@0b5933e7fd96 fdfs]# /usr/bin/fdfs_trackerd /etc/fdfs/tracker.conf restart [root@0b5933e7fd96 fdfs]# /usr/bin/fdfs_storaged /etc/fdfs/storage.conf restart [root@0b5933e7fd96 storage-data]# ps -ef | grep fdfs
五、构建java client api
github访问地址:https://github.com/happyfish100/fastdfs-client-java
搭建环境
1、创建一个maven工程
2、添加依赖
<parent>
<groupid>org.springframework.boot</groupid>
<artifactid>spring-boot-parent</artifactid>
<version>1.5.9.release</version>
</parent>
<groupid>cn.ibo</groupid>
<artifactid>fastdfs</artifactid>
<version>1.0-snapshot</version>
<dependencies>
<dependency>
<groupid>org.springframework.boot</groupid>
<artifactid>spring-boot-starter-web</artifactid>
</dependency>
<dependency>
<groupid>net.oschina.zcx7878</groupid>
<artifactid>fastdfs-client-java</artifactid>
<version>1.27.0.0</version>
</dependency>
<dependency>
<groupid>org.springframework.boot</groupid>
<artifactid>spring-boot-starter-test</artifactid>
<scope>test</scope>
</dependency>
<dependency>
<groupid>org.apache.commons</groupid>
<artifactid>commons-io</artifactid>
<version>1.3.2</version>
</dependency>
</dependencies>
3、新增配置文件fastdfs-client.properties
## fastdfs-client.properties fastdfs.connect_timeout_in_seconds = 5 fastdfs.network_timeout_in_seconds = 30 fastdfs.charset = utf-8 fastdfs.http_anti_steal_token = false fastdfs.http_secret_key = fastdfs1234567890 fastdfs.http_tracker_http_port = 80 fastdfs.tracker_servers = 192.168.174.128:22122
4、编写测试类
4.1) 上传
需求:将window 盘符下的指定文件通过fastfds上传到linux指定目录下
/** * created by 佳先森 on 2019/1/13. * 在此文件中通过fastdsf的client代码访问tracker和storage * 通过client的api代码访问tracker和storage,他们中间走的socket协议 */ public class testfastdfs { // 测试文件上传 @test public void testupload(){ //通过fastdsf的client代码访问tracker和storage string local_filename = "77.jpg";//上传文件名 try { //加载fastdfs客户端的配置文件 clientglobal.initbyproperties("config/fastdfs-client.properties"); system.out.println("network_timeout=" + clientglobal.g_network_timeout + "ms"); system.out.println("charset=" + clientglobal.g_charset); //创建tracker的客户端 trackerclient tracker = new trackerclient(); trackerserver trackerserver = tracker.getconnection(); storageserver storageserver = null; //定义storage的客户端 storageclient1 client = new storageclient1(trackerserver, storageserver); //文件元数据(如文件名称大小等) namevaluepair[] metalist = new namevaluepair[1]; metalist[0] = new namevaluepair("filename",local_filename);//这是个数组,可以继续添加 //执行上传 string fileid = client.upload_file1("e:\\壁纸\\77.jpg", "jpg", metalist); system.out.println("upload success. file id is: " + fileid); //关闭trackerserver的连接 trackerserver.close(); } catch (exception ex) { ex.printstacktrace(); } } }
验证:此时在storage-data目录下可以看到上传的文件
[root@bogon 00]# pwd /home/fastdfs/storage-data/data/00/00 [root@bogon 00]# ll -rw-r--r--. 1 root root 55310 1月 13 21:15 wkiugfw7oesamli9aadydjkgjly905.jpg
4.2) 检索
根据控制台打印的信息,利用文件id检索文件信息
//查询上传的文件 @test public void testsearch(){ try { //加载fastdfs客户端的配置文件 clientglobal.initbyproperties("config/fastdfs-client.properties"); system.out.println("network_timeout=" + clientglobal.g_network_timeout + "ms"); system.out.println("charset=" + clientglobal.g_charset); //创建tracker的客户端 trackerclient tracker = new trackerclient(); trackerserver trackerserver = tracker.getconnection(); storageserver storageserver = null; //定义storage的客户端 storageclient1 client = new storageclient1(trackerserver, storageserver); fileinfo group1 = client.query_file_info("group1", "m00/00/00/wkiugfw7oesamli9aadydjkgjly905.jpg"); fileinfo fileinfo = client.query_file_info1("group1/m00/00/00/wkiugfw7oesamli9aadydjkgjly905.jpg"); system.out.println(group1); system.out.println(fileinfo); //查询文件元信息 namevaluepair[] metadata1 = client.get_metadata1("group1/m00/00/00/wkiugfw7oesamli9aadydjkgjly905.jpg"); for(namevaluepair temp:metadata1){ system.out.print(temp.getname()+"\t"+temp.getvalue()); } //关闭trackerserver的连接 trackerserver.close(); } catch (exception ex) { ex.printstacktrace(); } }
4.3) 下载文件
根据文件id下载文件到指定目录
@test public void download(){ try { //加载fastdfs客户端的配置文件 clientglobal.initbyproperties("config/fastdfs-client.properties"); system.out.println("network_timeout=" + clientglobal.g_network_timeout + "ms"); system.out.println("charset=" + clientglobal.g_charset); //创建tracker的客户端 trackerclient tracker = new trackerclient(); trackerserver trackerserver = tracker.getconnection(); storageserver storageserver = null; //定义storage的客户端 storageclient1 client = new storageclient1(trackerserver, storageserver); //下载 byte[] bytes = client.download_file1("group1/m00/00/00/wkiugfw7oesamli9aadydjkgjly905.jpg"); file file = new file("e:/a.jpg"); fileoutputstream fos = new fileoutputstream(file); fos.write(bytes); fos.close(); //关闭trackerserver的连接 trackerserver.close(); } catch (exception ex) { ex.printstacktrace(); } }
六、文件服务案列
先将文件存放到临时目录,再通过tracker server将临时文件上传到fastdfs中的storage中
注意:得先安装fastdfs-nginx-module,再安装nginx,在编译nginx过程中添加fastdfs-nginx-module,顺序弄反,会报
[emerg] unknown directive "ngx_fastdfs_module" ngix整合fastdfs启动后报错
fastdfs通过tracker服务器,将文件放在storage服务器存储,但是同组存储服务器之间需要进入文件复制,有同步延迟的问题。假设tracker服务器将文件上传到了192.168.4.125,上传成功后文件id已经返回给客户端。此时fastdfs存储集群机制会将这个文件同步到同组存储192.168.4.126,在文件还没有复制完成的情况下,客户端如果用这个文件id在192.168.4.126上取文件,就会出现文件无法访问的错误。而fastdfs-nginx-module可以重定向文件连接到源服务器取文件,避免客户端由于复制延迟导致的文件无法访问错误。
1、安装fastdfs-nginx-module
参考博客:https://blog.csdn.net/u010098331/article/details/51646921
[root@bogon local]# wget https://sourceforge.net/projects/fastdfs/files/fastdfs%20nginx%20module%20source%20code/fastdfs-nginx-module_v1.16.tar.gz [root@bogon local]# tar xf fastdfs-nginx-module_v1.16.tar.gz [root@bogon local]# cd fastdfs-nginx-module/src/ [root@bogon local]# vim config #编辑config文件,执行如下命令进行批量替换并保存退出,目的:去掉local前缀,指定正确路径目录 :%s+/usr/local/+/usr/+g
[root@bogon src]# pwd /usr/local/fastdfs-nginx-module/src [root@bogon src]# cp mod_fastdfs.conf /etc/fdfs [root@bogon local]# cd /etc/fdfs [root@bogon fdfs]# vim mod_fastdfs.conf
#修改内容如下: connect_timeout=10 base_path=/tmp(默认为/tmp) tracker_server=192.168.174.128:22122 storage_server_port=23000(默认配置为23000) url_have_group_name = true store_path0=/home/fastdfs/storage-data group_name=group1(默认配置为group1)
[root@bogon fastdfs-5.05]# pwd /usr/java/fastdfs-5.05 [root@bogon fastdfs-5.05]# cd conf [root@bogon conf]# cp http.conf /etc/fdfs [root@bogon conf]# cp mime.types /etc/fdfs
安装nginx
[root@bogon local]# tar -zxvf nginx-1.6.0 [root@bogon local]# cd nginx-1.6.0/ [root@bogon nginx-1.6.0]# ./configure --prefix=/usr/local/nginx --add-module=/usr/local/fastdfs-nginx-module/src [root@bogon nginx-1.6.0]#make && make install
[root@bogon local]# cd nginx [root@bogon nginx]# cd conf [root@bogon conf]# vim nginx.conf
location /group1/m00 { root /home/fastdfs/storage-data; ngx_fastdfs_module; }
[root@bogon conf]# cd .. [root@bogon nginx]# cd sbin/ [root@bogon sbin]# ./nginx ngx_http_fastdfs_set pid=11005
配置nginx:
server { listen 80; server_name localhost; #charset koi8-r; #access_log logs/host.access.log main; location /group1/m00 { root /home/fastdfs/storage-data/data; #storage_path0 ngx_fastdfs_module; #虚拟模块 } }
此时根据刚刚测试类中上传的文件利用nginx访问:如果nginx能够正常访问表示配置nginx成功
这是通过http的方式(集成nginx)访问上传图片
七、搭建文件管理服务
文件管理服务提供通过http方式上传文件、删除文件、查询文件的功能,管理员通过文件管理服务队文件服务器上的文件进行管理
文件管理服务采用spring boot开发,文件管理服务通过与fastdfs交互最终将用户上传的文件存储到fastdfs上
1、建立实体类
/** * created by 佳先森 on 2019/1/13. */ public class filesystem { private string fileid; private string filepath; private long filesize; private string filename; private string filetype; //省去set/get }
2、编写配置文件application.yml
server: port: 22100 fastdfs: #文件上传临时目录 upload_location: e:\\download\\
3、建立controller
/** * created by 佳先森 on 2019/1/13. */ @restcontroller @requestmapping("/filesystem") public class fileservercontroller { @value("${fastdfs.upload_location}") //读取配置文件 private string upload_location; @postmapping("/upload") @responsebody public filesystem upload(@requestparam("file") multipartfile file) throws ioexception { //先将文件存储在web服务器上(本机),再使用fastdfs的client将文件到fastdfs服务器 filesystem filesystem = new filesystem(); //得到文件的原始名称 string originalfilename = file.getoriginalfilename(); string extention = originalfilename.substring(originalfilename.lastindexof(".")); //重构上传文件名 string newfilename = uuid.randomuuid()+extention; //定义file,使用file存储上传的文件 file file1 =new file(upload_location+newfilename); file.transferto(file1); //获取新上传文件的物理路径 string newfilepath = file1.getabsolutepath(); try { //加载fastdfs客户端的配置文件 clientglobal.initbyproperties("config/fastdfs-client.properties"); system.out.println("network_timeout=" + clientglobal.g_network_timeout + "ms"); system.out.println("charset=" + clientglobal.g_charset); //创建tracker的客户端 trackerclient tracker = new trackerclient(); trackerserver trackerserver = tracker.getconnection(); storageserver storageserver = null; //定义storage的客户端 storageclient1 client = new storageclient1(trackerserver, storageserver); //文件元数据(如文件名称大小等) namevaluepair[] metalist = new namevaluepair[1]; metalist[0] = new namevaluepair("filename",originalfilename);//这是个数组,可以继续添加 //执行上传,将上传成功的存放在web服务器(本机)上的文件上传到fastdfs string fileid = client.upload_file1(newfilepath, extention, metalist); system.out.println("upload success. file id is: " + fileid); filesystem.setfileid(fileid); filesystem.setfilepath(fileid); filesystem.setfilename(originalfilename); //通过调用service即dao将文件路径存储到数据库中 //....todo //关闭trackerserver的连接 trackerserver.close(); } catch (exception ex) { ex.printstacktrace(); } return filesystem; } }
4、编写启动类
/** * created by 佳先森 on 2019/1/13. */ @springbootapplication public class fileserviceapplication { public static void main(string[] args) { springapplication.run(fileserviceapplication.class); } }
5、编写前端界面
<!doctype html public "-//w3c//dtd html 4.01//en" "http://www.w3.org/tr/html4/strict.dtd"> <html lang="en"> <head> <meta http-equiv="content-type" content="text/html;charset=utf-8"> <title>upload</title> </head> <!-- 引入样式 --> <link rel="stylesheet" href="https://unpkg.com/element-ui/lib/theme-chalk/index.css"> <body> <div id="clj"> <el-upload action="" list-type="picture-card" :on-preview="handlepicturecardpreview" :on-remove="handleremove"> <i class="el-icon-plus"></i> </el-upload> <el-dialog :visible.sync="dialogvisible"> <img width="100%" :src="dialogimageurl" alt=""> </el-dialog> </div> <script type="text/javascript" src="vue.min.js"></script> <!-- 引入组件库 --> <script src="https://unpkg.com/element-ui/lib/index.js"></script> <script> new vue({ el: '#clj', data: { dialogvisible: false, dialogimageurl: '' },
methods: {
handlepicturecardpreview(file){ #预览
this.dialogimageurl ="http://192.168.174.128/"+file.response.filepath #注意:预览时访问的图片地址是通过nginx代理的,需要配置图片所在宿主机地址
this.dialogvisible = true;
},
handleremove(){}
}
}) </script> </body> </html>
6、nginx中配置页面访问路径
#图片服务测试(图片页面) server { listen 7283; server_name localhost; location / { root /usr/java/javademo/springboot-vue-fastdfs/upload/; index index.html; } }
此时访问http://192.168.174.128:7283/upload.html,若成功访问到页面,则配置成功
但是这样远远不够的,因为页面中的action配置的请求后台的action是访问不到的,存在跨域问题(页面与后端代码不在同一台宿主机上),解决跨域问题常见的思路就是在nginx中配置代理
#图片服务测试(图片页面) server { listen 7283; server_name localhost; #这里配置的是页面所在宿主机ip location / { root /usr/java/javademo/springboot-vue-fastdfs/upload/; index index.html; } location ^~ /filesystem/ { #配置代理 proxy_pass http://192.168.2.174:22100/filesystem/; #注意这里配置的是后端代码所在宿主机ip } }
此时上传和预览功能已经完成
克隆源码地址:git@gitee.com:mr_jiaxiansen/fastdfs.git
下一篇: 利用npm 安装删除模块的方法
推荐阅读