Я использую RoboGuice 2.0b4 для подключения моего приложения Android, в котором используется SyncAdapter (производный от AbstractThreadedSyncAdapter), размещенный в обычной службе:
@ContextSingleton
public class SyncAdapter extends AbstractThreadedSyncAdapter {
@Inject
private AccountManager accountManager;
@Inject
public SyncAdapter(Context context) {
super(context, true, false);
Injector injector = RoboGuice.getInjector(context);
ContextScope scope = injector.getInstance(ContextScope.class);
synchronized(ContextScope.class) {
scope.enter(context);
try {
injector.injectMembers(this);
} finally {
scope.exit(context);
}
}
}
[...]
}
@ContextSingleton
public class SynchronizationService extends RoboService {
@Inject
private SyncAdapter syncAdapter;
@Override
public IBinder onBind(Intent intent) {
if (equal("android.content.SyncAdapter", intent.getAction())) {
return syncAdapter.getSyncAdapterBinder();
} else [...]
}
return null;
}
}
К сожалению, RoboGuice выручает во время зависимостивнедрение, когда оно становится рекурсивным при вызове scope.enter
из конструктора SyncAdapter
:
java.lang.IllegalArgumentException: Scope for com.example.SynchronizationService@412c65b0 must be closed before scope for com.example.SynchronizationService@412c65b0 may be opened
, когда он видит контекст, присоединенный к текущему потоку с помощью вызова onCreate
SynchronizationService
.Трассировка стека:
at roboguice.inject.ContextScope.enter(ContextScope.java:67)
at roboguice.inject.ContextScopedRoboInjector.getInstance(ContextScopedRoboInjector.java:141)
at com.example.SyncAdapter.<init>(SyncAdapter.java:65)
at java.lang.reflect.Constructor.constructNative(Constructor.java:-1)
at java.lang.reflect.Constructor.newInstance(Constructor.java:417)
at com.google.inject.internal.DefaultConstructionProxyFactory$1.newInstance(DefaultConstructionProxyFactory.java:85)
at com.google.inject.internal.ConstructorInjector.construct(ConstructorInjector.java:85)
at com.google.inject.internal.ConstructorBindingImpl$Factory.get(ConstructorBindingImpl.java:254)
at com.google.inject.internal.ProviderToInternalFactoryAdapter$1.call(ProviderToInternalFactoryAdapter.java:46)
at com.google.inject.internal.InjectorImpl.callInContext(InjectorImpl.java:1031)
at com.google.inject.internal.ProviderToInternalFactoryAdapter.get(ProviderToInternalFactoryAdapter.java:40)
at roboguice.inject.ContextScope$1.get(ContextScope.java:126)
at com.google.inject.internal.InternalFactoryToProviderAdapter.get(InternalFactoryToProviderAdapter.java:40)
at com.google.inject.internal.SingleFieldInjector.inject(SingleFieldInjector.java:53)
at com.google.inject.internal.MembersInjectorImpl.injectMembers(MembersInjectorImpl.java:110)
at com.google.inject.internal.MembersInjectorImpl$1.call(MembersInjectorImpl.java:75)
at com.google.inject.internal.MembersInjectorImpl$1.call(MembersInjectorImpl.java:73)
at com.google.inject.internal.InjectorImpl.callInContext(InjectorImpl.java:1024)
at com.google.inject.internal.MembersInjectorImpl.injectAndNotify(MembersInjectorImpl.java:73)
at com.google.inject.internal.MembersInjectorImpl.injectMembers(MembersInjectorImpl.java:60)
at com.google.inject.internal.InjectorImpl.injectMembers(InjectorImpl.java:944)
at roboguice.inject.ContextScopedRoboInjector.injectMembersWithoutViews(ContextScopedRoboInjector.java:243)
at roboguice.inject.ContextScopedRoboInjector.injectMembers(ContextScopedRoboInjector.java:236)
at roboguice.service.RoboService.onCreate(RoboService.java:57)
at android.app.ActivityThread.handleCreateService(ActivityThread.java:2253)
[...]
at dalvik.system.NativeStart.main(NativeStart.java:-1)
Таким образом, очевидно, что внедрение зависимостей в RoboGuice не является реентерабельным.
К сожалению, SyncAdapter требует внедрения на основе конструктора, поскольку AbstractThreadedSyncAdapter
ожидает, что контекст будет предоставленконструктор.
Есть идеи?Я мог бы взломать RoboGuice, чтобы разрешить повторный вход, но это, вероятно, не легкое мероприятие, поэтому я бы предпочел менее инвазивный обходной путь.; -)