【问题标题】:Android volatile not working?Android 不稳定不工作?
【发布时间】:2012-01-15 07:20:11
【问题描述】:

我有一个 Activity 类,其中我有一个静态标志,比如说

public static volatile flag = false;

然后在课堂上,我启动一个线程,它检查标志并做不同的事情。

我还有一个广播接收器,它将标志设置为真或假。

虽然 volatile 会强制标志为最新值。但是我可以看到我的广播接收器将静态标志设置为 true,但我的线程仍然将其设置为 false。

我在这里缺少一些基本的东西吗?任何帮助将不胜感激!

简化代码(更新) - 所以标志应该在一分钟后变为真。但它从来没有。但是来自广播接收器的消息显示它已更改为 true

TestappActivity.java:

package com.test;

import java.util.Calendar;

import android.app.Activity;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.os.Bundle;

public class TestappActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        Intent intent0 = new Intent(this, TestService.class);
        this.startService(intent0);

        Intent intent = new Intent(this, TestReceiver.class);
        AlarmManager am = (AlarmManager) getSystemService(ALARM_SERVICE);
        PendingIntent sender = PendingIntent.getBroadcast(this,
                1, intent,
                PendingIntent.FLAG_UPDATE_CURRENT);
        Calendar slot = Calendar.getInstance();
        int min = slot.get(Calendar.MINUTE);
        slot.set(Calendar.MINUTE, min+1);
        am.set(AlarmManager.RTC_WAKEUP, slot.getTimeInMillis(), sender);
    }
}

TestService.java:

package com.test;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.util.Log;

public class TestService extends Service {

    private static final String TAG = "TestService";

    public static volatile boolean flag = false;

    private MyTopThread mTopThread;

    public TestService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }


    @Override
    public void onCreate() {

    }

    @Override
    public void onDestroy() {

    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        protect();

        // We want this service to continue running until it is explicitly
        // stopped, so return sticky.
        return START_STICKY;
    }


    /**
     * Run protection
     * 
     */
    private void protect() {

        mTopThread = new MyTopThread();
        mTopThread.start();
    }


    private class MyTopThread extends Thread {

        @Override
        public void run() {
            while (true) {
                try {
                    Thread.sleep(150);
                    Log.d(TAG, "Flag is " + TestService.flag);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

            }
        }

    }
}

TestReceiver.java:

package com.test;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

public class TestReceiver extends BroadcastReceiver {
    final static private String TAG = "TestReceiver";

    @Override
    public void onReceive(Context context, Intent intent) {
        Log.d(TAG, "onReceive is triggered ...");
        TestService.flag = true;
        Log.d(TAG, "flag is changed to " + TestService.flag);

    }
}

AndroidManifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.test"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk android:minSdkVersion="8" />

    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <activity
            android:name=".TestappActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <service android:name=".TestService" />

        <receiver
            android:name=".TestReceiver"
            android:process=":remote" >
        </receiver>
    </application>

</manifest>

【问题讨论】:

  • 它应该可以工作。你能发布你的代码吗?
  • 简化版如下:
  • 我没有看到 flag 的值在哪里被更改。
  • 它在我的 Broadcastreceiver 中的 onReceive() 中被更改。我将尝试用完整的代码做一个简单的例子。我当前的代码中有太多其他内容。
  • 您的清单是否声明服务在自己的进程中运行?

标签: android multithreading static broadcastreceiver volatile


【解决方案1】:

我认为问题在于您在自己的进程中运行接收器。来自&lt;receiver&gt;android:process 属性的文档:

如果分配给此属性的名称以冒号 (':') 开头,则会在需要时创建一个专用于应用程序的新进程,并且广播接收器在该进程中运行。

我认为接收者正在修改TestService.flag 的进程本地版本,而不是TestService 使用的版本。尝试从清单中的 &lt;receiver&gt; 标记中删除 android:process 属性。

【讨论】:

  • 你是对的!看到你的服务流程问题后,我也在想同样的问题。我刚试过,它有效!非常感谢!
【解决方案2】:

从此链接

http://www.javamex.com/tutorials/synchronization_volatile.shtml

本质上,volatile 用于指示变量的值将 被不同的线程修改。

【讨论】:

  • 是的,它是在 Broadcastreceiver 中修改的,与我的服务中运行的线程不同
【解决方案3】:

我真的希望你的服务线程不是这个(我没有看到其他的):

private class MyTopThread extends Thread {

    @Override
    public void run() {
        while (true) {
            try {
                Thread.sleep(150);
                Log.d(TAG, "Flag is " + TestService.flag);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

        }
    }

}

因为你这里有while(true),而不是应该的while(!flag)

【讨论】:

  • 这只是我快速组合起来的一个测试程序以显示问题。它是 while(true) 因为我想显示标志值以查看它是否被广播接收器更改。但即使我从广播接收器看到“标志已更改为真”,logcat 仍显示它保持为“标志为假”。
  • 运行它,你就会明白我的意思。您必须为“flag is changed to true”设置过滤器,因为消息“Flag is false”继续在while循环中打印出来。
  • @Howard Li:你确定你设置为 false 的 onReceive 被执行了吗?
  • onReceive 我设置为true,从logcat中可以看到“flag is changed to true”被打印出来了。
  • 您是否也尝试过测试 while(!flag) 并查看线程是否停止?
猜你喜欢
  • 1970-01-01
  • 2015-03-16
  • 2016-10-29
  • 1970-01-01
  • 2012-12-19
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-04-05
相关资源
最近更新 更多