通常開發中,都會把圖片上傳到一個目錄,而後將目錄和文件名拼接存儲在數據庫中,可是,這種方法若是沒弄好的話可能有必定的缺陷。html
若項目搬遷,即時這臺服務器自己還在用,存放在服務器的跟項目相關的圖片也要跟着搬遷,同時還要修改代碼,會形成較多麻煩。若是搭建一個專門存放圖片的服務器,並且經過ftp傳輸,那麼你的代碼就不用改了。前端
關於Docker和Nginx的介紹和優勢在這裏就不展開了,網上博客和教程也有不少。java
環境搭建有些模塊可能不夠詳細,若是想知道更多的話能夠查看其餘資料,這裏只是簡單操做。mysql
yum install docker
docker pull nginx:1.16.0
1.16.0 是版本號,能夠經過 docker search nginx 查詢鏡像;拉取其餘鏡像好比 tomcat、mysql 等同理。nginx
mkdir -p /home/nginx/www /home/nginx/logs /home/nginx/conf
docker run -d -p 80:80 -p 443:443 --name nginx-server -v /home/nginx/www:/usr/share/nginx/html
-v /home/nginx/conf/nginx.conf:/etc/nginx/nginx.conf -v /home/nginx/logs:/var/log/nginx nginx
在/home/nginx/conf 中新建文件 nginx.conf,並寫入以下配置:spring
user root; worker_processes 1; error_log /var/log/nginx/error.log warn; pid /var/run/nginx.pid; events { worker_connections 1024; } http { include /etc/nginx/mime.types; default_type application/octet-stream; log_format main '$remote_addr - $remote_user [$time_local] "$request" ' '$status $body_bytes_sent "$http_referer" ' '"$http_user_agent" "$http_x_forwarded_for"'; access_log /var/log/nginx/access.log main; sendfile on; #tcp_nopush on; keepalive_timeout 65; #gzip on; include /etc/nginx/conf.d/*.conf; server { listen 443 ssl; server_name 你的 ip 地址或域名; root /usr/share/nginx/html; ssl_certificate 你的密鑰.pem; ssl_certificate_key 你的密鑰.key; ssl_session_timeout 5m; ssl_session_cache shared:SSL:1m; ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:ECDHE:ECDH:AES:HIGH:!NULL:aNULL:!MD5:!ADH:!RC4; ssl_protocols TLSv1 TLSv1.1 TLSv1.2; ssl_prefer_server_ciphers on; # 跳轉到8080端口,即個人tomcat容器,若是瀏覽器輸入 https://ip/ 則跳轉到tomcat容器(已經運行並映射端口號) location / { root /usr/share/nginx/html; proxy_pass http://ip:8080/; index index.html index.htm; } # 若是瀏覽器輸入 https://ip/images/ 則對應宿主機上的 /home/nginx/www/images/ 由於前面已經掛載了,即宿主機某個目錄和docker容器內的某個文件夾共享數據
# 要先在宿主機的 /home/nginx/www/ 中建立 images 目錄 location ~ /images/ { root /usr/share/nginx/html/; } } server { listen 80; server_name 你的 ip 地址或域名; rewrite ^ https://$host$1 permanent; } }
這裏面有關於ssl的配置,你須要先申請域名,而後搞個ssl證書,再進行配置,這裏也不作詳細介紹。sql
環境配置差很少完成,接下來安裝 vsftpd 搭建 ftpdocker
安裝vsftpd數據庫
yum -y install vsftpd
添加用戶apache
useradd 自定義用戶名
passwd 自定義密碼
修改配置文件
vi /etc/vsftpd/vsftpd.conf
須要修改的地方以下:
# 配置文件默認爲YES,即支持匿名訪問,設置爲不支持 anonymous_enable=NO # 在最後添加這兩行 pasv_min_port=30000 pasv_max_port=30009
保存退出後設置用戶訪問權限
chown ftpadmin /home/nginx/www/image
chmod 777 -R /home/nginx/www/image
完過後測試一波,經過ftp上傳後,經過瀏覽器https請求獲得以下結果:
*補充一點:阿里雲服務器的話記得配置安全組,將21號端口打開,以及44三、80端口,還有30000-30009端口
先在resource中建立 ftpResource.properties 文件,並寫入參數
FTP_ADDRESS=你的IP FTP_PORT=21 FTP_USERNAME=用戶名 FTP_PASSWORD=用戶密碼 FTP_BASE_PATH=/home/nginx/www/images IMAGE_BASE_URL=https://域名/images/
以後寫一個工具類,用於FTP傳輸
import org.apache.commons.net.ftp.FTP; import org.apache.commons.net.ftp.FTPClient; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.PropertySource; import org.springframework.stereotype.Component; import java.io.*; /** * @author Max */ @Component @PropertySource("classpath:ftpResource.properties") public class FtpUtil { /** * FTP_ADDRESS: ftp 服務器ip地址 * FTP_PORT: ftp 服務器port,默認是21 * FTP_USERNAME: ftp 服務器用戶名 * FTP_PASSWORD: ftp 服務器密碼 * FTP_BASE_PATH: ftp 服務器存儲圖片的絕對路徑 * IMAGE_BASE_URL: ftp 服務器外網訪問圖片路徑 */ @Value("${FTP_ADDRESS}") private String FTP_ADDRESS; @Value("${FTP_PORT}") private Integer FTP_PORT; @Value("${FTP_USERNAME}") private String FTP_USERNAME; @Value("${FTP_PASSWORD}") private String FTP_PASSWORD; @Value("${FTP_BASE_PATH}") private String FTP_BASE_PATH; @Value("${IMAGE_BASE_URL}") private String IMAGE_BASE_URL; /** * 上傳圖片 * @param inputStream 輸入流 * @param name 文件名 * @return 圖片 url * @throws IOException IO異常 */ public String uploadImage(InputStream inputStream, String name) throws IOException { FTPClient ftpClient = new FTPClient(); try { System.out.println(FTP_ADDRESS); ftpClient.enterLocalPassiveMode(); ftpClient.connect(FTP_ADDRESS, FTP_PORT); ftpClient.login(FTP_USERNAME, FTP_PASSWORD); ftpClient.changeWorkingDirectory(FTP_BASE_PATH); ftpClient.setFileType(FTP.BINARY_FILE_TYPE); boolean isSucceed = ftpClient.storeFile(name, inputStream); if (isSucceed){ return IMAGE_BASE_URL + name; } }catch (Exception e){ e.printStackTrace(); }finally { ftpClient.logout(); } return IMAGE_BASE_URL + "error"; } }
而後在Service處理邏輯,此處只展現部分代碼
@Autowired private FtpUtil ftpUtil; @Override public int insertImg(MultipartFile file) throws IOException { /* 一、獲取上傳的文件流 inputStream 以及文件名 getOriginalFilename 二、調用 FtpUtil 中的函數,將圖片上傳到圖片服務器並返回 https 地址 三、若返回的是圖片地址,則將其插入數據庫 */ InputStream inputStream = file.getInputStream(); String filename = file.getOriginalFilename(); String picUrl = ftpUtil.uploadImage(inputStream, filename); }
就這樣完事兒,之後客戶端請求過來,經過FTP傳輸到服務器,而後將地址保存到數據庫,前端只要獲取了圖片url就會顯示。
若是以爲對你有幫助,能夠關注公衆號:Max的平常操做,或掃碼關注