【安卓筆記】作為內部類的廣播接收者

rowandjj發表於2014-02-19
【全面解析service元件】一文中我們提到了採用廣播接收者(BroadcastReceiver)使service與activity進行通訊。在該例子中我們在service子類中定義了一個自定義的廣播接收者,該廣播接收者監聽activity發出的特定的廣播,並從而觸發onReceive方法,我們在此方法中間接呼叫service類的方法,從而實現了service與activity間的通訊。例子中的廣播接收者是採用動態的方式註冊的(即使用registerReceiver和unregisterReceiver方法),但是我們知道,廣播接收者還有一種靜態的方式註冊,即在清單檔案中配置receiver節點(之所以沒采用靜態註冊是因為這種方式有一些需要注意的地方)。
       那麼下面我們就採用靜態方式註冊一個作為內部類的廣播接收者  

1.activity定義(佈局很簡單,就一個button,不貼程式碼了)
package com.example.brocastdemo;

import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

public class MainActivity extends Activity
{
    private Button but = null;

    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        but = (Button) findViewById(R.id.but);
        but.setOnClickListener(new OnClickListener()
        {
            
            @Override
            public void onClick(View v)
            {
                Intent intent = new Intent("com.aaa");
                MainActivity.this.sendBroadcast(intent);
            }
        });
    }

    public static class MyReceiver extends BroadcastReceiver//作為內部類的廣播接收者
    {
        @Override
        public void onReceive(Context context, Intent intent)
        {
            if (intent.getAction().equals("com.aaa"))
            {
                Log.i("MainActivity","成功收到廣播");
            }
        }
    }
}
2.清單檔案:
<receiver 
            android:name="com.example.brocastdemo.MainActivity$MyReceiver"
            >
            <intent-filter >
                <action android:name="com.aaa"/>
            </intent-filter>
            
  </receiver>

程式碼寫好之後,我們測試一下,點選button,發現logcat成功列印了日誌,測試成功。

下面總結一下作為內部類的廣播接收者在註冊的時候需要注意的地方
1.清單檔案註冊廣播接收者時,廣播接收者的名字格式需要注意因為是內部類,所以需要在內部類所在的類與內部類之間加上$符號:
 android:name="com.example.brocastdemo.MainActivity$MyReceiver"
2.內部類在宣告時一定要寫成靜態內部類(class關鍵字前加上static)否則會丟擲類似這樣的異常:
02-19 09:11:14.650: E/AndroidRuntime(1771): java.lang.RuntimeException: Unable to instantiate receiver com.example.brocastdemo.MainActivity$MyReceiver: java.lang.InstantiationException: can't instantiate class com.example.brocastdemo.MainActivity$MyReceiver; no empty constructor
大家可能會發現,採用靜態註冊的作為內部類的廣播接收者使用起來很不方便,因為是靜態內部類,所以該類中如果想使用外部類的變數/方法,該變數/方法也得是靜態的。
所以還是強烈推薦大家使用動態方式註冊,下面將例子改為動態方式註冊。

修改後的activity:
package com.example.brocastdemo;

import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
//採用動態方式註冊一個作為內部類的廣播接收者
public class MainActivity extends Activity
{
    private Button but = null;
    private MyReceiver receiver = null;

    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        but = (Button) findViewById(R.id.but);
        
        /*動態方式註冊廣播接收者*/
        receiver = new MyReceiver();
        IntentFilter filter = new IntentFilter();
        filter.addAction("com.aaa");
        this.registerReceiver(receiver, filter);
        
        but.setOnClickListener(new OnClickListener()
        {
            
            @Override
            public void onClick(View v)
            {
                Intent intent = new Intent("com.aaa");
                MainActivity.this.sendBroadcast(intent);
            }
        });
    }

    public class MyReceiver extends BroadcastReceiver
    {
        @Override
        public void onReceive(Context context, Intent intent)
        {
            if (intent.getAction().equals("com.aaa"))
            {
                Log.i("MainActivity","成功收到廣播");
            }
        }
    }
    @Override
    protected void onDestroy()
    {
        if(receiver!=null)
            this.unregisterReceiver(receiver);
    }
}
此時,清單檔案中不用加receiver節點了。測試當然也是通過的,這樣做就方便多了!




相關文章