內部類廣播需要public和static

採用廣播接收者(BroadcastReceiver)使serviceactivity進行通信。在該例子中我們在service子類中定義了一個自定義的廣播接收者,該廣播接收者監聽activity發出的特定的廣播,並從而觸發onReceive方法,我們在此方法中間接調用service類的方法,從而實現了serviceactivity間的通信。例子中的廣播接收者是採用動態的方式註冊的(即使用registerReceiverunregisterReceiver方法),但是我們知道,廣播接收者還有一種靜態的方式註冊,即在清單文件中配置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>

下面總結一下作爲內部類的廣播接收者在註冊的時候需要注意的地方
1.清單文件註冊廣播接收者時,廣播接收者的名字格式需要注意。因爲是內部類,所以需要在內部類所在的類與內部類之間加上$符號:

 android:name="com.example.brocastdemo.MainActivity$MyReceiver"

2.內部類在聲明時一定要寫成靜態內部類(class關鍵字前加上staticpublic。否則會拋出類似這樣的異常:動態則不要

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節點了。測試當然也是通過的,這樣做就方便多了!

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章