У меня есть небольшой Android-проект, который включает в себя несколько IPC, где клиентские действия связаны с моим сервисом.Я использую AIDL для IPC и RPC, который работает довольно хорошо, но у меня возникают проблемы с возвратом клиенту реализации интерфейса AIDL на стороне службы:
Когда клиент работает в том же процессе, что исервис - то есть запуск сервиса локально - все работает просто отлично.Но когда клиент и служба разделены в разных процессах, метод startLogSession, определенный в ILogDroidBinder.aidl, всегда возвращает ноль.Другой метод, реализованный в этом интерфейсе - getSessionIds - который возвращает List, содержащий целые числа, всегда работает (локально и перекрестно).
Я предполагаю, что моя реализация ILogDroidSession также должна реализоватьРазмещается, но это не сработает, потому что я не могу выполнить разбор объекта, связанного со ссылкой на базу данных SQLite (или я могу?).
Вот соответствующий код.Я был бы очень рад, если бы кто-нибудь мог помочь мне здесь.Может быть, я просто упускаю точку где-то, так как это мой первый проект Android, и я еще не совсем вовлечен.
ILogDroidSession.aidl (Реализация этого - то, что я хочу вернуть клиенту):
package net.sourceforge.projects.logdroid;
interface ILogDroidSession {
/**
* Logs the given text to the error message channel of the current logging
* session.
* @param text Text to log.
*/
void logError(in String text);
}
ILogDroidBinder.aidl (Интерфейс IBinder передан клиенту onServiceConnected):
package net.sourceforge.projects.logdroid;
import net.sourceforge.projects.logdroid.ILogDroidSession;
interface ILogDroidBinder {
/**
* Starts a new LogDroid session which handles all logging events.
* @param sessionName The name of the session.
* @return An instance of ILogDroidSession.
*/
ILogDroidSession startLogSession(in String sessionName);
/**
* Gets a list with all available LogSession ids.
*/
List getSessionIds();
}
LogDroidService.java (Соответствующий код из моей службы):
public class LogDroidService extends Service {
/**
* The binder interface needed for Activities to bind to the
* {@code LogDroidService}.
*/
private final ILogDroidBinder.Stub binder = new ILogDroidBinder.Stub() {
/**
* Starts a new LogDroidSession.
*/
public ILogDroidSession startLogSession(String sessionName) {
return LogDroidService.this.createSession(sessionName);
}
/**
* Gets all available session ids.
*/
public List<Integer> getSessionIds() {
return LogDroidService.this.getSessionIds();
}
};
/**
* The database connection to be used for storing and retrieving log entries.
*/
private LogDroidDb database;
@Override
public void onCreate() {
super.onCreate();
database = new LogDroidDb(getApplicationContext());
try {
database.open(); // opens as writable database
} catch ( SQLException ignorefornow ) {
}
}
@Override
public IBinder onBind(Intent ignore) {
return binder;
}
/**
* Creates a new LogDroidSession which will be returned to the user as a
* AIDL remote object.
* @param sessionName Name of the session.
* @return A new instance of ILogDroidSession
*/
ILogDroidSession createSession(String sessionName) {
LogDroidSession session = new LogDroidSession(database, sessionName);
session.addLoggingOccurredListener(this);
return session;
}
/**
* Retrieves all session ids.
* @return Array containing all LogDroidSession ids.
*/
ArrayList<Integer> getSessionIds() {
return database.getSessionIds();
}
}
MainActivity.java (соответствующий код клиента):
public class MainActivity extends Activity {
private ILogDroidSession session;
private ILogDroidBinder binder;
private ServiceConnection con = new ServiceConnection() {
public void onServiceConnected(ComponentName arg0, IBinder arg1) {
binder = ILogDroidBinder.Stub.asInterface(arg1); // always works
try {
// works locally but always returns null when cross-process
session = binder.startLogSession("TestSession");
// always works
List<Integer> ids = binder.getSessionIds();
} catch ( Exception ex) {
// no exceptions are thrown either when running locally or cross-process
Toast.makeText(getApplicationContext(), ex.getMessage(),
Toast.LENGTH_LONG).show();
}
}
public void onServiceDisconnected(ComponentName arg0) {
}
};
}