Android BroadcastReceiver onReceive在MainActivity中更新TextView

时间:2014-04-04 18:02:21

标签: java android user-interface background textview

在MainActivity中我有一个TextView:textV1。我还在MainActivity中有一个方法来更新textview:

public void updateTheTextView(final String t) {
    MainActivity.this.runOnUiThread(new Runnable() {
        public void run() {
            TextView textV1 = (TextView) findViewById(R.id.textV1);
            textV1.setText(t);
        }
    });
}

在BroadcasrReceiver中,我需要更新MainActivity中textV1中的文本。

public class NotifAlarm extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
            // other things done here like notification

            // NEED TO UPDATE TEXTV1 IN MAINACTIVITY HERE
    }
}

如何做到这一点? BroadcastReceiver从服务运行。这段代码我无法改变。我可以从onReceive()访问和更改MainActivity中的textV1吗?我尝试过很多东西,但都失败了。

5 个答案:

答案 0 :(得分:34)

MainActivity初始化MainActivity类的变量,如下所示。

public class MainActivity extends Activity {
    private static MainActivity ins;
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ins = this;     
    }

    public static MainActivity  getInstace(){
        return ins;
    }

    public void updateTheTextView(final String t) {
        MainActivity.this.runOnUiThread(new Runnable() {
            public void run() {
                TextView textV1 = (TextView) findViewById(R.id.textV1);
                textV1.setText(t);
            }
        });
    }
}


public class NotifAlarm extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        try {
            MainActivity  .getInstace().updateTheTextView("String");
        } catch (Exception e) {

        }           
    }
}

答案 1 :(得分:2)

创建一个类的实例,然后将值传递给更改TextView值的函数请按照以下步骤操作: 在你的BroadcastReceiver overRide onReceive方法并粘贴这些行或更改主题,如你所愿

private Handler handler = new Handler(); // Handler used to execute code on the UI thread
// Post the UI updating code to our Handler
handler.post(new Runnable() {
    @Override
    public void run() {
    //Toast.makeText(context, "Toast from broadcast receiver", Toast.LENGTH_SHORT).show();
    YourActivityToUpdate.updateTheTextView(message);
    YourActivityToUpdateinst = YourActivityToUpdate.instance();
        if(inst != null)  { // your activity can be seen, and you can update it's context
        inst.updateTheTextView(message);
        }
    }
});

现在我们解释一下updateTheTextView和inst 在YourActivityToUpdate类中粘贴这些行

private static SignUpVerify mInst;

public static SignUpVerify instance() {
        return mInst;
}
@Override
public void onStart() {
    super.onStart();
    mInst = this;
}

@Override
public void onStop() {
    super.onStop();
    mInst = null;
}

这是应该放在YourActivityToUpdate类中的updateTheTextView方法

public void updateTheTextView(final String verifyCodeValue) {
                Log.i("verifyCodeValue", verifyCodeValue);
                YourTextViewToUpdate.setText(verifyCodeValue);
    }

我认为这是一个更好的方式,感谢" kevin-lynx"

答案 2 :(得分:0)

如果有人正在搜索这个确切的解决方案,但是在Kotlin中,请执行以下操作:

class MainActivity : AppCompatActivity() {

    companion object {
        var ins: MainActivity? = null
        fun getInstance(): MainActivity? {
            return ins
        }
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        ins = this
    }

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ins = this;     
    }

    fun updateTheTextView(t: String) {
        this@MainActivity.runOnUiThread {
            val textV1 = findViewById<TextView>(R.id.textV1)
            textV1.text = t
        }
    }
}

class NotifAlarm : BroadcastReceiver() {
    override fun onReceive(context: Context?, intent: Intent?) {
        try {
            MainActivity.getInstance()?.updateTheTextView("The String")
        } catch (e: Exception) {

        }
    }
}

答案 3 :(得分:0)

使用界面

处理这种情况的另一种方法是使用接口。我将描述使用这种方法的优势,但首先让我们看看它是如何完成的。

请遵循以下步骤:
1)创建一个界面

public interface MyBroadcastListener{

    public void doSomething(String result);

}

2)在BroadCastReceiver中初始化监听器

public class NotifAlarm extends BroadcastReceiver {

    private MyBroadcastListener listener;

    @Override
    public void onReceive(Context context, Intent intent) {

        listener = (MyBroadcastListener)context;

        // other things done here like notification

        // NUPDATE TEXTV1 IN MAINACTIVITY HERE
        listener.doSomething("Some Result");
    }
}

3)在“活动”中实现界面并覆盖方法

public YourActivity extends AppCompatActivity implements MyBroadcastListener{

    // Your Activity code 

    public void updateTheTextView(String t) {
        TextView textV1 = (TextView) findViewById(R.id.textV1);
        textV1.setText(t);
    }

    @Override
    public void doSomething(String result){
         updateTheTextView(result);          // Calling method from Interface
    }

 }

使用界面的优势吗?

  • 当您将BroadcastReceiver放在其他文件中时
  • 解耦的BroadcastReceiver
  

使用接口方法使BroadcastReceiver独立于任何   活动。可以说,将来您想使用此BroadCastReceiver   与另一个活动,该活动从BroadcastReceiver获取结果   然后从结果开始一个DetailActivity。这是完全   不同的任务,但您将使用相同的BroadcastReceiver,甚至   更改BroadcastReceiver内部的代码。

该怎么做?
在“活动”中实现接口并覆盖方法。就是这样!

public ListActivity extends AppCompatActivity implements MyBroadcastListener{

    // Your Activity code 

    public void startDetailActivity(String title) {
        Intent i = new Intent(ListActivity,this, DetailActivity.class);
        i.putExtra("Title", title);
        startActivity(i);
    }

    @Override
    public void doSomething(String result){
         startDetailActivity(String title);    // Calling method from Interface
    }

 }

答案 4 :(得分:0)

在您的broadcastreceiver类中发送广播

public class mybroadcaster extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
    context.sendBroadcast(new Intent("updatetext"));
  }
}

在活动中注册广播接收器并调用它,在onReceive进行工作,然后在onDestroy()取消广播器的注册

public class MyActivity  extends Activity{

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    registerReceiver(broadcastReceiver, new IntentFilter("updatetext"));
}

BroadcastReceiver broadcastReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        // do your work here
    }
};

@Override
protected void onDestroy() {
    super.onDestroy();
    unregisterReceiver(broadcastReceiver);
  }
}