检测您是在应用程序中的主进程还是远程服务进程中



我有一个应用程序,它有一个远程服务运行在一个单独的进程中:

<service android:name=".MyService" android:process=":remote"/>

我还使用了一个Application类:

<application android:label="@string/app_name" android:name=".MyApplication" ...

我可以这样做吗?

public class MyApplication extends Application {
    public MyApplication() {
        if (isRemoteService()) {
            setupLog("remoteservice.log");
        } else {
            setupLog("application.log");
        }
    }

我想我可以得到进程名并使用它来检测我是否在远程服务或主应用程序中,但我还没有发现如何获得进程名。我可以从android.os.Process.myPID()获得PID,但这对我没有多大帮助。

例如,如果您想要检查您是否在主进程中,您可以在应用程序中编写这样的代码:

public class MyApplication extends Application {
    @Override
    public void onCreate() {
        //...code here will be execute in every process...
        if (isMainProcess()) {
            //...code here will be execute only in main process
        }
        super.onCreate();
    }
    // your package name is the same with your main process name
    private boolean isMainProcess() {
        return getPackageName().equals(getProcessName());
    }
    // you can use this method to get current process name, you will get
    // name like "com.package.name"(main process name) or "com.package.name:remote"
    private String getProcessName() {
        int mypid = android.os.Process.myPid();
        ActivityManager manager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
        List<RunningAppProcessInfo> infos = manager.getRunningAppProcesses();
        for(RunningAppProcessInfo info : infos) {
            if (info.pid == mypid) {
                return info.processName;
            }
        }
        // may never return null
        return null;
    }
}

我可以提供一个间接的解决方案:

在每个StartUp方法中设置System Property:

System.setProperty("PROCESS_TYPE","SERVICE");
System.setProperty("PROCESS_TYPE","RECEIVER");
System.setProperty("PROCESS_TYPE","ACTIVITY");

属性是静态的,与流程隔离,并且可以从任何地方访问。附加的好处是它们可以直接被日志框架使用,比如Logback。

我有一个类似的问题,这就是我所做的。

MyServiceMyActivity有一个共同的部分,MyEngine,但在这两种情况下,行为必须有所不同。

不同的是设置,但是这个设置是在MyServiceMyActivity类中完成的。

活动和服务的另一个不同之处是通过侦听器完成的:MyEngine定义接口MyEngine.Listener,而MyServiceMyActivity为引擎提供该接口的不同实现。

所以,如果你想传递一个布尔值,有两种可能的方法:

// Method 1: different initialization
class MyEngine {
    MyEngine(boolean isService) { ... }
}
class MyActivity extends Activity {
    private MyEngine = new MyEngine(false);
    ...
}
class MyService extends Service {
    private MyEngine = new MyEngine(true);
    ...
}
// Method 2: callbacks
class MyEngine {
    interface Listener {
        boolean isService();
    }
    private Listener mListener;
    MyEngine(Listener listener) { mListener = listener; }
}
class MyActivity extends Activity {
    private mListener = new MyEngine.Listener() {
        boolean isService() { return false; }
    }
    private MyEngine = new MyEngine(mListener);
    ...
}
class MyService extends Service {
    private mListener = new MyEngine.Listener() {
        boolean isService() { return true; }
    }
    private MyEngine = new MyEngine(mListener);
    ...
}

  1. 上面示例中使用的布尔值在现实世界中是无用的:如果您想使用不同的日志文件名,最好传递文件名而不是布尔值。如果要执行两个不同的操作,最好使用一个侦听器函数和两个实现。

  2. 当然,可以通过Context并检查它是否是ActivityService的子进程,或者获得当前进程的名称,但这些都是android特定的实现细节,除非绝对必要,否则最好不要依赖它们。

我使用的是一段代码,取自Google的WorkManager库。在写这篇文章时,它位于GreedyScheduler.java。这样定义方法getProcessName():

    @Nullable
    private String getProcessName() {
        if (SDK_INT >= 28) {
            return Application.getProcessName();
        }
        // Try using ActivityThread to determine the current process name.
        try {
            Class<?> activityThread = Class.forName(
                    "android.app.ActivityThread",
                    false,
                    GreedyScheduler.class.getClassLoader());
            final Object packageName;
            if (SDK_INT >= 18) {
                Method currentProcessName = activityThread.getDeclaredMethod("currentProcessName");
                currentProcessName.setAccessible(true);
                packageName = currentProcessName.invoke(null);
            } else {
                Method getActivityThread = activityThread.getDeclaredMethod(
                        "currentActivityThread");
                getActivityThread.setAccessible(true);
                Method getProcessName = activityThread.getDeclaredMethod("getProcessName");
                getProcessName.setAccessible(true);
                packageName = getProcessName.invoke(getActivityThread.invoke(null));
            }
            if (packageName instanceof String) {
                return (String) packageName;
            }
        } catch (Throwable exception) {
            Log.d("TAG", "Unable to check ActivityThread for processName", exception);
        }
        // Fallback to the most expensive way
        int pid = Process.myPid();
        ActivityManager am =
                (ActivityManager) mContext.getSystemService(ACTIVITY_SERVICE);
        if (am != null) {
            List<ActivityManager.RunningAppProcessInfo> processes = am.getRunningAppProcesses();
            if (processes != null && !processes.isEmpty()) {
                for (ActivityManager.RunningAppProcessInfo process : processes) {
                    if (process.pid == pid) {
                        return process.processName;
                    }
                }
            }
        }
        return null;
    }

然后在调用方:

    boolean isMainProcess() {
        return TextUtils.equals(mContext.getPackageName(), getProcessName());
    }

最新更新