自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Android用Retrofit 2實(shí)現(xiàn)多文件上傳實(shí)戰(zhàn)

移動(dòng)開發(fā) Android
本文作為階段性小結(jié),將使用結(jié)合Python中的Flask框架實(shí)現(xiàn)Android端多文件上傳功能。如果讀者沒有使用過Python中的Flask也沒有關(guān)系,可以只看Android客戶端部分,畢竟客戶端工程師只使用API也是可以的。

前一段時(shí)間我翻譯了Future Studio的Retrofit2教程,從中也學(xué)習(xí)到了一些Retrofit2的使用方法,如果你最近也打算入手學(xué)習(xí),我博客上Retrofit教程,你也許可以參考下:Retrofit教程

本文作為階段性小結(jié),將使用結(jié)合Python中的Flask框架實(shí)現(xiàn)Android端多文件上傳功能。如果讀者沒有使用過Python中的Flask也沒有關(guān)系,可以只看Android客戶端部分,畢竟客戶端工程師只使用API也是可以的。

1.實(shí)驗(yàn)效果

Android端操作截圖 

 

 

Android端操作截圖 

Server端接收到的圖片 

 

 

Server端接收到的圖片 

2. Server端實(shí)戰(zhàn)

Server端負(fù)責(zé)接收保存客戶端上傳來的圖片并提供訪問圖片的能力,Server有很多技術(shù)可以實(shí)現(xiàn),Python作為一門具有強(qiáng)大的第三方庫的語言,擁有很多web服務(wù)框架,如Flask,Django等。筆者采用Flask框架,F(xiàn)lask是微框架,實(shí)現(xiàn)小型功能十分方便,筆者實(shí)現(xiàn)的多文件上傳功能,程序不超過30行。

下面具體來看看。

2.1 環(huán)境安裝

筆者使用的Python版本為3.4,可以去 Python3.4下載 選擇下載適合自己系統(tǒng)的版本。完整安裝Python教程請(qǐng)自行搜索。

Python安裝完成后需要安裝Server端程序依賴庫。通過pip安裝:

  1. pip install Flask 
  2. pip install werkzeug  

2.2 程序?qū)崿F(xiàn)

首先要引入依賴庫:

  1. from flask import Flask,request,send_from_directory,jsonify 
  2. import os 
  3. from werkzeug import secure_filename  

本實(shí)驗(yàn)需要上傳文件,需要將所上傳文件的文件類型以及文件名做出限制,防止某些破壞服務(wù)器的程序運(yùn)行,另外有些非法文件名如:

filename = "../../../../home/username/.bashrc"

如果黑客們能夠操作這樣的文件,對(duì)服務(wù)器系統(tǒng)來說,將是致命打擊。所以werkzeug提供了secure_filename對(duì)上傳文件的文件名進(jìn)行合法校驗(yàn)。

判斷文件后綴是否合法

  1. ALLOWED_EXTENSIONS=set(['png','jpg','jpeg','gif']) 
  2. def allowed_file(filename): 
  3. return '.' in filename and filename.rsplit('.',1)[1] in ALLOWED_EXTENSIONS  

接收上傳文件的函數(shù)代碼如下:

  1. @app.route('/upload',methods=['POST']) 
  2. def upload_file(): 
  3. if request.method=='POST'
  4. for k in request.files: 
  5. file = request.files[k] 
  6. image_urls = [] 
  7. if file and allowed_file(file.filename): 
  8. filename=secure_filename(file.filename) 
  9. file.save(os.path.join(app.config['IMAGE_FOLDER'],filename)) 
  10. image_urls.append("images/%s"%filename) 
  11. return jsonify({"code":1,"image_urls":image_urls})  

Flask支持GET,POST,PUT,DELETE等HTTP請(qǐng)求方式,使用裝飾器進(jìn)行修飾,類似于Java中的注解概念,/upload為客戶端請(qǐng)求的相對(duì)地址,請(qǐng)求方式限制為POST.根據(jù)request內(nèi)置對(duì)象,可以訪問客戶端發(fā)來的文件,將文件檢查后保存在本地,其中image_urls為上傳后的圖片的相對(duì)地址數(shù)組。***將圖片的地址以json格式返回給客戶端。

完整的Server端代碼如下:

  1. from flask import Flask,request,send_from_directory,jsonify 
  2.  import os 
  3.  from werkzeug import secure_filename 
  4.  
  5. app = Flask(__name__) 
  6.  app.config['IMAGE_FOLDER'] = os.path.abspath('.')+'\\images\\' 
  7.  ALLOWED_EXTENSIONS=set(['png','jpg','jpeg','gif']) 
  8.  
  9. def allowed_file(filename): 
  10.  return '.' in filename and filename.rsplit('.',1)[1] in ALLOWED_EXTENSIONS 
  11.  
  12. @app.route('/upload',methods=['POST']) 
  13.  def upload_file(): 
  14.  if request.method=='POST'
  15.  for k in request.files: 
  16.  file = request.files[k] 
  17.  print(file) 
  18.  image_urls = [] 
  19.  if file and allowed_file(file.filename): 
  20.  filename=secure_filename(file.filename) 
  21.  file.save(os.path.join(app.config['IMAGE_FOLDER'],filename)) 
  22.  image_urls.append("images/%s"%filename) 
  23.  return jsonify({"code":1,"image_urls":image_urls}) 
  24.  
  25. #讓文件映射訪問,否則默認(rèn)只能訪問static文件夾中的文件 
  26.  @app.route("/images/<imgname>",methods=['GET']) 
  27.  def images(imgname): 
  28.  return send_from_directory(app.config['IMAGE_FOLDER'],imgname) 
  29.  
  30. if __name__ == "__main__"
  31.  
  32. # 檢測 IMAGE_FOLDER 是否存在 
  33.  if not os.path.exists(app.config['IMAGE_FOLDER']): 
  34.  os.mkdir(app.config['IMAGE_FOLDER']) 
  35.  app.run("192.168.1.102",debug=True 

這里有一個(gè)小技巧,寫完Server端代碼后可以使用Postman進(jìn)行測試,測試成功后再進(jìn)行客戶端程序開發(fā)。 

 

 

 

3. 客戶端開發(fā)

因?yàn)樯婕拔募纳蟼?,筆者這里以圖片為例進(jìn)行上傳實(shí)驗(yàn),圖片上傳除了重頭戲Retrofit之外,還需要選擇圖片,筆者這里推薦一個(gè)模仿微信的圖片選擇庫 ImagePicker .

3.1 添加依賴庫

圖片加載庫筆者喜歡使用Glide

  1. compile 'com.squareup.retrofit2:retrofit:2.1.0' 
  2. compile 'com.squareup.retrofit2:converter-gson:2.1.0' 
  3. compile 'com.github.bumptech.glide:glide:3.7.0' 
  4. compile 'com.lzy.widget:imagepicker:0.4.1'  

3.2 程序?qū)崿F(xiàn)

如果沒有接觸過Retrofit 2,可以來我的博客Retrofit教程 了解。

Retrofit2 是一個(gè)支持RESTful API的請(qǐng)求庫,實(shí)際上只是對(duì)API請(qǐng)求方式的封裝,真正的網(wǎng)絡(luò)請(qǐng)求由OkHttp發(fā)出。

Retrofit2一般會(huì)定義一個(gè)ServiceGenerator類,用于動(dòng)態(tài)生成Retrofit對(duì)象。

  1. public class ServiceGenerator { 
  2. public static final String API_BASE_URL = "http://192.168.1.102:5000/"
  3. private static OkHttpClient.Builder httpClient = new OkHttpClient.Builder(); 
  4.   
  5. private static Retrofit.Builder builder = 
  6. new Retrofit.Builder() 
  7. .baseUrl(API_BASE_URL) 
  8. .addConverterFactory(GsonConverterFactory.create()); 
  9.   
  10. public static <S> S createService(Class<S> serviceClass) { 
  11. Retrofit retrofit = builder.client(httpClient.build()).build(); 
  12. return retrofit.create(serviceClass); 
  13.  

具體的API操作由FlaskClient接口操作,

  1. public interface FlaskClient { 
  2.     //上傳圖片 
  3.     @Multipart 
  4.     @POST("/upload"
  5.     Call<UploadResult> uploadMultipleFiles(@PartMap Map<String,RequestBody> files); 
  6.  

上傳文件需要使用@Multipart關(guān)鍵字注解,@POST表明HTTP請(qǐng)求方式為POST,/upload為請(qǐng)求服務(wù)器的相對(duì)地址,uploadMultipleFiles是自定義的方法名,參數(shù)為Map<String,RequestBody> files即多個(gè)文件組成的Map對(duì)象,@PartMap表明這是多文件上傳,如果單文件可以使用@Part MultipartBody.Part file,方法的返回類型默認(rèn)為Response,由于我們已經(jīng)開發(fā)了Server端,所以知道Server端的返回?cái)?shù)據(jù)格式為Json,因此我們針對(duì)返回?cái)?shù)據(jù)格式新建一個(gè)UploadResut類。

  1. public class UploadResult { 
  2.      public int code; // 1 
  3.      public List<String> image_urls; 
  4.  

界面布局如圖所示: 

 

 

 

點(diǎn)擊Upload按鈕后執(zhí)行上傳操作,核心的方法:

  1. public void uploadFiles() { 
  2.     if(imagesList.size() == 0) { 
  3.         Toast.makeText(MainActivity.this, "不能不選擇圖片", Toast.LENGTH_SHORT).show(); 
  4.         return
  5.     } 
  6.     Map<String, RequestBody> files = new HashMap<>(); 
  7.     final FlaskClient service = ServiceGenerator.createService(FlaskClient.class); 
  8.     for (int i = 0; i < imagesList.size(); i++) { 
  9.         File file = new File(imagesList.get(i).path); 
  10.         files.put("file" + i + "\"; filename=\"" + file.getName(), RequestBody.create(MediaType.parse(imagesList.get(i).mimeType), file)); 
  11.     } 
  12.     Call<UploadResult> call = service.uploadMultipleFiles(files); 
  13.     call.enqueue(new Callback<UploadResult>() { 
  14.         @Override 
  15.         public void onResponse(Call<UploadResult> call, Response<UploadResult> response) { 
  16.             if (response.isSuccessful() && response.body().code == 1) { 
  17.                 Toast.makeText(MainActivity.this, "上傳成功", Toast.LENGTH_SHORT).show(); 
  18.                 Log.i("orzangleli""---------------------上傳成功-----------------------"); 
  19.                 Log.i("orzangleli""基礎(chǔ)地址為:" + ServiceGenerator.API_BASE_URL); 
  20.                 Log.i("orzangleli""圖片相對(duì)地址為:" + listToString(response.body().image_urls,',')); 
  21.                 Log.i("orzangleli""---------------------END-----------------------"); 
  22.             } 
  23.         } 
  24.         @Override 
  25.         public void onFailure(Call<UploadResult> call, Throwable t) { 
  26.             Toast.makeText(MainActivity.this, "上傳失敗", Toast.LENGTH_SHORT).show(); 
  27.         } 
  28.     }); 
  29.  

其中構(gòu)建上傳多文件的方法的參數(shù)較為關(guān)鍵,MediaType.parse(imagesList.get(i).mimeType)獲取圖片的mimeType,如果指定錯(cuò)誤,可能會(huì)導(dǎo)致上傳失敗。

  1. Map<String, RequestBody> files = new HashMap<>(); 
  2. final FlaskClient service = ServiceGenerator.createService(FlaskClient.class); 
  3. for (int i = 0; i < imagesList.size(); i++) { 
  4.      File file = new File(imagesList.get(i).path); 
  5.      files.put("file" + i + "\"; filename=\"" + file.getName(), RequestBody.create(MediaType.parse(imagesList.get(i).mimeType), file)); 
  6.  

集成Callback借口的匿名回調(diào)類的onResponse方法的第二個(gè)參數(shù)為服務(wù)器響應(yīng),通過訪問body()方法返回UploadResult類型對(duì)象,接著就可以通過組合ServiceGenerator.API_BASE_URL和response.body().image_urls中每一項(xiàng)訪問上傳完成的圖片。

4. 項(xiàng)目地址

本項(xiàng)目Client端和Server端均以開源,歡迎各位老總們Star。

Client地址: RetrofitMultiFilesUploadClient

Server地址: MultiFileUploadServer

原文鏈接:http://www.orzangleli.com/2017/04/03/2017-04-03_Android%20%E7%94%A8%20Retrofit%202%20%E5%AE%9E%E7%8E%B0%E5%A4%9A%E6%96%87%E4%BB%B6%E4%B8%8A%E4%BC%A0%E5%AE%9E%E6%88%98/ 

責(zé)任編輯:龐桂玉 來源: Android開發(fā)中文站
相關(guān)推薦

2009-06-08 16:44:00

Struts2文件上傳

2011-09-14 09:20:03

PhonegapAndroid平臺(tái)

2009-06-03 15:57:29

Struts1.2動(dòng)態(tài)多文件

2011-09-05 15:09:06

Android平臺(tái)Phonegap

2010-04-07 11:09:53

2009-11-16 10:49:43

PHP上傳文件代碼

2009-07-14 17:20:31

Webwork文件上傳

2022-06-13 14:06:33

大文件上傳前端

2024-11-12 09:54:23

2009-11-16 10:40:02

PHP上傳文件代碼

2009-07-23 10:37:43

2016-10-20 19:36:01

androiddagger2依賴注入

2009-11-24 14:52:45

PHP動(dòng)態(tài)多文件上傳

2009-11-16 10:16:24

PHP文件上傳

2009-11-24 13:15:35

Zend框架PHP上傳文件

2009-11-16 10:25:40

PHP上傳文件

2009-07-06 17:11:38

Servlet文件上傳

2009-06-26 13:46:13

Struts

2009-07-08 09:29:58

WebWork

2009-06-25 15:50:03

Struts2教程上傳任意多個(gè)文件
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)