转载于: http://www. androidsdn.com/article/show/137
由于android系统中应用程序之间不能共享内存。因此,在不同应用程序之间交互数据(跨进程通讯)就稍微麻烦一些。在android SDK中提供了4种用于跨进程通讯的方式。这4种方式正好对应于android系统中4种应用程序组件:Activity、Content Provider、Broadcast和Service。其中Activity可以跨进程调用其他应用程序的Activity;Content Provider可以跨进程访问其他应用程序中的数据(以Cursor对象形式返回),当然,也可以对其他应用程序的数据进行增、删、改操 作;Broadcast可以向android系统中所有应用程序发送广播,而需要跨进程通讯的应用程序可以监听这些广播;Service和Content Provider类似,也可以访问其他应用程序中的数据,但不同的是,Content Provider返回的是Cursor对象,而Service返回的是Java对象,这种可以跨进程通讯的服务叫AIDL服务。
完整示例请参阅本文提供的源代码。
方式一:访问其他应用程序的Activity
Activity既可以在进程内(同一个应用程序)访问,也可以跨进程访问。如果想在同一个应用程序中访问Activity,需要指定Context对象和Activity的Class对象,代码如下:
Intent intent = new Intent(this , Test.class );
startActivity(intent);
[java] view plaincopy
Intent intent = new Intent(this, Test.class);
startActivity(intent);
Activity的跨进程访问与进程内访问略有不同。虽然它们都需要Intent对象,但跨进程访问并不需要指定Context对象和Activity的Class对象,而需要指定的是要访问的Activity所对应的Action(一个字符串)。有些Activity还需要指定一个Uri(通过Intent构造方法的第2个参数指定)。
在android系统中有很多应用程序提供了可以跨进程访问的Activity,例如,下面的代码可以直接调用拨打电话的Activity。
Intent callIntent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:12345678" );
startActivity(callIntent);
[java] view plaincopy
Intent callIntent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:12345678");
startActivity(callIntent);
执行上面的代码后,系统会自动拨号,界面如图1所示。
在调用拨号程序的代码中使用了一个Intent.ACTION_CALL常量,该常量的定义如下:
public static final String ACTION_CALL = "android.intent.action.CALL" ;
[java] view plaincopy
public static final String ACTION_CALL = "android.intent.action.CALL";
这个常量是一个字符串常量,也是我们在这节要介绍的跨进程调用Activity的关键。如果在应用程序中要共享某个Activity,需要为这个Activity指定一个字符串ID,也就是Action。也可以将这个Action看做这个Activity的key。在其他的应用程序中只要通过这个Action就可以找到与Action对应的Activity,并通过startActivity方法来启动这个Activity。
下面先来看一下如何将应用程序的Activity共享出来,读者可按如下几步来共享Activity:
1. 在AndroidManifest.xml文件中指定Action。指定Action要使用
2. 在AndroidManifest.xml文件中指定访问协议。在指定Uri(Intent类的第2个参数)时需要访问协议。访问协议需要使 用标签的android:scheme属性来指定。如果该属性的值是“abc”,那么Uri就应该是“abc://Uri的主体 部分”,也就是说,访问协议是Uri的开头部分。
3. 通过getIntent().getData().getHost()方法获得协议后的Uri的主体部分。这个Host只是个称谓,并不一定是主机名。读者可以将其看成是任意的字符串。
4. 从Bundle对象中获得其他应用程序传递过来的数据。
5. 这一步当然是获得数据后做进一步的处理了。至于如何处理这些数据,就得根据具体的需求决定了。
下面来根据这些步骤共享一个Activity。首先建立一个android工程(ActionActivity),工程的主Activity是Main。在 本例中我们会共享这个Main类。首先打开AndroidManifest.xml文件,添加一个
[java] view plaincopy
在配置AndroidManifest.xml时要注意,不能在同一个
从上面的代码可以看出,
info://任意字符串
[java] view plaincopy
info://任意字符串
一般
下面来看看如何在Main类的onCreate方法中获得其他应用程序传递过来的数据。
package net.blogjava.mobile.actionactivity;
... ...
public class Main extends Activity implements OnClickListener
{
private EditText editText;
@Override
public void onClick(View view)
{
// 单击按钮,会显示文本框中的内容(以Toast信息框形式显示)
Toast.makeText(this , editText.getText().toString(), Toast.LENGTH_LONG)
.show();
}
@Override
public void onCreate(Bundle savedInstanceState)
{
super .onCreate(savedInstanceState);
setContentView(R.layout.main);
Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(this );
editText = (EditText) findViewById(R.id.edittext);
// 获得其他应用程序传递过来的数据
if (getIntent().getData() != null )
{
// 获得Host,也就是info://后面的内容
String host = getIntent().getData().getHost();
Bundle bundle = getIntent().getExtras();
// 其他的应用程序会传递过来一个value值,在该应用程序中需要获得这个值
String value = bundle.getString("value" );
// 将Host和Value组合在一下显示在EditText组件中
editText.setText(host + ":" + value);
// 调用了按钮的单击事件,显示Toast信息提示框
onClick(button);
}
}
}
[java] view plaincopy
package net.blogjava.mobile.actionactivity;
... ...
public class Main extends Activity implements OnClickListener
{
private EditText editText;
@Override
public void onClick(View view)
{
// 单击按钮,会显示文本框中的内容(以Toast信息框形式显示)
Toast.makeText(this, editText.getText().toString(), Toast.LENGTH_LONG)
.show();
}
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(this);
editText = (EditText) findViewById(R.id.edittext);
// 获得其他应用程序传递过来的数据
if (getIntent().getData() != null)
{
// 获得Host,也就是info://后面的内容
String host = getIntent().getData().getHost();
Bundle bundle = getIntent().getExtras();
// 其他的应用程序会传递过来一个value值,在该应用程序中需要获得这个值
String value = bundle.getString("value");
// 将Host和Value组合在一下显示在EditText组件中
editText.setText(host + ":" + value);
// 调用了按钮的单击事件,显示Toast信息提示框
onClick(button);
}
}
}
从上面的程序可以看出,首先通过getIntent().getData()来判断其他的应用程序是否传递了Uri(getData方法返回了一个Uri 对象)。如果运行该程序,Uri为null,因此,不会执行if语句里面的代码。当其他的应用程序传递了Uri对象后,系统会执行if语句里面的代码。当 运行ActionActivity后,在文本框中输入“Running”,单击“显示文本框的内容”按钮,会显示如图2所示的Toast提示信息框。
下面来看一下其他的应用程序是如何调用ActionActivity中的Main。新建一个android工程(InvokeActivity),并添加一个按钮,按钮的单击事件方法代码如下:
public void onClick(View view)
{
// 需要使用Intent类的第2个参数指定Uri
Intent intent = new Intent("net.blogjava.mobile.MYACTION" , Uri
.parse("info://调用其他应用程序的Activity" ));
// 设置value属性值
intent.putExtra("value" , "调用成功" );
// 调用ActionActivity中的Main
startActivity(intent);
}
[java] view plaincopy
public void onClick(View view)
{
// 需要使用Intent类的第2个参数指定Uri
Intent intent = new Intent("net.blogjava.mobile.MYACTION", Uri
.parse("info://调用其他应用程序的Activity"));
// 设置value属性值
intent.putExtra("value", "调用成功");
// 调用ActionActivity中的Main
startActivity(intent);
}
在运行InvokeActivity之前,先要运行ActionActivity以便在android模拟器中安装该程序。然后单击InvokeActivity中的按钮,就会显示如图3所示的效果。
当然,也可以使用startActivityForResult方法来启动其他应用程序的Activity,以便获得Activity的返回值。例如,可以将ActionActivity中Main类的onClick代码修改为下面的形式。
public void onClick(View view)
{
Toast.makeText(this , editText.getText().toString(), Toast.LENGTH_LONG).show();
Intent intent = new Intent();
// 设置要返回的属性值
intent.putExtra("result" , editText.getText().toString());
// 设置返回码和Intent对象
setResult(2 , intent);
// 关闭Activity
finish();
}
[java] view plaincopy
public void onClick(View view)
{
Toast.makeText(this, editText.getText().toString(), Toast.LENGTH_LONG).show();
Intent intent = new Intent();
// 设置要返回的属性值
intent.putExtra("result", editText.getText().toString());
// 设置返回码和Intent对象
setResult(2, intent);
// 关闭Activity
finish();
}
然后在InvokeActivity中使用下面的代码来调用Main。
intent = new Intent("net.blogjava.mobile.MYACTION" , Uri
.parse("info://调用其他应用程序的Activity" ));
// 传递数据
intent.putExtra("value" , "调用成功" );
startActivityForResult(intent, 1 ); // 1为请求码
[java] view plaincopy
intent = new Intent("net.blogjava.mobile.MYACTION", Uri
.parse("info://调用其他应用程序的Activity"));
// 传递数据
intent.putExtra("value", "调用成功");
startActivityForResult(intent, 1); // 1为请求码
要想接收Activity返回的值,需要覆盖onActivityResult事件方法,代码如下:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
Toast.makeText(this , "返回值:" + data.getExtras().getString("result" ),
Toast.LENGTH_LONG).show();
}
[java] view plaincopy
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
Toast.makeText(this, "返回值:" + data.getExtras().getString("result"),
Toast.LENGTH_LONG).show();
}
当单击InvokeActivity中的相应按钮后,并且Main关闭后,会显示如图4所示的Toast信息提示框。
从本节的介绍可以看出,跨进程访问Activity(访问其他应用程序中的Activity)主要是通过一个Action来完成的,如果要传递数据,还需 要指定一个Uri。当然,传递数据也可以通过Intent来完成。传递数据的过程可以是双向的。如果要想从调用的Activity中返回数据,就需要使用startActivityForResult方法来启动Activity了。
方式二:Content Provider
Android应用程序可以使用文件或SqlLite数据库来存储数据。Content Provider提供了一种在多个应用程序之间数据共享的方式(跨进程共享数据)。应用程序可以利用Content Provider完成下面的工作
1. 查询数据
2. 修改数据
3. 添加数据
4. 删除数据
虽然Content Provider也可以在同一个应用程序中被访问,但这么做并没有什么意义。Content Provider存在的目的向其他应用程序共享数据和允许其他应用程序对数据进行增、删、改操作。
Android系统本身提供了很多Content Provider,例如,音频、视频、联系人信息等等。我们可以通过这些Content Provider获得相关信息的列表。这些列表数据将以Cursor对象返回。因此,从Content Provider返回的数据是二维表的形式。
对于访问Content Provider的程序,需要使用ContentResolver对象。该对象需要使用getContentResolver方法获得,代码如下:
ContentResolver cr = getContentResolver();
[java] view plaincopy
ContentResolver cr = getContentResolver();
与Activity一样,Content Provider也需要与一个URI对应。每一个Content Provider可以控制多个数据集,在这种情况下,每一个数据集会对应一个单独的URI。所有的URI必须以“content://”开头。
为了程序更容易维护,也为了简化程序代码,一般将URI定义成一个常量。例如,下面的常量表示系统的联系人电话号码。
android.provider.Contacts.Phones.CONTENT_URI
[java] view plaincopy
android.provider.Contacts.Phones.CONTENT_URI
下面来看一下编写Content Provider的具体步骤。
1. 编写一个继承于android.content.ContentProvider的子类。该类是ContentProvider的核心类。在该类中会实现query、insert、update及delete方法。实际上调用ContentResolver类的这4个方法就是调用ContentProvider类中与之要对应的方法。在本文中只介绍query。至于insert、update、delete和query的用法类 似。也是通过Uri传递参数,然后在这些方法中接收这些参数,并做进一步地处理。
2. 在AndroidManifest.xml文件中配置ContentProvider。要想唯一确定一个ContentProvider,需要指定这个ContentProvider的URI,除此之外,还需要指定URI所对应的ContentProvider类。这有些象Servlet的定义,除了要 指定Servlet对应的Web地址,还要指定这个地址所对应的Servlet类。
现在来看一下Uri的具体格式,先看一下如图5所示的URI。
下面对图5所示的URI的4个部分做一下解释。
A:Content Provider URI的固定前缀,也就是说,所有的URI必须以content://开头。
B:URI中最重要的部分。该部分是Content Provider的唯一标识。对于第三方应用程序来说,该部分最后使用完整的类名(包名+类名),以确保URI的唯一性。该部分需要在AndroidManifest.xml文件中
[java] view plaincopy
C:这部分是URI的路径(path)。表示URI中各种被请求的数据。这部分是可选的, 如果Content Provider仅仅提供一种请求的数据,那么这部分可以省略。如果Content Provider要提供多种请求数据。就需要添加多个路径,甚至是子路径。例如,“land/bus”、“land/train”、“sea/ship” 就指定了3种可能提供的数据。
D:这部分也是可选的。如果要传递一个值给Content Provider,可以通过这部分传递。当然,如果不需要传值,这部分也可以省略,省略后的URI如下所示:
content://com.example.transportationprovider/trains
[java] view plaincopy
content://com.example.transportationprovider/trains
本例利用了《基于android SDK1.5的英文电子词典的实现》一文中实现的电子词典程序。通过ContentProvider,将电子词典的查词功能共享成Cursor对象。这样 其他的应用程序就可以通过ContentProvider来查词英文单词了。关于英文词典的具体实现细节,读者可以通过如下的地址查看《基于android SDK1.5的英文电子词典的实现》一文。
http://www. androidsdn.com/article/show/111
[java] view plaincopy
http://www.ophonesdn.com/article/show/111
在电子词典程序中需要一个DictionaryContentProvider类,该类是ContentProvider的子类。在该类中实现了query方法,并根据不同的URI来返回不同的结果。让我们先看一下DictionaryContentProvider类,然后再对这些代码做一些解 释。
... ...
public class DictionaryContentProvider extends ContentProvider
{
private static UriMatcher uriMatcher;
private static final String AUTHORITY = "net.blogjava.mobile.dictionarycontentprovider" ;
private static final int SINGLE_WORD = 1 ;
private static final int PREFIX_WORDS = 2 ;
public static final String DATABASE_PATH = android.os.Environment
.getExternalStorageDirectory().getAbsolutePath()
+ "/dictionary" ;
public static final String DATABASE_FILENAME = "dictionary.db" ;
private SQLiteDatabase database;
static
{
// 添加访问ContentProvider的Uri
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
uriMatcher.addURI(AUTHORITY, "single" , SINGLE_WORD);
uriMatcher.addURI(AUTHORITY, "prefix/*" , PREFIX_WORDS);
}
// 该方法在Activity的onCreate方法之前调用
@Override
public boolean onCreate()
{
database = openDatabase();
return true ;
}
// 在本例中只实现了query方法,其他的方法(insert、update和delete)与query方法的实现
// 类似
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder)
{
Cursor cursor = null ;
switch (uriMatcher.match(uri))
{
case SINGLE_WORD:
// 查找指定的单词
cursor = database.query("t_words" , projection, selection,
selectionArgs, null , null , sortOrder);
break ;
case PREFIX_WORDS:
String word = uri.getPathSegments().get(1 );
// 查找以指定字符串开头的单词集合
cursor = database
.rawQuery(
"select english as _id, chinese from t_words where english like ?" ,
new String[]
{ word + "%" });
break ;
default :
throw new IllegalArgumentException("<" + uri + ">格式不正确." );
}
return cursor;
}
... ...
}
[java] view plaincopy
... ...
public class DictionaryContentProvider extends ContentProvider
{
private static UriMatcher uriMatcher;
private static final String AUTHORITY = "net.blogjava.mobile.dictionarycontentprovider";
private static final int SINGLE_WORD = 1;
private static final int PREFIX_WORDS = 2;
public static final String DATABASE_PATH = android.os.Environment
.getExternalStorageDirectory().getAbsolutePath()
+ "/dictionary";
public static final String DATABASE_FILENAME = "dictionary.db";
private SQLiteDatabase database;
static
{
// 添加访问ContentProvider的Uri
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
uriMatcher.addURI(AUTHORITY, "single", SINGLE_WORD);
uriMatcher.addURI(AUTHORITY, "prefix/*", PREFIX_WORDS);
}
// 该方法在Activity的onCreate方法之前调用
@Override
public boolean onCreate()
{
database = openDatabase();
return true;
}
// 在本例中只实现了query方法,其他的方法(insert、update和delete)与query方法的实现
// 类似
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder)
{
Cursor cursor = null;
switch (uriMatcher.match(uri))
{
case SINGLE_WORD:
// 查找指定的单词
cursor = database.query("t_words", projection, selection,
selectionArgs, null, null, sortOrder);
break;
case PREFIX_WORDS:
String word = uri.getPathSegments().get(1);
// 查找以指定字符串开头的单词集合
cursor = database
.rawQuery(
"select english as _id, chinese from t_words where english like ?",
new String[]
{ word + "%" });
break;
default:
throw new IllegalArgumentException("<" + uri + ">格式不正确.");
}
return cursor;
}
... ...
}
关于DictionaryContentProvider类的代码需要做如下的解释。
1. 在DictionaryContentProvider类的开头定义的AUTHORITY是访问ContentProvider的URI的前半部分。
2. 访问ContentProvider的URI的后半部分由uriMatcher.addURI(...)方法指定。该方法的第1个参数就是AUTHORITY(Uri的前半部分),第2个参数是Uri的后半部分,第3个参数是与第2个参数值对应的代码。当其他的应用程序通过Uri访问ContentProvider时。系统解析Uri后,将addURI方法的第2个参数值转换成与之对应的代码(第3个参数值)。
3. addURI的第2个参数值可以使用通配符。例如,prefix/*中的*表示所有字符。prefix/abc、prefix/xxx都会匹配成功。
4. 访问ContentProvider的URI是addURI的第1个和第2个参数值的组件,例如,按着DictionaryContentProvider中设置的两个URI,可以分别匹配下面的两个URI。
content://net.blogjava.mobile.dictionarycontentprovider/single
content://net.blogjava.mobile.dictionarycontentprovider/prefix/wo
[java] view plaincopy
content://net.blogjava.mobile.dictionarycontentprovider/single
content://net.blogjava.mobile.dictionarycontentprovider/prefix/wo
要注意的是,访问ContentProvider的URI必须以“content://”开头。
5. 在query方法中建议使用SQLiteDatabase对象的query方法查询。因为query方法的参数正好和DictionaryContentProvider类中的query方法的参数对应,这样使用起来比较方便。
6. 由于安装了ContentProvider的应用程序会先调用ContentProvider的onCreate方法(该方法会在Activity的onCreate方法之前调用),因此,只需要将打开或复制数据库的方法(openDatabase)放在DictionaryContentProvider类中,并在onCreate方法中调用即可。
7. 在DictionaryContentProvider类中只实现了query方法。在该方法中判断了其他应用程序发送的是哪一个Uri。并进行相应的处理。这两个Uri一个是查询指定单词的,另外一个是查询以某个字符串开头的所有单词的(用于显示单词列表)。
下面在AndroidManifest.xml文件中配置DictionaryContentProvider类。
[java] view plaincopy
OK,现在来看看应用程序如何调用ContentProvider。调用ContentProvider的关键是使用getContentResolver方法来获得一个ContentResolver对象,并通过ContentResolver对象的query方法来 访问ContentProvider。
首先来定义两个访问ContentProvider的常量。
public final String DICTIONARY_SINGLE_WORD_URI = "content://net.blogjava.mobile.dictionarycontentprovider/single" ;
public final String DICTIONARY_PREFIX_WORD_URI = "content://net.blogjava.mobile.dictionarycontentprovider/prefix" ;
[java] view plaincopy
public final String DICTIONARY_SINGLE_WORD_URI = "content://net.blogjava.mobile.dictionarycontentprovider/single";
public final String DICTIONARY_PREFIX_WORD_URI = "content://net.blogjava.mobile.dictionarycontentprovider/prefix";
然后在查询按钮的单击事件中编写如下的代码来查询单词。
public void onClick(View view)
{
Uri uri = Uri.parse(DICTIONARY_SINGLE_WORD_URI);
// 通过ContentProvider查询单词,并返回Cursor对象,然后的操作就和直接从数据中获得
// Cursor对象后的操作是一样的了
Cursor cursor = getContentResolver().query(uri, null , "english=?" ,
new String[]{ actvWord.getText().toString() }, null );
String result = "未找到该单词." ;
if (cursor.getCount() > 0 )
{
cursor.moveToFirst();
result = cursor.getString(cursor.getColumnIndex("chinese" ));
}
new AlertDialog.Builder(this ).setTitle("查询结果" ).setMessage(result)
.setPositiveButton("关闭" , null ).show();
}
[java] view plaincopy
public void onClick(View view)
{
Uri uri = Uri.parse(DICTIONARY_SINGLE_WORD_URI);
// 通过ContentProvider查询单词,并返回Cursor对象,然后的操作就和直接从数据中获得
// Cursor对象后的操作是一样的了
Cursor cursor = getContentResolver().query(uri, null, "english=?",
new String[]{ actvWord.getText().toString() }, null);
String result = "未找到该单词.";
if (cursor.getCount() > 0)
{
cursor.moveToFirst();
result = cursor.getString(cursor.getColumnIndex("chinese"));
}
new AlertDialog.Builder(this).setTitle("查询结果").setMessage(result)
.setPositiveButton("关闭", null).show();
}
下面是显示单词列表的代码。
public void afterTextChanged(Editable s)
{
if ("" .equals(s.toString()))
return ;
Uri uri = Uri.parse(DICTIONARY_PREFIX_WORD_URI + "/" + s.toString());
// 从ContentProvider中获得以某个字符串开头的所有单词的Cursor对象
Cursor cursor = getContentResolver().query(uri, null , null , null , null );
DictionaryAdapter dictionaryAdapter = new DictionaryAdapter(this ,
cursor, true );
actvWord.setAdapter(dictionaryAdapter);
}
[java] view plaincopy
public void afterTextChanged(Editable s)
{
if ("".equals(s.toString()))
return;
Uri uri = Uri.parse(DICTIONARY_PREFIX_WORD_URI + "/" + s.toString());
// 从ContentProvider中获得以某个字符串开头的所有单词的Cursor对象
Cursor cursor = getContentResolver().query(uri, null, null, null, null);
DictionaryAdapter dictionaryAdapter = new DictionaryAdapter(this,
cursor, true);
actvWord.setAdapter(dictionaryAdapter);
}
现在来运行本例,会看到如图6所示的界面。当查询单词时会显示如图7所示的单词列表,查询出结果后,会显示如图8所示的界面。