是否可以在Realm(Java,Android)中进行进程间通信。因此,如果我们从一个进程更新RealmObject的一个字段的值,我们将在另一个进程中获取RealmChangeListener(对应于提到的RealmObject)的触发器。我认为Realm的版本至少是2.0.2。
答案 0 :(得分:4)
Realm 2.0.0及更高版本支持非加密领域的进程间通知,但有are some edge cases that aren't covered yet (multi-process migrations, primarily).
我已经整理了一个示例项目,它对我有用。
在这里,RecyclerView显示从属于远程进程的广播接收器插入的狗。
10-11 20:53:46.340 7302-7302/com.zhuinden.realm_multiprocess_example I/MainActivity: Activity in Process [7302]
10-11 20:53:46.344 7302-7302/com.zhuinden.realm_multiprocess_example D/RealmManager: Incrementing Activity Count [0]: opening Realm.
10-11 20:53:46.352 7302-7302/com.zhuinden.realm_multiprocess_example D/RealmManager: Increment: Count [1]
10-11 20:54:01.352 7560-7560/com.zhuinden.realm_multiprocess_example:remote D/AlarmReceiver: Alarm received in process [7560]
10-11 20:54:01.352 7560-8705/com.zhuinden.realm_multiprocess_example:remote I/AlarmReceiver: Inserting DOGE [Juice]
完整的源代码:
public class RealmManager {
private static final String TAG = "RealmManager";
static Realm realm;
static RealmConfiguration realmConfiguration;
public static RealmConfiguration buildDefaultRealmConfiguration() {
return new RealmConfiguration.Builder()
.deleteRealmIfMigrationNeeded()
.build();
}
public static void initializeRealmConfig(Context appContext) {
if(realmConfiguration == null) {
Realm.init(appContext);
Log.d(TAG, "Initializing Realm configuration.");
setRealmConfiguration(buildDefaultRealmConfiguration());
}
}
public static void setRealmConfiguration(RealmConfiguration realmConfiguration) {
RealmManager.realmConfiguration = realmConfiguration;
Realm.setDefaultConfiguration(realmConfiguration);
}
private static int activityCount = 0;
public static Realm getRealm() {
return realm;
}
public static void incrementCount() {
if(activityCount == 0) {
if(realm != null) {
if(!realm.isClosed()) {
Log.w(TAG, "Unexpected open Realm found.");
realm.close();
}
}
Log.d(TAG, "Incrementing Activity Count [0]: opening Realm.");
realm = Realm.getDefaultInstance();
}
activityCount++;
Log.d(TAG, "Increment: Count [" + activityCount + "]");
}
public static void decrementCount() {
activityCount--;
Log.d(TAG, "Decrement: Count [" + activityCount + "]");
if(activityCount <= 0) {
Log.d(TAG, "Decrementing Activity Count: closing Realm.");
activityCount = 0;
realm.close();
if(Realm.compactRealm(realmConfiguration)) {
Log.d(TAG, "Realm compacted successfully.");
}
realm = null;
}
}
}
public class MainActivity
extends AppCompatActivity {
private static final String TAG = "MainActivity";
@BindView(R.id.recyclerview)
RecyclerView recyclerView;
Realm realm;
@Override
protected void onCreate(Bundle savedInstanceState) {
RealmManager.initializeRealmConfig(getApplicationContext());
Log.i(TAG, "Activity in Process [" + android.os.Process.myPid() + "]");
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
RealmManager.incrementCount(); //lazy for retained fragment
realm = RealmManager.getRealm();
recyclerView.setLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false));
recyclerView.setAdapter(new RealmRecyclerViewAdapter<Dog, DogViewHolder>(this, realm.where(Dog.class).findAllAsync(), true) {
@Override
public DogViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
TextView textView = new TextView(MainActivity.this);
RecyclerView.LayoutParams layoutParams = new RecyclerView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
ViewGroup.LayoutParams.WRAP_CONTENT);
textView.setLayoutParams(layoutParams);
return new DogViewHolder(textView);
}
@Override
public void onBindViewHolder(DogViewHolder holder, int position) {
holder.bind(getData().get(position));
}
});
AlarmManager alarmManager = (AlarmManager)getSystemService(ALARM_SERVICE);
Calendar calendar = Calendar.getInstance();
calendar.add(Calendar.SECOND, 15);
Intent intent = new Intent(this, AlarmReceiver.class);
PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
addNotification(alarmManager, pendingIntent, calendar.getTimeInMillis());
}
public void addNotification(AlarmManager alarmManager, PendingIntent pendingIntent, long timeOfNotification) {
if(Build.VERSION.SDK_INT < 19) {
alarmManager.set(AlarmManager.RTC_WAKEUP, timeOfNotification, pendingIntent);
} else if(Build.VERSION.SDK_INT >= 19 && Build.VERSION.SDK_INT < 23) {
alarmManager.setExact(AlarmManager.RTC_WAKEUP, timeOfNotification, pendingIntent);
} else if(Build.VERSION.SDK_INT >= 23) {
alarmManager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, timeOfNotification, pendingIntent);
}
}
@Override
protected void onDestroy() {
RealmManager.decrementCount();
super.onDestroy();
}
}
public class DogViewHolder extends RecyclerView.ViewHolder {
TextView textView;
public DogViewHolder(View itemView) {
super(itemView);
textView = (TextView) itemView;
}
public void bind(Dog dog) {
textView.setText(dog.getName());
}
}
public enum DogNames {
Jack,
Jill,
Joe,
Jarble,
Juice,
Jen,
Jim,
Munch,
Slurp,
Boop,
Largo,
Boson,
Pete,
Boner,
Derp,
Roger,
Bunbun,
Twix,
Dog
}
public class Dog extends RealmObject {
@PrimaryKey
private long id;
private String name;
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
public class AlarmReceiver extends BroadcastReceiver {
private static final String TAG = "AlarmReceiver";
RealmConfiguration realmConfiguration;
Executor executor;
@Override
@SuppressLint("NewApi")
public void onReceive(Context context, Intent intent) {
Log.d(TAG, "Alarm received in process [" + android.os.Process.myPid() + "]");
Realm.init(context);
if(realmConfiguration == null) {
realmConfiguration = RealmManager.buildDefaultRealmConfiguration();
}
if(executor == null) {
executor = Executors.newSingleThreadExecutor();
}
executor.execute(() -> {
try(Realm r = Realm.getInstance(realmConfiguration)) {
if(Looper.getMainLooper() == Looper.myLooper()) {
Log.i(TAG, "UI thread transaction: this should be fixed!");
}
r.executeTransaction(realm -> {
Dog dog = new Dog();
long id = realm.where(Dog.class).findAll().size()+1;
dog.setId(id);
dog.setName(DogNames.values()[((Long)id).intValue() % DogNames.values().length].name());
realm.insert(dog);
Log.i(TAG, "Inserting DOGE [" + dog.getName() + "]");
});
}
});
}
}
<?xml version="1.0" encoding="utf-8"?>
<manifest package="com.zhuinden.realm_multiprocess_example"
xmlns:android="http://schemas.android.com/apk/res/android">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
<receiver
android:name=".AlarmReceiver"
android:process=":remote"/>
</application>
</manifest>
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
android:id="@+id/activity_main"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.zhuinden.realm_multiprocess_example.MainActivity">
<android.support.v7.widget.RecyclerView
android:id="@+id/recyclerview"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</RelativeLayout>
// Top-level build file where you can add configuration options common to all sub-projects/modules.
buildscript {
repositories {
mavenCentral()
jcenter()
maven {url "https://clojars.org/repo/"}
maven { url "https://jitpack.io" }
}
dependencies {
classpath 'com.android.tools.build:gradle:2.2.0'
classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
classpath "io.realm:realm-gradle-plugin:2.0.2"
classpath 'me.tatarka:gradle-retrolambda:3.2.5'
// NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files
}
}
allprojects {
repositories {
mavenCentral()
jcenter()
maven {url "https://clojars.org/repo/"}
maven { url "https://jitpack.io" }
}
}
task clean(type: Delete) {
delete rootProject.buildDir
}
apply plugin: 'com.android.application'
apply plugin: 'com.neenbedankt.android-apt'
apply plugin: 'me.tatarka.retrolambda'
apply plugin: 'realm-android'
android {
compileSdkVersion 24
buildToolsVersion "24.0.2"
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
defaultConfig {
applicationId "com.zhuinden.realm_multiprocess_example"
minSdkVersion 14
targetSdkVersion 24
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
packagingOptions {
// Exclude file to avoid
// Error: Duplicate files during packaging of APK
exclude 'META-INF/DEPENDENCIES'
exclude 'META-INF/LICENSE'
exclude 'META-INF/LICENSE.txt'
exclude 'META-INF/license.txt'
exclude 'META-INF/NOTICE'
exclude 'META-INF/NOTICE.txt'
exclude 'META-INF/notice.txt'
exclude 'META-INF/ASL2.0'
exclude 'META-INF/services/javax.annotation.processing.Processor'
}
}
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
exclude group: 'com.android.support', module: 'support-annotations'
})
compile 'com.android.support:appcompat-v7:24.2.1'
testCompile 'junit:junit:4.12'
compile "com.android.support:recyclerview-v7:24.2.1"
apt 'com.jakewharton:butterknife-compiler:8.4.0'
compile 'com.jakewharton:butterknife:8.4.0'
apt 'dk.ilios:realmfieldnameshelper:1.1.0'
compile 'io.realm:android-adapters:1.3.0'
}