详解Android系统中跨应用数据分享功能的实现

时间:2021-05-19

一个Andoird应用程序的重要的地方是他们有相互沟通和整合的能力,一个应用程序可以和另一个应用程序交互,接下来我们来看看Android应用之间的内容分享
当你构建Intent的时候,必须要指定Intent的Action触发,定义Intent的Action例如ACTION_SEND,你大概可以猜到指定Action为ACTION_SEND的Intent的作用是从一个Activity发送数据到另一个Activity,甚至跨越不同的进程,将数据发送给另一个Activity,你需要指定数据和他的类型,系统将识别接收的Activity,并显示他们给用户(假如有多个选择的时候)或者立即启动Activity(只要一个选择的时候)
1.发送文本内容:

最直接最常用的是用ACTION_SEND来从一个应用发送文本内容到另一个应用,例如,我们的内置浏览器可以与任何页面当前的URL来共享文字,下面的例子就是共享文本内容

Intent sendIntent = new Intent(); sendIntent.setAction(Intent.ACTION_SEND); //这里是你发送的文本 sendIntent.putExtra(Intent.EXTRA_TEXT, "我用手机测试分享内容"); sendIntent.setType("text/plain"); startActivity(sendIntent);

假如你安装了过滤Action为ACTION_SEND,MIMEtype为“text/plain”的应用程序,这个系统就会启动,假如系统匹配到多个这样子的应用程序,他就会弹出一个Dialog列出所有的应用供用户选择(一个选择器),如下图

可能你觉得上面的代码很简单,我很容易就能写出来,可是你发现了没有,当我们的手机里面没有地图类型的应用程序,你执行startActivity方法,你的程序就会出现Crash(崩溃),所以我们需要先判断下系统是否有这类型的应用程序,判断也很简单

public boolean hasApplication(Intent intent){ PackageManager packageManager = getPackageManager(); //查询是否有该Intent的Activity List<ResolveInfo> activities = packageManager.queryIntentActivities(intent, 0); //activities里面不为空就有,否则就没有 return activities.size() > 0 ? true : false;

所以我们在startActivity的时候要做上面的判断,这样子我们开发出来的程序才会更加的健壮,少一些Crash用户体验也会更好,假如每次都要判断你师傅会觉得烦?所以我们也可以使用Intent的Intent.createChooser方法
他有如下优点
1.1 即使用户选择了默认的方式,选择器仍然会弹出来
1.2 如果没有匹配到这样的应用程序,系统就会提示消息
1.3 你可以指定选择器Dialog的标题

Intent sendIntent = new Intent(); sendIntent.setAction(Intent.ACTION_SEND); sendIntent.putExtra(Intent.EXTRA_TEXT, "我用手机测试分享内容"); sendIntent.setType("text/plain"); startActivity(Intent.createChooser(sendIntent, "我是弹出框的标题"));

注:某些邮件类应用程序,如Gmail,我们可以使用putExtra(Intent.EXTRA_EMAIL, String[]) 添加一个字符串数组到Intent中

2.发送二进制内容

二进制内容的共享使用Action为ACTION_SEND,在设置适当的MIME类型,并且添加putExtra(Intent.EXTRA_STREAM, URI),下面是一个常用的共享图片的代码,也可以共享任何类型的二进制内容

Intent shareIntent = new Intent(); shareIntent.setAction(Intent.ACTION_SEND); shareIntent.putExtra(Intent.EXTRA_STREAM, uriToImage); shareIntent.setType("image/jpeg"); startActivity(Intent.createChooser(shareIntent, getResources().getText(R.string.send_to)));

主要下面几点:

  • 你可以使用MIME type为“ */*” ,他只会匹配处理一般数据流的Activity
  • 接收应用程序需要Uri指定数据的访问权限
  • 在SD卡的文件,我们使用File file = new File(filePath); Uri.fromFile(file),然后把他传递给Intent
  • 在应用程序的目录中,openFileOutput的模式为MODE_WORLD_READABLE,然后我们可以使用getFileStreamPath()返回一个File,然后利用Uri.fromFile(file)传递给Intent
  • 可以扫描图像,视频和音频等媒体文件,添加到系统MediaStore使用SCANFILE()来扫描文件,扫描完成后会调用 onScanCompleted()回调方法来返回一个Uri
  • 图片使用insertImage()方法插入到系统MediaStore也会返回一个图片的Uri

下面是共享图片的例子:

Intent shareIntent = new Intent(); shareIntent.setAction(Intent.ACTION_SEND); File file = new File("mnt/sdcard/share.png"); System.out.println(Uri.fromFile(file)); shareIntent.putExtra(Intent.EXTRA_STREAM, Uri.fromFile(file)); shareIntent.setType("image/jpeg"); startActivity(Intent.createChooser(shareIntent, "共享图片"));

选择QQ空间后,图片就显示在那里了,如下图

我们要共享多个二进制内容利用Action为ACTION_SEND_MULTIPLE,下面是我们共享多张图片

ArrayList<Uri> imageUris = new ArrayList<Uri>(); imageUris.add(imageUri1); // Add your image URIs here imageUris.add(imageUri2); Intent shareIntent = new Intent(); shareIntent.setAction(Intent.ACTION_SEND_MULTIPLE); shareIntent.putParcelableArrayListExtra(Intent.EXTRA_STREAM, imageUris); shareIntent.setType("image/*"); startActivity(Intent.createChooser(shareIntent, "Share images to.."));

下面我列出几个常用的MIME type类型

  • text/plain(纯文本)
  • text/html(HTML文档)
  • application/xhtml+xml(XHTML文档)
  • image/gif(GIF图像)
  • image/jpeg(JPEG图像)【PHP中为:image/pjpeg】
  • image/png(PNG图像)【PHP中为:image/x-png】
  • video/mpeg(MPEG动画)
  • application/octet-stream(任意的二进制数据)
  • application/pdf(PDF文档)
  • application/msword(Microsoft Word文件)
  • message/rfc822(RFC 822形式)
  • multipart/alternative(HTML邮件的HTML形式和纯文本形式,相同内容使用不同形式表示)
  • application/x-.example.sharedcontext; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.InputStream; import java.util.ArrayList; import android.app.Activity; import android.content.Context; import android.content.Intent; import android.net.Uri; import android.os.Bundle; import android.view.View; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.BaseAdapter; import android.widget.GridView; import android.widget.ImageView; import android.widget.TextView; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Intent intent = getIntent(); //获得Intent的Action String action = intent.getAction(); //获得Intent的MIME type String type = intent.getType(); if(Intent.ACTION_SEND.equals(action) && type != null){ //我们这里处理所有的文本类型 if(type.startsWith("text/")){ //处理获取到的文本,这里我们用TextView显示 handleSendText(intent); } //图片的MIME type有 image/png , image/jepg, image/gif 等, else if(type.startsWith("image/")){ //处理获取到图片,我们用ImageView显示 handleSendImage(intent); } } else if(Intent.ACTION_SEND_MULTIPLE.equals(action) && type != null){ if (type.startsWith("image/")) { //处理多张图片,我们用一个GridView来显示 handleSendMultipleImages(intent); } } } /** * 用TextView显示文本 * 可以打开一般的文本文件 * @param intent */ private void handleSendText(Intent intent){ TextView textView = new TextView(this); //一般的文本处理,我们直接显示字符串 String sharedText = intent.getStringExtra(Intent.EXTRA_TEXT); if(sharedText != null){ textView.setText(sharedText); } //文本文件处理,从Uri中获取输入流,然后将输入流转换成字符串 Uri textUri = (Uri) intent.getParcelableExtra(Intent.EXTRA_STREAM); if(textUri != null){ try { InputStream inputStream = this.getContentResolver().openInputStream(textUri); textView.setText(inputStream2Byte(inputStream)); } catch (Exception e) { e.printStackTrace(); } } //设置给Activity setContentView(textView); } /** * 将输入流转换成字符串 * @param inputStream * @return * @throws IOException */ private String inputStream2Byte(InputStream inputStream) throws IOException{ ByteArrayOutputStream bos = new ByteArrayOutputStream(); byte [] buffer = new byte[1024]; int len = -1; while((len = inputStream.read(buffer)) != -1){ bos.write(buffer, 0, len); } bos.close(); //指定编码格式为UIT-8 return new String(bos.toByteArray(), "UTF-8"); } /** * 用ImageView显示单张图片 * @param intent */ private void handleSendImage(Intent intent) { Uri imageUri = (Uri) intent.getParcelableExtra(Intent.EXTRA_STREAM); if (imageUri != null) { ImageView imageView = new ImageView(this); imageView.setImageURI(imageUri); setContentView(imageView); } } /** * 用GridView显示多张图片 * @param intent */ private void handleSendMultipleImages(Intent intent) { final ArrayList<Uri> imageUris = intent.getParcelableArrayListExtra(Intent.EXTRA_STREAM); if (imageUris != null) { GridView gridView = new GridView(this); //设置item的宽度 gridView.setColumnWidth(130); //设置列为自动适应 gridView.setNumColumns(GridView.AUTO_FIT); gridView.setAdapter(new GridAdapter(this, imageUris)); setContentView(gridView); gridView.setOnItemClickListener(new OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, final int position, long id) { //点击GridView的item 可以分享图片给其他应用 //这里可以参考http://blog.csdn.net/xiaanming/article/details/9395991 Intent intent = new Intent(); intent.setAction(Intent.ACTION_SEND); intent.putExtra(Intent.EXTRA_STREAM, imageUris.get(position)); intent.setType("image/*"); startActivity(Intent.createChooser(intent, "共享图片")); } }); } } /** * 重写BaseAdapter * @author xiaanming * */ public class GridAdapter extends BaseAdapter{ private Context mContext; private ArrayList<Uri> list; public GridAdapter(Context mContext, ArrayList<Uri> list){ this.list = list; this.mContext = mContext; } @Override public int getCount() { return list.size(); } @Override public Object getItem(int position) { return list.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { ImageView imageView; if(convertView == null){ imageView = new ImageView(mContext); imageView.setPadding(8, 8, 8, 8); }else{ imageView = (ImageView) convertView; } imageView.setImageURI(list.get(position)); return imageView; } } }

    运行程序之后,然后我们选择系统图库,选择多张图片(如图一)进行分享,我们自己的应用程序分享多张界面(如图二)点击我们应用程序的item,选择分享单张图片(如图三)我们继续选择我们自己的应用程序来显示(如图四),新建一个备忘录保存,长按备忘录进行分享(如图五),分享文本文件的显示界面(如图六)

声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。

相关文章