读书笔记Android访问远程数据的步骤(MessengerAIDLContentProvider
Posted lankton
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了读书笔记Android访问远程数据的步骤(MessengerAIDLContentProvider相关的知识,希望对你有一定的参考价值。
阅读书籍:《Android开发艺术探索》
作者:任玉刚
本文为阅读其中IPC相关章节所做的简单总结,相关示例代码来自于书中。
IPC相关
Messenger
1.远程service 创建Messenger对象: mMessenger,并通过onBind方法提供IBinder对象;
private static class MessengerHandler extends Handler
@Override
public void handleMessage(Message msg)
super.handleMessage(msg);
private final Messenger mMessenger = new Messenger(new MessengerHandler());
@Nullable
@Override
public IBinder onBind(Intent intent)
return mMessenger.getBinder();
2.client 绑定远程 service,通过ServiceConnection对象获得IBinder对象, 再通过该IBinder对象获得Messenger对象1:cm1;
private ServiceConnection mConnection = new ServiceConnection()
@Override
public void onServiceConnected(ComponentName name, IBinder service)
mService = new Messenger(service);
Message msg = Message.obtain();
msg.what = 99;
Bundle data = new Bundle();
data.putString("msg", "hello, this is client");
msg.setData(data);
msg.replyTo = mGetReplyMessenger;
try
mService.send(msg);
catch (RemoteException e)
e.printStackTrace();
@Override
public void onServiceDisconnected(ComponentName name)
;
3.同时要用Handler创建一个新的Messenger对象,cm2,用来处理远程service返回的消息。用msg.replyTo = cm2 进行设置。之后通过cm1.send(msg)向远程service发送消息。远程service中的sm1接受消息并且处理。
3.sm1中Handler对象处理消息时, 可以通过msg.replyTo获取要回复的Messenger对象。使用msg.replyTo.send(msg), 即可在客户端Messenger对象cm2 的handler中进行消息的处理。
AIDL
- 创建 Bean, 实现Parcelable接口;
- 创建 Manager AIDL文件, 如果用到上述 Bean, 需要建立 Bean.AIDL, 并申明那个类为parcelable
package cn.lankton.aidl;
parcelable Book;
Manager 的 aidl文件如下:
// IBookManager.aidl
package cn.lankton.aidl;
import cn.lankton.aidl.Book;
// Declare any non-default types here with import statements
interface IBookManager
List<Book> getBookList();
void addBook(in Book book);
提供了供client调用的方法
之后编译运行一次, IDE自动生成Manager对应的同名java文件(根据此例为 IBookManager.java)。Eclipse位于gen目录下,android Studio位于build/generated目录下
3. 创建远程service。内部创建一个继承IBookManager.Stub的Binder对象, 实现其中的方法, 并通过onBind方法, 返回该Binder对象。
public class BookManagerService extends Service
private CopyOnWriteArrayList<Book> mBookList = new CopyOnWriteArrayList<>();
private Binder mBinder = new IBookManager.Stub()
@Override
public List<Book> getBookList() throws RemoteException
return mBookList;
@Override
public void addBook(Book book) throws RemoteException
mBookList.add(book);
;
@Nullable
@Override
public IBinder onBind(Intent intent)
return mBinder;
@Override
public void onCreate()
super.onCreate();
mBookList.add(new Book(1, "Android"));
mBookList.add(new Book(2, "ios"));
- client 创建ServiceConnection, 并进行bindService,通过onServiceConnected获得IBinder对象,并通过asInterface方法获得远程service中的IBookManager对象,即可对远程service的数据进行操作。
public class MainActivity extends AppCompatActivity
private ServiceConnection mConnection = new ServiceConnection()
@Override
public void onServiceConnected(ComponentName name, IBinder service)
IBookManager bookManager = IBookManager.Stub.asInterface(service);
try
List<Book> list = bookManager.getBookList();
Log.i("AIDLDemo", "query book list, list type: " + list.getClass().getCanonicalName());
Log.i("AIDLDemo", "query book list: " + list.toString());
Book book = new Book(3, "swift");
bookManager.addBook(book);
List<Book> newList = bookManager.getBookList();
Log.i("AIDLDemo", "after add book, list: " + newList.toString());
catch (RemoteException e)
e.printStackTrace();
@Override
public void onServiceDisconnected(ComponentName name)
;
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent(this, BookManagerService.class);
bindService(intent, mConnection, BIND_AUTO_CREATE);
ContentProvider
一个提供跨进程访问数据的解决方案。
1. 建立远端ContentProvider, 并在manifest中申明authority和name;
其中, authorities最为重要, 其是本地访问远端ContentProvider的唯一表识。
<provider
android:authorities="cn.lankton.contentproviderdemo.book.provider"
android:name=".BookProvider"
android:process=":provider"/>
2. 实现ContentProvider中的各类方法:oncreate、 insert、 delete、 update、 query、 getType等, 暴露对数据增删改查的能力;
3. 在ContentProvider中, 可以通过UriMatcher, 将Uri子路径和int型绑定, 方便处理不同子路径的Uri。
private static final UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
static
sUriMatcher.addURI(AUTHORITY, "book", BOOK_URI_CODE);
sUriMatcher.addURI(AUTHORITY, "user", USER_URI_CODE);
private String getTableName(Uri uri)
String tableName = null;
switch (sUriMatcher.match(uri))
case BOOK_URI_CODE:
tableName = DBOpenHelper.BOOK_TABLE_NAME;
break;
case USER_URI_CODE:
tableName = DBOpenHelper.USER_TABLE_NAME;
break;
default:
break;
return tableName;
接下来, 在本地访问远程数据
1. 通过远程ContentProvider的authorities拼接Uri;
2. 通过getContentResolver获得ContentResolver对象
3. 通过该ContentResolover对象和上面的Uri对象,对远程的数据增删改查。(实际可以理解为调用远程ContentProvider的各种相应方法
Uri bookUri = Uri.parse("content://cn.lankton.contentproviderdemo.book.provider/book");
ContentValues values = new ContentValues();
values.put("_id", 6);
values.put("name", "程序的现代艺术");
getContentResolver().insert(bookUri, values);
Cursor bookCursor = getContentResolver().query(bookUri, new String[]"_id", "name", null, null, null);
while (bookCursor.moveToNext())
Book book = new Book();
book.bookId = bookCursor.getInt(bookCursor.getColumnIndex("_id"));
book.bookName = bookCursor.getString(bookCursor.getColumnIndex("name"));
Log.d(TAG, "query book:" + book.bookId + "," + book.bookName);
bookCursor.close();
需要注意:
ContentProvider的query, update, insert, delete四大方法存在多线程并发访问的。用同一个SQLiteDataBase对象可以实现同步, 但如果远程数据是List等情况,必须自己想办法保证同步。
以上是关于读书笔记Android访问远程数据的步骤(MessengerAIDLContentProvider的主要内容,如果未能解决你的问题,请参考以下文章
Android深度探索(卷1)HAL与驱动开发--读书笔记(第六章)