2

我有一个按钮,按下时会启动服务。此服务开始记录来自设备的信息。但是,当我按下运行时,屏幕会在记录时暂时冻结大约 8 秒,然后一旦完成,屏幕就会解冻,会显示一条 toast 消息(这意味着只要你按下按钮就会显示)然后我可以做任何事情。如果我再次进入应用程序屏幕,当应用程序正在记录时(注意:服务始终在运行,因此不会冻结它,只是记录)然后活动再次被冻结,并且在记录完成之前不允许我做任何事情。

我已经尝试过 asynctasks 并在新的可运行/新线程上运行,但这些似乎都不适合我。我是否正确实施它们我不确定,但我想解决这个问题。

这是我在服务类中的 onStartCommand:

    @Override
public int onStartCommand(Intent intent, int flags, int startId) {
    // THIS WOULD BE THE METHOD THAT RUNS WHATEVER YOU WANT TO DO EVERY SO OFTEN SO FOR ME THIS IS GETTING ALL DATA ETC
    getProcessInfo();

    //  LOG DELAY = SECONDS BETWEEN RUNNING SERVICE/METHOD. SET AT THE TOP
    myPendingIntent = PendingIntent.getService(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
    Alarmmgr.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis()+LOG_DELAY*1000, myPendingIntent);
    Intent notificationIntent = new Intent(this, LogOrCheck.class);
    PendingIntent contentIntent = PendingIntent.getActivity(this,
            101, notificationIntent,
            PendingIntent.FLAG_NO_CREATE);

    NotificationManager nm = (NotificationManager) this
            .getSystemService(Context.NOTIFICATION_SERVICE);

    Resources res = this.getResources();
    Notification.Builder builder = new Notification.Builder(this);

    builder.setContentIntent(contentIntent)
                .setSmallIcon(R.drawable.arrow_up_float)
                .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.arrow_down_float))
                .setWhen(System.currentTimeMillis())
                .setAutoCancel(true)
                .setContentTitle("Androigenius")
                .setContentText("Service running OK");
    Notification n = builder.getNotification();


    startForeground(100, n);
    return Service.START_STICKY;
}

这里是我调用 startService 的地方:

but1.setOnClickListener(new View.OnClickListener() {
        public void onClick(View v) {

            if (isClicked) {
                Context context = getApplicationContext();
                Toast toast = Toast.makeText(context, "Please press 'MINIMIZE' at the top to continue logging.", Toast.LENGTH_LONG);
                toast.setGravity(Gravity.TOP|Gravity.RIGHT, 10, 55);
                toast.show();
                System.out.println("Start logging");
                but1.setText("Stop Logging");
                but1.setTextColor(Color.RED);
                // START SERVICE, DONE. (STOP IS BELOW).
                startService(myIntent);
                isClicked = false;
            } 
            else if (!isClicked) {
                System.out.println("Stop Logging");
                but1.setText("Start Logging");
                // STOP SERVICE, DONE.
                stopService(myIntent);
                but1.setTextColor(getResources().getColor(color.cornblue));
                isClicked = true;
            }
4

2 回答 2

1

服务在导致“冻结”的 UI 线程上运行。使用 IntentService 将创建一个在后台自动运行的服务,而不是在 UI 线程上。下面是 IntentService 类的一些细节:

IntentService 类

于 2013-02-14T15:42:41.563 回答
0

默认情况下,该服务也运行在您的应用程序的主线程中,您的 UI 操作在该主线程中发生。因此,如果您在 onStartCommand() 方法中执行长任务,它将阻塞主线程。

为避免此问题,您必须将日志记录任务迁移到单独的线程中。您可以使用 AsycTask 类来执行此操作。类的使用方法请参考http://developer.android.com/reference/android/os/AsyncTask.html

于 2013-02-14T15:29:30.793 回答