需求是這樣的, 在服務器上有 運營上傳的zip 包,內容是用戶的照片,我需要做的是 獲取這些照片上傳,並保存到 數據庫。 這里面的 上傳照片,保存數據庫都不難,主要問題是解壓zip包,和刪除zip 包,以及 解壓后的文件。
之前在網上找的解壓的文件的代碼,都存在同一個問題,就是解壓之后,無法刪除 zip 包。查百度說是 資源占用,可是我已經把所有的流都關閉了哇。
我把解壓和刪除分成兩部分用 jUnit 測試: 第一次測試解壓,並不刪除包;第二次只刪除包。我發現,解壓方法和刪除方法在同一個test 里面測試的話,zip 包刪不掉,要是單獨一個test 測試刪除方法的話,zip 包是可以刪除的。總結一下,在同一個線程里面 解壓之后包刪不掉。這肯定是資源在占用中。
我在代碼里面加了個垃圾回收(System.gc();),依然不行...
我仍然不放棄,還是在 百度上找代碼,看看別人有沒有遇到這樣的問題,我看了好多,也測試了好多代碼,發現博客好多都是復制的,或許是其他人看了這個博客解決了問題,然后自己 不想總結,於是乎,就把 這個博客 復制到自己的 博客里,這雖然省事了,但畢竟不是自己的原創 ,我是一個支持原創的人。(雖然我可能也有復制過,但是我沒有發布哇)。看了這么多 依然不行。突然我的扣扣亮了一下,老大發來一張圖片。
果然精辟呀,加上去之后E盤的文件真的消失了。。
困擾了我一上午的問題就這樣解決了,真是應了我的那句話,"越是看上去很難奇怪的問題,越是最簡單的問題." "奇怪的問題",只是一個概念,只是存在我心里面的,所以這句話也只能是我跟我自己說。
不說了,看代碼吧.一共有兩個類 PhotoTaskService.java 和 ZipUtil.java
package com.hupu.smart.user.service.photoTask; import com.hupu.smart.user.domain.UserPhoto; import com.hupu.smart.user.interfaces.UserPhotoService; import com.hupu.smart.user.service.util.UploadImageUtil; import com.hupu.smart.user.service.util.ZipUtil; import org.slf4j.Logger; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Service; import java.io.*; import java.util.ArrayList; import java.util.List; /** * Created by Wangjunnan on 2017/4/1. */ @Service("photoTaskService") public class PhotoTaskService { @Value("#{configProperties['oss_endpoint_url']}") private String ENDPOINT; @Value("#{configProperties['oss_access_id']}") private String ACCESS_ID; @Value("#{configProperties['oss_access_key']}") private String ACCESS_KEY; @Value("#{configProperties['oss_domain']}") private String OSS_DOMAIN; @Value("#{configProperties['oss_bucket']}") private String BUCKET; @Autowired private UserPhotoService photoService; private Logger log = org.slf4j.LoggerFactory.getLogger(PhotoTaskService.class); private String url="E:\\image"; // 檢查是不是存在 zip 的 文件,並返回 這些文件 @Scheduled(cron="0/10 * * * * ? ") public void searchZipFile(){ log.info("解壓用戶照片任務開始執行"); List<File> list= new ArrayList<>(); File file=new File(url); if (file.isDirectory()) { File[] files = file.listFiles(); for (File f : files) { if (f.getName().endsWith(".zip")) { // zip文件 判斷 是否存在 list.add(f); } } } if(null !=list && list.size()>0){ // 如果有文件就解壓 並保存 releaseZipFile(list); } log.info("解壓用戶照片任務執行完畢"); // 刪除 解壓后的文件 deleteDir(new File(url)); } //解壓 zip文件 返回 List<userPhoto> public void releaseZipFile(List<File> files){ List<UserPhoto> photos=new ArrayList<>(); try{ for(File f:files){ List<File> fileList = ZipUtil.upZipFile(f, url); delZipFile(); // 解壓完成之后刪除 zip 包 for (int i=0;i<fileList.size();i++){ String userId=fileList.get(i).getParentFile().getName(); // 上傳 String photoUrl = UploadImageUtil.uploadPic(fileList.get(i), "photo", ENDPOINT, ACCESS_ID, ACCESS_KEY, BUCKET, OSS_DOMAIN); log.info("用戶圖片上傳成功"); // 封裝成 user_photo 對象 UserPhoto photo=new UserPhoto(); photo.setUserId(userId); photo.setPhotoPath(photoUrl); photo.setState(0); photo.setIsDel(0); photos.add(photo); } } // 保存到數據庫 photoService.addUserPhoto(photos); log.info("數據已保存到數據庫"); }catch (Exception e){ log.error("照片上傳出錯"+e.getMessage()); } } // 刪除zip 文件 public void delZipFile(){ File file=new File(url); if (file.isDirectory()) { File[] files = file.listFiles(); for (File f : files) { if (f.getName().endsWith(".zip")) { // zip文件 判斷 是否存在 if(f.delete()) { log.info("zip文件已經刪除"); }else{ log.info("zip文件刪除失敗"); } } } } } // 刪除解壓后的文件 private boolean deleteDir(File dir) { if (dir.isDirectory()) { String[] children = dir.list(); //遞歸刪除目錄中的子目錄下 for (int i=0; i<children.length; i++) { boolean success = deleteDir(new File(dir, children[i])); if (!success) { return false; } } } // 目錄此時為空,可以刪除 return dir.delete(); } }
ZipUtil.java
package com.hupu.smart.user.service.util; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.util.ArrayList; import java.util.Enumeration; import java.util.List; import org.apache.tools.zip.ZipEntry; import org.apache.tools.zip.ZipFile; import org.slf4j.Logger; /** * 壓縮或解壓zip: * 由於直接使用java.util.zip工具包下的類,會出現中文亂碼問題,所以使用ant.jar中的org.apache.tools.zip下的工具類 * @author Administrator */ public class ZipUtil { private static byte[] _byte = new byte[1024] ; private static Logger log = org.slf4j.LoggerFactory.getLogger(ZipUtil.class); /** * 對.zip文件進行解壓縮 * @param zipFile 解壓縮文件 * @param descDir 壓縮的目標地址,如:D:\\測試 或 /mnt/d/測試 * @return */ public static List<File> upZipFile(File zipFile, String descDir) { List<File> _list = new ArrayList<>() ; try { ZipFile _zipFile = new ZipFile(zipFile , "GBK") ; for( Enumeration entries = _zipFile.getEntries() ; entries.hasMoreElements() ; ){ ZipEntry entry = (ZipEntry)entries.nextElement() ; File _file = new File(descDir + File.separator + entry.getName()) ; if( entry.isDirectory() ){ _file.mkdirs() ; }else{ File _parent = _file.getParentFile() ; if( !_parent.exists() ){ _parent.mkdirs() ; } InputStream _in = _zipFile.getInputStream(entry); OutputStream _out = new FileOutputStream(_file) ; int len ; while( (len = _in.read(_byte)) > 0){ _out.write(_byte, 0, len); } if (null !=_out){ _out.flush(); _out.close(); } if (null !=_in){ _in.close(); } _list.add(_file); } }
// 加了這行代碼之后就可以刪除了 _zipFile.close();// 加了這行代碼之后就可以刪除了 // 加了這行代碼之后就可以刪除了
} catch (IOException e) { log.error(e.getMessage()); } return _list ; } }
然后用junit 測試一下就可以了
@Test public void testPhoto(){ taskService.searchZipFile(); }