手头的问题:
我必须创建一个持续运行的Service
。这项服务监控5个应用程序说你的手机上安装了5个安卓游戏。该服务需要获取以下信息:
1.游戏开启和运行了多少次?
2.每场比赛的运行时间。
例如:说如果我在我的应用中安装了此服务。我让它运行了一个月。我需要在应用程序的屏幕上显示此类信息:
游戏 游戏运行次数 游戏持续时间
第一场比赛20次,总共打了15个小时
第2场比赛16次总共打了25个小时
...
...
第5场比赛共10次,共12小时
可能的方法: 当应用程序加载时,它会进入内存。注意到系统在应用程序启动时计时。当应用程序结束或被放入后台时,请注意时间。
所以说如果一个应用程序在晚上9点被带到内存并且在晚上9:30退出到后台,这给了我们30分钟的游戏时间。下次播放应用程序时,持续时间将从存储在某种变量中的上一个播放添加到30,依此类推。 每次将应用程序带入内存时,正在播放的计数器应该增加1。因此,给我们一个应用程序的播放次数。
编码:
我不知道Android中的Service
,因为我从未真正研究过它们。任何与我手头的问题相关的教程都会非常有用。
其次,如果还有另一种方法可以做到这一点。我也想知道。我真的可以使用一些代码片段来启动这个项目。
答案 0 :(得分:29)
正如您所写,任务是关于监控第三方应用程序,除了定期读取进程列表和检测前台进程之外,没有其他解决方案。你需要一个服务。不幸的是,Android没有为前台进程更改提供广播事件等手段。
这项任务实际上需要大量代码,至少比普通答案所能包含的代码要多得多。我在这里发布了它的一部分,但你应该解决幕后留下的许多细微差别,例如同步和发布之间的持久信息。这只是一个骨架。
首先,让代码编写一个应用程序对象,这是一个注册所有实例相关内容的好地方。
<强> MonitorApp 强>
public class MonitorApp extends Application
{
// actual store of statistics
private final ArrayList<HashMap<String,Object>> processList = new ArrayList<HashMap<String,Object>>();
// fast-access index by package name (used for lookup)
private ArrayList<String> packages = new ArrayList<String>();
public ArrayList<HashMap<String,Object>> getProcessList()
{
return processList;
}
public ArrayList<String> getPackages()
{
return packages;
}
// TODO: you need to save and load the instance data
// TODO: you need to address synchronization issues
}
然后让我们起草一项活动。
<强> MonitorActivity 强>
import static ProcessList.COLUMN_PROCESS_NAME;
import static ProcessList.COLUMN_PROCESS_PROP;
import static ProcessList.COLUMN_PROCESS_COUNT;
import static ProcessList.COLUMN_PROCESS_TIME;
public class MonitorActivity extends Activity implements MonitorService.ServiceCallback
{
private ArrayList<HashMap<String,Object>> processList;
private MonitorService backgroundService;
private MyCustomAdapter adapter = null;
private ListView listView = null;
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main); // TODO: provide your layout
listView = (ListView)findViewById(R.id.id_process_listview);
createAdapter();
this.bindService(
new Intent(this, MonitorService.class),
serviceConnection,
Context.BIND_AUTO_CREATE);
}
private void createAdapter()
{
processList = ((MonitorApp)getApplication()).getProcessList();
adapter = new MyCustomAdapter(this, processList, R.layout.complex_list_item,
new String[]
{
COLUMN_PROCESS_NAME,
COLUMN_PROCESS_PROP, // TODO: you may calculate and pre-fill this field
// from COLUMN_PROCESS_COUNT and COLUMN_PROCESS_TIME
// so eliminating the need to use the custom adapter
},
new int[]
{
android.R.id.text1,
android.R.id.text2
});
listView.setAdapter(adapter);
}
// callback method invoked by the service when foreground process changed
@Override
public void sendResults(int resultCode, Bundle b)
{
adapter.notifyDataSetChanged();
}
private class MyCustomAdapter extends SimpleAdapter
{
MyCustomAdapter(Context context, List<? extends Map<String, ?>> data, int resource, String[] from, int[] to)
{
super(context, data, resource, from, to);
}
@Override
public View getView (int position, View convertView, ViewGroup parent)
{
View result = super.getView(position, convertView, parent);
// TODO: customize process statistics display
int count = (Integer)(processList.get(position).get(COLUMN_PROCESS_COUNT));
int seconds = (Integer)(processList.get(position).get(COLUMN_PROCESS_TIME));
return result;
}
}
private ServiceConnection serviceConnection = new ServiceConnection()
{
@Override
public void onServiceConnected(ComponentName className, IBinder service)
{
LocalBinder binder = (LocalBinder)service;
backgroundService = binder.getService();
backgroundService.setCallback(MonitorActivity.this);
backgroundService.start();
}
@Override
public void onServiceDisconnected(ComponentName className)
{
backgroundService = null;
}
};
@Override
public void onResume()
{
super.onResume();
if(backgroundService != null)
{
backgroundService.setCallback(this);
}
}
@Override
public void onPause()
{
super.onPause();
if(backgroundService != null)
{
backgroundService.setCallback(null);
}
}
}
该活动启动后台工作服务,该服务实际上监控进程。您可以将服务注册从活动移动到应用程序实例中。服务本身就是这样的:
<强> MonitorService 强>
public class MonitorService extends Service
{
private boolean initialized = false;
private final IBinder mBinder = new LocalBinder();
private ServiceCallback callback = null;
private Timer timer = null;
private final Handler mHandler = new Handler();
private String foreground = null;
private ArrayList<HashMap<String,Object>> processList;
private ArrayList<String> packages;
private Date split = null;
public static int SERVICE_PERIOD = 5000; // TODO: customize (this is for scan every 5 seconds)
private final ProcessList pl = new ProcessList(this)
{
@Override
protected boolean isFilteredByName(String pack)
{
// TODO: filter processes by names, return true to skip the process
// always return false (by default) to monitor all processes
return false;
}
};
public interface ServiceCallback
{
void sendResults(int resultCode, Bundle b);
}
public class LocalBinder extends Binder
{
MonitorService getService()
{
// Return this instance of the service so clients can call public methods
return MonitorService.this;
}
}
@Override
public void onCreate()
{
super.onCreate();
initialized = true;
processList = ((MonitorApp)getApplication()).getProcessList();
packages = ((MonitorApp)getApplication()).getPackages();
}
@Override
public IBinder onBind(Intent intent)
{
if(initialized)
{
return mBinder;
}
return null;
}
public void setCallback(ServiceCallback callback)
{
this.callback = callback;
}
private boolean addToStatistics(String target)
{
boolean changed = false;
Date now = new Date();
if(!TextUtils.isEmpty(target))
{
if(!target.equals(foreground))
{
int i;
if(foreground != null && split != null)
{
// TODO: calculate time difference from current moment
// to the moment when previous foreground process was activated
i = packages.indexOf(foreground);
long delta = (now.getTime() - split.getTime()) / 1000;
Long time = (Long)processList.get(i).get(COLUMN_PROCESS_TIME);
if(time != null)
{
// TODO: add the delta to statistics of 'foreground'
time += delta;
}
else
{
time = new Long(delta);
}
processList.get(i).put(COLUMN_PROCESS_TIME, time);
}
// update count of process activation for new 'target'
i = packages.indexOf(target);
Integer count = (Integer)processList.get(i).get(COLUMN_PROCESS_COUNT);
if(count != null) count++;
else
{
count = new Integer(1);
}
processList.get(i).put(COLUMN_PROCESS_COUNT, count);
foreground = target;
split = now;
changed = true;
}
}
return changed;
}
public void start()
{
if(timer == null)
{
timer = new Timer();
timer.schedule(new MonitoringTimerTask(), 500, SERVICE_PERIOD);
}
// TODO: startForeground(srvcid, createNotification(null));
}
public void stop()
{
timer.cancel();
timer.purge();
timer = null;
}
private class MonitoringTimerTask extends TimerTask
{
@Override
public void run()
{
fillProcessList();
ActivityManager activityManager = (ActivityManager)MonitorService.this.getSystemService(ACTIVITY_SERVICE);
List<RunningTaskInfo> taskInfo = activityManager.getRunningTasks(1);
String current = taskInfo.get(0).topActivity.getPackageName();
// check if current process changed
if(addToStatistics(current) && callback != null)
{
final Bundle b = new Bundle();
// TODO: pass necessary info to UI via bundle
mHandler.post(new Runnable()
{
public void run()
{
callback.sendResults(1, b);
}
});
}
}
}
private void fillProcessList()
{
pl.fillProcessList(processList, packages);
}
}
该服务使用帮助程序类来构建进程列表。
<强> ProcessList中强>
public abstract class ProcessList
{
// process package name
public static final String COLUMN_PROCESS_NAME = "process";
// TODO: arbitrary property (can be user-fiendly name)
public static final String COLUMN_PROCESS_PROP = "property";
// number of times a process has been activated
public static final String COLUMN_PROCESS_COUNT = "count";
// number of seconds a process was in foreground
public static final String COLUMN_PROCESS_TIME = "time";
private ContextWrapper context;
ProcessList(ContextWrapper context)
{
this.context = context;
}
protected abstract boolean isFilteredByName(String pack);
public void fillProcessList(ArrayList<HashMap<String,Object>> processList, ArrayList<String> packages)
{
ActivityManager activityManager = (ActivityManager)context.getSystemService(Context.ACTIVITY_SERVICE);
List<RunningAppProcessInfo> procInfo = activityManager.getRunningAppProcesses();
HashMap<String, Object> hm;
final PackageManager pm = context.getApplicationContext().getPackageManager();
for(int i = 0; i < procInfo.size(); i++)
{
String process = procInfo.get(i).processName;
String packageList = Arrays.toString(procInfo.get(i).pkgList);
if(!packageList.contains(process))
{
process = procInfo.get(i).pkgList[0];
}
if(!packages.contains(process) && !isFilteredByName(process))
{
ApplicationInfo ai;
String applicationName = "";
for(int k = 0; k < procInfo.get(i).pkgList.length; k++)
{
String thisPackage = procInfo.get(i).pkgList[k];
try
{
ai = pm.getApplicationInfo(thisPackage, 0);
}
catch(final NameNotFoundException e)
{
ai = null;
}
if(k > 0) applicationName += " / ";
applicationName += (String)(ai != null ? pm.getApplicationLabel(ai) : "(unknown)");
}
packages.add(process);
hm = new HashMap<String, Object>();
hm.put(COLUMN_PROCESS_NAME, process);
hm.put(COLUMN_PROCESS_PROP, applicationName);
processList.add(hm);
}
}
// optional sorting
Comparator<HashMap<String, Object>> comparator = new Comparator<HashMap<String, Object>>()
{
public int compare(HashMap<String, Object> object1, HashMap<String, Object> object2)
{
return ((String)object1.get(COLUMN_PROCESS_NAME)).compareToIgnoreCase((String)object2.get(COLUMN_PROCESS_NAME));
}
};
Collections.sort(processList, comparator);
packages.clear();
for(HashMap<String, Object> e : processList)
{
packages.add((String)e.get(COLUMN_PROCESS_NAME));
}
}
}
最后,清单。
<强>的AndroidManifest.xml 强>
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.yourpackage"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk android:minSdkVersion="8" android:targetSdkVersion="18" />
<uses-permission android:name="android.permission.GET_TASKS" />
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<activity
android:name=".MonitorActivity"
android:label="@string/app_name"
android:configChanges="orientation|keyboardHidden" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service android:name=".MonitorService" />
</application>
</manifest>
正如您所看到的,已经有很多代码了。它部分从工作应用程序中提取,但我根据您的需求进行了快速更改,因此可能存在拼写错误,所有导入都被跳过,等等。不过,我希望这会有所帮助。
ADDENDUM:Lollipop +
注意:最新的Android版本打破了上述方法。以下是官方文档中关于getRunningTasks方法和其他方法的说明:
从LOLLIPOP开始,此方法不再适用于第三方应用程序:引入以文档为中心的最新版本意味着它可以将人员信息泄露给调用者。为了向后兼容,它仍将返回其数据的一小部分:至少是调用者自己的任务,以及可能已知不敏感的其他一些任务,如home。
我认为这是一种矫枉过正,可以更有选择性和方便的方式完成。更值得一提的是,考虑到Google的许多内置功能存在隐私问题,这似乎太戏剧化了。无论如何,我们无能为力。
唯一的解决方法是实施Android辅助功能服务(更多信息here和here)并拦截所有操作,使应用程序从那里获得并失去焦点。用户应手动启用该服务!您的应用程序应该以某种方式指导用户这样做。
答案 1 :(得分:9)
我的想法,
开发应用程序,将其部分数据公开给其他进程以访问它们。
Google“如何使用内容提供商”。
您的每个应用程序都需要记录他们执行的所有活动。这些数据将被公开访问。
您不需要一直运行服务。每当您的监控应用程序打开时。它只需要从您想要的所有应用程序中获取数据并相应地显示它们。
对于您希望在后台定期在服务器上提交该数据的情况。 然后你需要一项服务。 但是你仍然不需要永远运行它。 做这样的事。
i)从服务获取数据并上传到服务器。
ii)停止服务并在T1时间后安排警报再次开始自己的服务以再次执行步骤。
iii)T1取决于要求。如何在服务器上刷新数据。
如果你想要如何实现我上面说的话,我甚至可以告诉你代码片段指针。 但我建议你在google上找到它并先自己做。如果您遇到任何困难,请在StackOverflow上发布另一个特定问题。
希望这会有所帮助。 :)
编辑:
首先。它只有你最终必须编码。从这里获取指针和帮助。不要期望完整的代码。请查看以下详细信息。
A)
您需要将您的应用设为内容提供商。检查以下链接。并按照其中的示例进行操作。您将最终进入一个应用程序,其他人可以使用其数据。它很容易前进。
http://developer.android.com/guide/topics/providers/content-providers.html
http://developer.android.com/guide/topics/providers/content-provider-creating.html
http://www.vogella.com/articles/AndroidSQLite/article.html
您的所有游戏应用都需要公开访问数据。 一旦你完成了这件事。
现在您只需要访问监控应用中的所有数据并显示它。
B)
正如我所说,如果你想通过服务这样做。 你不需要永远运行它。只需启动服务一次,从内容提供商加载数据并执行您想做的任何事情。设置警报以在稍后阶段调用服务以重新执行相同的操作。
我假设您的服务每10分钟可以从内容提供商(您想要监控的游戏应用程序)获取数据。这就像是。
public class MonitoringService extends Service {
private static final String ACTION_FETCH = "action_fetch";
private static final int REPEAT_DATALOAD_INTERVAL_MS = 10 * 60 * 1000; // 10 Min
private static PendingIntent makeSelfPendingIntent(Context context) {
PendingIntent intent = PendingIntent.getService(
context, 0, makeSelfIntent(context), 0);
return intent;
}
private static Intent makeSelfIntent(Context context) {
Intent intent = new Intent(context, MonitoringService.class);
intent.setAction(ACTION_FETCH);
return intent;
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
if (intent != null && intent.getAction() != null) {
String action = intent.getAction();
if (action.equals(ACTION_FETCH)) {
loadDataFromContentProviderDoWhateverYouWantThen();
setAlarmToRedoLoad();
stopSelf();
}
}
return Super.onStartCommand(intent, flags, startId);
}
private void setAlarmToRedoLoad() {
AlarmManager alarmManager = (AlarmManager)
getSystemService(Context.ALARM_SERVICE);
alarmManager.set(AlarmManager.RTC_WAKEUP,
System.currentTimeMillis() + REPEAT_DATALOAD_INTERVAL_MS,
makeSelfPendingIntent(getApplicationContext()));
}
private void loadDataFromContentProviderDoWhateverYouWantThen(){
check this link how to load data from content provider.
as your games app are content providers. It should be loaded easily.
then do whatever you want display, upload anything
http://developer.android.com/guide/topics/providers/content-provider-basics.html
}
// Call this method from onCreate of your monitoring app
public static void start(Context context) {
Intent intent = makeSelfIntent(context);
context.startService(intent);
}
}
C)确保您允许监控应用的用户随时停止此服务。在这种情况下不要忘记取消警报。所以它永远不会在后台运行。
D)你也可以根据广播进行广播。每当您的游戏应用程序保存数据时,他们应该广播此事件,并在收听广播后调用您的服务。然后加载它。
E)正如Phill所说,您也可以使用Google Analytics。
您希望如何实际执行此操作现在取决于您的要求
希望这会有所帮助。尝试一下,让我知道你面临的更多问题
答案 2 :(得分:1)
要获取当前正在运行的应用程序,您可以使用:
ActivityManager am = (ActivityManager) mContext.getSystemService(Activity.ACTIVITY_SERVICE);
String packageName = am.getRunningTasks(1).get(0).topActivity.getPackageName();
您可以定期从正在运行的服务中查看此信息。您还需要在清单中定义所需的权限:android.permission.GET_TASKS
答案 3 :(得分:0)
使用此示例代码检查浏览器是否正在运行
ActivityManager activityManager = (ActivityManager) this.getSystemService( ACTIVITY_SERVICE );
List<RunningAppProcessInfo> procInfos = actvityManager.getRunningAppProcesses();
for(int i = 0; i < procInfos.size(); i++){
if(procInfos.get(i).processName.equals("com.android.browser")) {
Toast.makeText(getApplicationContext(), "Browser is running", Toast.LENGTH_LONG).show();
}
}
要创建服务,您必须创建一个从Service基类扩展的类,并在while(true)
函数中的onStartCommand()
等无限循环中实现该函数。不要忘记在<service></service>
标记之间的清单文件中添加您的服务,一个好的教程在http://www.vogella.com/articles/AndroidServices/article.html上
答案 4 :(得分:0)
以下解决方案基于您拥有您尝试监控的所有五个应用程序的假设。如果是其他方面,我将为您提供另一种解决方案。
让我们现在解决问题。
您应该使用Intent
来广播有关您的应用运行的消息。使用活动onPause
和onResume
来广播消息。
现在从onPause和onResume,
调用此函数 public void broadcastMessage(String method_name){
Intent intent=new Intent("com.service.myservice");
Bundle bundle=new Bundle();
bundle.putString("app_name", this.getApplicationInfo().packageName);
bundle.putLong("time", System.currentTimeMillis());
//just to keep track if it is on pause or in on resume
bundle.putString("method", method_name);
intent.putExtras(bundle);
this.startService(intent);
}
您可以按照以下方式调用此方法
public void onPause(){
super.onPause();
broadcastMessage(Thread.currentThread().getStackTrace()[1].getMethodName());
}
或者
public void onPause(){
super.onPause();
broadcastMessage("onPause");
}
那么为什么onPause和onResume因为游戏玩家只在这些函数调用之间玩游戏。
设计服务
如果您想知道原因,请使用IntentService
并阅读this。
@Override
protected void onHandleIntent(Intent intent) {
// TODO Auto-generated method stub
Bundle bundle=intent.getExtras();
String app_name=bundle.getString("app_name");
long time=bundle.getLong("time");
//then do whatever you wanna do with this data
}
<强>存储强>
ContentProvider
是存储数据的好选择,它也非常强大;在Android上集成和扩展,但在这种情况下,您似乎只想存储一些非常简单的信息,因此您也可以选择非常简单的东西。
下面的代码使用SharedPreferences
。
protected void storeData(String app_name, long time) {
if ((app_name==null)||(time<=0)) return;
SharedPreferences sharedPreferences = PreferenceManager .getDefaultSharedPreferences(getApplicationContext());
Editor editor = sharedPreferences.edit();
//concatenating app name just to make key unique
long start_time=sharedPreferences.getLong(app_name+"_start_time", -1);
if (start_time==-1){
//start time is empty means this is my start time
// and user just started the app
editor.putLong(app_name+"_start_time", time);
editor.putLong(app_name+"_counter", sharedPreferences.getLong(app_name+"_counter", 0)+1);
}else{
//user is now closing the app.
long time_played=time-start_time;
long total_time=sharedPreferences.getLong(app_name+"_total_time", 0);
total_time=total_time+time_played;
//saving total time
editor.putLong(app_name+"_total_time", time);
//removing start time for next pass
editor.remove(app_name+"_start_time");
}
editor.commit();
}
public long getTotalTimeUserSpent(String app_name){
SharedPreferences sharedPreferences = PreferenceManager .getDefaultSharedPreferences(getApplicationContext());
return sharedPreferences.getLong(app_name+"_total_time", 0);
}
public long numberOfTimeUserPlayed(String app_name){
SharedPreferences sharedPreferences = PreferenceManager .getDefaultSharedPreferences(getApplicationContext());
return sharedPreferences.getLong(app_name+"_counter", 0);
}
只需从onHandleIntent
调用这些方法即可。
最后一块拼图将是分发。您可以将其分发到单独的应用程序或这五个应用程序的一部分。单独的应用程序是最简单的方法。但是,如果您选择将此作为所有这五个应用程序的一部分使用,请阅读this讨论。
答案 5 :(得分:-3)
在Android设备上安装Covenant Eyes将监控手机预安装浏览器上的所有互联网流量,并报告设备上正在使用的其他应用。