Android Oss上传图片的使用示例

前言

前一阵项目中的上传图片改为上传到阿里上,记录一下实现的过程,方便以后查看。

参考资料:官方文档

配置

Android studio添加依赖

dependencies {
  compile 'com.aliyun.dpa:oss-android-sdk:2.4.5'
  compile 'com.squareup.okhttp3:okhttp:3.4.1'
  compile 'com.squareup.okio:okio:1.9.0'
}

直接引入jar包(对Android studio 或者 Eclipse 都适用)

1.在官网下载 sdk 包
2.解压后得到 jar 包,目前包括 aliyun-oss-sdk-android-x.x.x.jar、okhttp-3.x.x.jar 和 okio-1.x.x.jar
3.将以上 3 个 jar 包导入 libs 目录

权限设置

确保AndroidManifest.xml 文件中已经配置了这些权限,否则,SDK 将无法正常工作。

<uses-permission android:name="android.permission.INTERNET"></uses-permission>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>

混淆设置

-keep class com.alibaba.sdk.android.oss.** { *; }
-dontwarn okio.**
-dontwarn org.apache.commons.codec.binary.**

实现过程

首先为了安全起见,采用的是STS鉴权模式,则要用到的数据都是从后台获得然后应用到前台的。

1.创建OSSClient (自己在这里命名为OssService)

OSSClient为OSS 服务的 Android 客户端,它为调用者提供了一系列的方法,可以用来操作,管理存储空间(bucket)和文件(object)等。

public class OssService {
  private OSS oss;
  private String bucket;
  private picResultCallback callback;//回调接口
  private String path=地址(后台告诉);
  public OssService(OSS oss, String bucket,picResultCallback callback) {
    this.oss = oss;
    this.bucket = bucket;
    this.callback=callback;
  }

 /**object字段为图片的上传地址(具体地址的前缀后端给,这个是拼起
  *来的一个路径)
  *localFile图片的本地地址
  *mProgress 进度条
  *img 显示图片的控件
  *type 类型
 */
  public void asyncPutImage(String object, final String localFile, final ProgressBar mProgress, final ImageView img,String type) {
    if (object.equals("")) {
      Log.w("AsyncPutImage", "ObjectNull");
      return;
    }
    File file = new File(localFile);
    if (!file.exists()) {
      Log.w("AsyncPutImage", "FileNotExist");
      Log.w("LocalFile", localFile);
      return;
    }
    // 构造上传请求
    PutObjectRequest put = new PutObjectRequest(bucket, object, localFile);
      put.setCallbackParam(new HashMap<String, String>() {
        {
          put("callbackUrl", path);
          put("callbackBody", "filename=${object}&size=${size}&id=${x:id}&action=${x:action}");
//https://help.aliyun.com/document_detail/31989.html?spm=5176.doc31984.6.883.brskVg
        }
      });
    HashMap<String, String> hashMap=new HashMap<>();
    hashMap.put("x:id",id);
    hashMap.put("x:action",type);
    put.setCallbackVars(hashMap);
    // 异步上传时可以设置进度回调
    put.setProgressCallback(new OSSProgressCallback<PutObjectRequest>() {
      @Override
      public void onProgress(PutObjectRequest request, long currentSize, long totalSize) {
        int progress = (int) (100 * currentSize / totalSize);
        mProgress.setProgress(progress);
      }
    });
    OSSAsyncTask task = oss.asyncPutObject(put, new OSSCompletedCallback<PutObjectRequest, PutObjectResult>() {
      @Override
      public void onSuccess(PutObjectRequest request, final PutObjectResult result) {
       Observable.just(result).observeOn(AndroidSchedulers.mainThread()).subscribe(new Action1<PutObjectResult>() {
          @Override
          public void call(PutObjectResult putObjectResult) {
            mProgress.setVisibility(View.GONE);
            img.setColorFilter(null);
            callback.getPicData(result,localFile);
          }
        });
      }
      @Override
      public void onFailure(PutObjectRequest request, ClientException clientExcepion, ServiceException serviceException) {
        String info = "";
        // 请求异常
        if (clientExcepion != null) {
          // 本地异常如网络异常等
          clientExcepion.printStackTrace();
          info = clientExcepion.toString();
        }
        if (serviceException != null) {
          // 服务异常
          Log.e("ErrorCode", serviceException.getErrorCode());
          Log.e("RequestId", serviceException.getRequestId());
          Log.e("HostId", serviceException.getHostId());
          Log.e("RawMessage", serviceException.getRawMessage());
          info = serviceException.toString();
        }
      }
    });
  }
//成功的回调接口
  public interface picResultCallback {
    void getPicData(PutObjectResult data,String oldPath);
  }
}

2.实现OssService的方法(在activity中)

public OssService initOSS(String endpoint, String bucket) {
  OSSCredentialProvider credentialProvider;
  credentialProvider = new STSGetter(tokenBean);
  //设置网络参数
  ClientConfiguration conf = new ClientConfiguration();
  conf.setConnectionTimeout(15 * 1000); // 连接超时,默认15秒
  conf.setSocketTimeout(15 * 1000); // socket超时,默认15秒
  conf.setMaxConcurrentRequest(5); // 最大并发请求书,默认5个
  conf.setMaxErrorRetry(2); // 失败后最大重试次数,默认2次
  OSS oss = new OSSClient(getApplicationContext(), endpoint, credentialProvider, conf);
  return new OssService(oss, bucket, this);
}

3.重载OSSFederationCredentialProvider生成自己的获取STS的功能(一般自动获得token写在这里,在getFederationToken()方法中,告诉它你获得token的规则即可)

1>官方demo代码(自动更新token)

public class OSSAuthCredentialsProvider extends OSSFederationCredentialProvider {
  private String mAuthServerUrl;
  private AuthDecoder mDecoder;
  public OSSAuthCredentialsProvider(String authServerUrl) {
    this.mAuthServerUrl = authServerUrl;
  }

  /**
   * set auth server url
   * @param authServerUrl
   */
  public void setAuthServerUrl(String authServerUrl) {
    this.mAuthServerUrl = authServerUrl;
  }

  /**
   * set response data decoder
   * @param decoder
   */
  public void setDecoder(AuthDecoder decoder) {
    this.mDecoder = decoder;
  }

  @Override
  public OSSFederationToken getFederationToken() throws ClientException {
    OSSFederationToken authToken;
    String authData;
    try {
      URL stsUrl = new URL(mAuthServerUrl);
      HttpURLConnection conn = (HttpURLConnection) stsUrl.openConnection();
      conn.setConnectTimeout(10000);
      InputStream input = conn.getInputStream();
      authData = IOUtils.readStreamAsString(input, OSSConstants.DEFAULT_CHARSET_NAME);
      if (mDecoder != null) {
        authData = mDecoder.decode(authData);
      }
      JSONObject jsonObj = new JSONObject(authData);
      int statusCode = jsonObj.getInt("StatusCode");
      if (statusCode == 200) {
        String ak = jsonObj.getString("AccessKeyId");
        String sk = jsonObj.getString("AccessKeySecret");
        String token = jsonObj.getString("SecurityToken");
        String expiration = jsonObj.getString("Expiration");
        authToken = new OSSFederationToken(ak, sk, token, expiration);
      } else {
        String errorCode = jsonObj.getString("ErrorCode");
        String errorMessage = jsonObj.getString("ErrorMessage");
        throw new ClientException("ErrorCode: " + errorCode + "| ErrorMessage: " + errorMessage);
      }
      return authToken;
    } catch (Exception e) {
      throw new ClientException(e);
    }
  }

  public interface AuthDecoder {
    String decode(String data);
  }
}

2>自己的代码(因为自己的所有数据都是从后台获得的,而且结合rxjava没有想到可以返回数据的方式,所以采用手动更新token的方式)

手动更新token的具体操作:

首先token的值存在MyApp中,第一次在进入需要用到token界面时候,先获得token的值更新MyApp中的值并记录当下的时间,如果下次再次进入任何一个需要用到token的界面的时候,则判断时间是否过期,过期则重新请求token更新token的值。

public class STSGetter extends OSSFederationCredentialProvider {
  private OSSFederationToken ossFederationToken;
  String ak;
  String sk;
  String token ;
  String expiration ;
  public STSGetter(TokenBean bean) {
    this.ak = bean.getCredentials().getAccessKeyId();
    this.sk = bean.getCredentials().getAccessKeySecret();
    this.token = bean.getCredentials().getSecurityToken();
    this.expiration = bean.getCredentials().getExpiration();
  }
  public OSSFederationToken getFederationToken() {
  return new OSSFederationToken(ak,sk,token,expiration);
  }
}

4.实例化OSSClient,调用上传图片方法

//实例化OSSClient (自己是在onCreate()中实例化的,当然考虑到token的过期问题,也有在onResume()中再次实例化一次)
ossService = initOSS(tokenBean.getBucket().getEndPoint(), tokenBean.getBucket().getBucketName());
//上传图片,需要根据自己的逻辑传参数
ossService.asyncPutImage(图片在阿里上的存储路径, 本地路径, ...);

5.回调处理图片逻辑

/**
 * 对图片上传回来的数据进行处理
 * @param data
 */
  @Override
  public void getPicData(PutObjectResult data, String oldPath) {
    Gson gson = new Gson();
    OssUploadImage uploadImage = gson.fromJson(data.getServerCallbackReturnBody(), OssUploadImage.class);
          ........逻辑自己写吧
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android图片压缩上传之基础篇

    在android程序开发中我们经常见到需要上传图片的场景,在这里有个技术点,需要把图片压缩处理,然后再进行上传.这样可以减少流量的消耗,提高图片的上传速度等问题. 关于android如何压缩,网上的资料也是很多,但大多数都是代码片段,讲解压缩步骤,而没有一个实用的工具类库.那么如何将压缩算法封装成一个实用工具库呢?其中会遇到些什么问题,比如: 1.需要压缩的图片有多少 2.压缩后的图片是覆盖还是保存到另外的目录 3.如果是另存目录需要将原始图片删除吗 4.如果改变压缩后的图片的尺寸大小是按照原图

  • Android图片上传实现预览效果

    首先具体分析一下,实现的功能,其中需求分析是必不可少的,需求.逻辑清除之后,再上手写代码,思路会很清晰. 1.多图上传首先得选择图片(这里项目需求是既可以拍照上传也可以从相册中选择) 2.拍照上传很简单了网上也有很多例子,调用照相机,返回uri,获取图片 3.从相册中选择图片  3.1 获取手机中的所有图片  3.2 将图片存到自定义图片数组中显示  3.3 自定义ViewPager浏览图片 主要的逻辑大体是这样,下面具体看一下实现: 一.首先看一下界面: <com.view.NoScrollG

  • android文件上传示例分享(android图片上传)

    主要思路是调用系统文件管理器或者其他媒体采集资源来获取要上传的文件,然后将文件的上传进度实时展示到进度条中. 主Activity 复制代码 代码如下: package com.guotop.elearn.activity.app.yunpan.activity; import java.io.File;import java.io.FileNotFoundException;import java.io.IOException; import android.app.Activity;impor

  • Android 开发 使用WebUploader解决安卓微信浏览器上传图片中遇到的bug

    先给大家分析下微信浏览器上传图片bug的原因 微信在新版本中采用的是自己的X5内核浏览器,而在较老的版本中还有可能是安卓的原生浏览器.具体的环境我也不太了解,但是经过实际多台安卓机型的测试,我采取的方案可以基本确保在安卓机中微信浏览器的成功上传.苹果机型没问题,因为微信的ios客户端使用的是Safari的内核,没有各种坑,且效果最好. 这里给出一个 WebUploader 官方关于移动端适配的 issues 链接.里面提供的方法确实有效,但就是解决的方案并没有很清楚的展示出来,从该issues中

  • Android中Okhttp3实现上传多张图片同时传递参数

    之前上传图片都是直接将图片转化为io流传给服务器,没有用框架传图片. 最近做项目,打算换个方法上传图片. Android发展到现在,Okhttp显得越来越重要,所以,这次我选择用Okhttp上传图片. Okhttp目前已经更新到Okhttp3版本了,用法跟之前相比,也有一些差别.在网上找了很多资料, 并和java后台同事反复调试,终于成功上传多张图片,同时传递一些键值对参数. 以下是我对该过程的封装: private static final MediaType MEDIA_TYPE_PNG =

  • android上传图片到PHP的过程详解

    今天在做上传头像的时候,总是提交连接超时错误,报错信息如下:XXXXXXSokcetTimeOutXXXXXXXX 然后自己设置HTTP的超时时间: 复制代码 代码如下: [java] view plaincopyprint? //设置超时时间  httpclient.setTimeout(20000); 再building,runing,还是不行....这就怪了,明明好好的,怎么会突然就变成连接超时了呢!又折腾了一阵子后,也跟后台那边的朋友沟通过,他也测试了上传接口,发现没什么问题,就让我自己

  • Android使用post方式上传图片到服务器的方法

    本文实例讲述了Android使用post方式上传图片到服务器的方法.分享给大家供大家参考,具体如下: /** * 上传文件到服务器类 * * @author tom */ public class UploadUtil { private static final String TAG = "uploadFile"; private static final int TIME_OUT = 10 * 1000; // 超时时间 private static final String CH

  • Android实现本地上传图片并设置为圆形头像

    先从本地把图片上传到服务器,然后根据URL把头像处理成圆形头像. 因为上传图片用到bmob的平台,所以要到bmob(http://www.bmob.cn)申请密钥. 效果图: 核心代码: 复制代码 代码如下: public class MainActivity extends Activity {         private ImageView iv;         private String appKey="";                //填写你的Applicatio

  • Android Retrofit 2.0框架上传图片解决方案

    本文为大家分享了 Android Retrofit 2.0框架上传图片解决方案,具体内容如下 1.单张图片的上传 /** * 上传一张图片 * @param description * @param imgs * @return */ @Multipart @POST("/upload") Call<String> uploadImage(@Part("fileName") String description, @Part("file\&qu

  • Android Oss上传图片的使用示例

    前言 前一阵项目中的上传图片改为上传到阿里上,记录一下实现的过程,方便以后查看. 参考资料:官方文档 配置 Android studio添加依赖 dependencies { compile 'com.aliyun.dpa:oss-android-sdk:2.4.5' compile 'com.squareup.okhttp3:okhttp:3.4.1' compile 'com.squareup.okio:okio:1.9.0' } 直接引入jar包(对Android studio 或者 Ec

  • Android 获取随机验证码功能示例

    验证码功能在各大网站都能用到,下面小编通过实例代码给大家分享Android 获取随机验证码功能,具体代码如下所示: package cn.hk.image; import java.awt.BasicStroke; import java.awt.Color; import java.awt.Font; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.io.IOException; impo

  • Android编程之SurfaceView学习示例详解

    本文实例讲述了Android编程之SurfaceView学习示例.分享给大家供大家参考,具体如下: SurfaceView是View的子类,使用的方式与任何View所派生的类都是完全相同的,可以像其他View那样应用动画,并把它们放到布局中. SurfaceView封装的Surface支持使用本章前面所描述的所有标准Canvas方法进行绘图,同时也支持完全的OpenGL ES库. 使用OpenGL,你可以再Surface上绘制任何支持的2D或者3D对象,与在2D画布上模拟相同的效果相比,这种方法

  • Android异步上传图片到PHP服务器

    原理 Android客户端模拟一个HTTP的Post请求到服务器端,服务器端接收相应的Post请求后,返回响应信息给给客户端. 背景 网上很多上传到java服务器上的,找了好久,找到了上传到php的了,思路跟我当初想的差不多,就是POST过去.废话不多说,直接上图看代码. php代码 <?php $target_path = "./upload/";//接收文件目录 $target_path = $target_path . basename( $_FILES['uploaded

  • Android实现上传图片至java服务器

    这几天有做到一个小的案例,手机拍照.相册照片上传到服务器.客户端和服务器的代码都贴出来: 客户端 AndroidManifest.xml添加以下权限 <uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="android.permission.CAMERA"/> <uses-permissionandroid

  • Android Material Design 阴影实现示例

    本文介绍了Android Material Design 阴影实现示例,分享给大家,具体如下: 让 View 产生阴影有以下 4 种方式: 控制 elevation 使用 OutlineProvider 使用 9 图 使用 MD 风格的控件,如 CardView,FloatingActionButton 0.Z轴的概念 Android 在 5.0 及以后采用了 Material Design 设计语言,引入了 Z 轴的概念,也就是垂直于屏幕的轴,Z 轴会让 View 产生阴影的效果: 想象有一束

  • Android 照片选择区域功能实现示例

    实现 Android 的照片选择区域功能 主要有参考 pqpo/SmartCropper 1, 显示 显示四条边和八个点, 八个点: 4 个角和 4 条边的中点 /* 裁剪区域, 0, 左上 -> LeftTop, 1, 右上 -> RightTop, 2, 右下 -> RightBottom, 3, 左下 -> LeftBottom */ Point[] mCropPoints; // 4 条边的中点 Point[] mEdgeMidPoints; 绘制 protected vo

  • Android实现微信登录的示例代码

    目录 一.布局界面 二.MainActivity.java 微信登录的实现与qq登录类似.不过微信登录比较麻烦,需要拿到开发者资质认证,花300块钱,然后应用的话还得有官网之类的,就是比较繁琐的前期准备工作,如果在公司里,这些应该都不是事,会有相关人提前准备好.在这里我们已经拿到了开发者认证,并且申请到了微信登录的授权. 现在直接介绍mob来实现微信登录的代码,并获取微信的相关数据,比较简单. 一.布局界面 布局界面只需要一个button来触发授权就可以 <Button android:id=&qu

  • Android 画中画模式的实现示例

    画中画支持 Android 8.0(API 级别 26)允许以画中画模式启动 Activity.画中画是一种特殊类型的多窗口模式,最常用于视频播放.使用该模式,用户可以通过固定到屏幕一角的小窗口观看视频,同时在应用之间进行导航或浏览主屏幕上的内容. 画中画利用 Android 7.0 中的多窗口模式 API 来提供固定的视频叠加窗口.要将画中画添加到您的应用中,您需要注册支持画中画的 Activity.根据需要将 Activity 切换为画中画模式,并确保当 Activity 处于画中画模式时,

  • Android实现上传图片功能

    本文实例为大家分享了Android实现上传图片功能的具体代码,供大家参考,具体内容如下 设定拍照返回的图片路径 /**      * 设定拍照返回的图片路径      * @param image 图片路径      * @param i 约定值      */     protected void image(String image, int i) {         //创建file对象,用于存储拍照后的图片,这也是拍照成功后的照片路径         outputImage = new

随机推荐