第一篇:[安卓開發] Broadcast 三種廣播的使用總結
[安卓開發] Broadcast 三種廣播的使用
總結
1、什么是安卓的Broadcast?
安卓的四大組件之一,是一種廣泛應用在應用程序之間傳輸信息的機制。
2、什么是安卓的BroadcastReceiver?
是對發送出來的廣播進行過濾接收并響應的一類組件,它就是用來接收來自系統和應用中的廣播。例如系統的廣播有開機廣播: 系統在開機時候會發送開機廣播,程序接收到之后,能進行開機自啟動。網絡狀態改變廣播: 3g變wifi、網絡斷開等。電量改變廣播等等。。
3、Anroid為什么要這樣設計?
大大減少開發工作量和開發周期
作為開發者,只需要掌握BroadcastReceiver
4、怎么理解Broadcast和BroadcastReceiver ?
Broadcast就像現實中的廣播電臺,他發廣播信號來,然后我們用收音機來接收,然后處理,并且播放出聲音,BroadcastReceiver就相當于那臺收音機。
5、使用方法
發送:
把信息裝入一個Intent對象(如:Action、Category),通過調相應的方法將Intent對象以廣播的方式發送出去: sendBroadcast();
sendOrederBroadcast();sendStickyBroadcast();接收:
當Intent發送之后,所有已經注冊receivedBroadcastReceiver會檢查注冊時的IntentFilter是否與發送的Intent相匹配,若匹配則就會調用BroadcastReceiver的onReceiver()方法。所以當我們定義一個BroadcastReceiver的時候,都需要實現onReceiver()方法。注意:
BroadcastReceiver需要注冊
靜態注冊
代碼動態注冊
6、注意!!
BroadReceiver生命周期只有十秒左右,不能直接執行耗時操作,不然會出現ANR(應用程序無響應),也不能用子線程來做,因為每次廣播來的時候都會創建一個Reveiver對象,并且調用onReceiver,執行完之后,對象會立刻被銷毀,子線程也沒了
要做耗時操作的話,應該通過發送Intent給Service,由Service來完成。
動態注冊廣播接受者的話要在Destory回調事件進行unregister
7、廣播的分類
普通廣播(Normal broadcast)所有監聽該廣播接受者都可以監聽到該廣播 同級別接收先后順序是隨機的(無序)級別低的后收到廣播
接收器不能截斷廣播的繼續傳播,也不能處理廣播 同級別動態注冊高于靜態注冊 有序廣播(Oredered broadcast)按照接收者的優先順序來接收廣播,優先級別在intent-filter中的priority中聲明,-1000到1000之間,值越大優先級越高,可以終止廣播的繼續傳播,接受者可以修改intent的內容。同級別接收順序是隨機的 級別低的后收到
能截斷廣播的繼續傳播,高級別的廣播接收器接收廣播后能決定時候截斷。能處理廣播
同級別動態注冊高于靜態注冊
異步廣播(粘滯性滯留廣播)ps:已被棄用
不能處理結果給下一個接收者,無法終止廣播。一直存在
可以先發送廣播,再注冊接收器
需要在清單文件添加android.permission.BROADCAST_STICKY權限
8、Demo
布局actibity_main三個按鈕:
android:layout_width=“match_parent” android:layout_height=“match_parent” android:orientation=“vertical”>
MainActivity.Java
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
private Button btOne;
private Button btTwo;
private Button btThree;
MyReiceiverThree myReiceiver = new MyReiceiverThree();
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btOne =(Button)findViewById(R.id.bt_one);
btTwo =(Button)findViewById(R.id.bt_two);
btThree =(Button)findViewById(R.id.bt_three);
btOne.setOnClickListener(this);
btTwo.setOnClickListener(this);
btThree.setOnClickListener(this);
//動態注冊,在當前activity的生命周期內運行
/*IntentFilter filter= new IntentFilter(Config.BC_ONE_ACTION);
MyReiceiver myReiceiver = new MyReiceiver();
registerReceiver(myReiceiver,filter);*/
}
@Override
public void onClick(View view){
Intent intent = new Intent();
switch(view.getId()){
case R.id.bt_one:
//發送普通廣播
intent.setAction(Config.BC_ONE_ACTION);
intent.putExtra(“msg”,“這是普通廣播”);
sendBroadcast(intent);
break;
case R.id.bt_two:
//有序廣播
intent.setAction(Config.BC_TWO_ACTION);
intent.putExtra(“msg”,“這是有序廣播”);
sendOrderedBroadcast(intent,null);//其中第二個參數是設置權限,即接收器必須具有相應的權限才能正常接收到廣播。
break;
case R.id.bt_three:
//異步廣播
intent.setAction(Config.BC_THREE_ACTION);
intent.putExtra(“msg”,“這是異步廣播”);
sendStickyBroadcast(intent);
//可以先發送 后注冊
IntentFilter filter = new IntentFilter(Config.BC_THREE_ACTION);
registerReceiver(myReiceiver, filter);
break;
}
}
@Override
protected void onDestroy(){
super.onDestroy();
unregisterReceiver(myReiceiver);
} }
MyReceiver.java
public class MyReiceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent){
//獲取處理的的廣播,普通廣播不能獲取處理
//true代表如果前面的接收器沒有存放數據,則自動創建一個空的Bundle對象,false則表示如果前面的接收器如果沒有存放任何數據則返回null。
Bundle bundle= getResultExtras(true);
System.out.println(“接收器1接收到處理的值:”+bundle.getString(“msg”));
System.out.println(“接收器1:”+intent.getStringExtra(“msg”));
} }
MyReceiverTwo.java
public class MyReiceiverTwo extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent){
//Toast.makeText(context,intent.getStringExtra(“msg”),Toast.LENGTH_SHORT).show();
System.out.println(“接收器2:”+intent.getStringExtra(“msg”));
abortBroadcast();
//截斷廣播,不讓別的接收器繼續接收,有序廣播才能成功攔截
//處理廣播
Bundle bundle = new Bundle();
bundle.putString(“msg”,“處理過后的廣播”);
setResultExtras(bundle);//
} }
MyReceiverThree.java
public class MyReiceiverThree extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent){
//Toast.makeText(context,intent.getStringExtra(“msg”),Toast.LENGTH_SHORT).show();
System.out.println(“接收器3:”+intent.getStringExtra(“msg”));
} }
Config.java
public class Config {
public static final String BC_ONE_ACTION = “com.example.testbroadcasetwo.bcone”;
public static final String BC_TWO_ACTION = “com.example.testbroadcasetwo.bctwo”;
public static final String BC_THREE_ACTION = “com.example.testbroadcasetwo.bcthree”;}
Androidmanifest.xml
package=“com.example.testbroadcasetwo”> //異步廣播需要 一個權限 android:allowBackup=“true” android:icon=“@mipmap/ic_launcher” android:label=“@string/app_name” android:supportsRtl=“true” android:theme=“@style/AppTheme”> String msg = intent.getStringExtra(“msg”); iShowView =(MainActivity)context; if(action.equals(Config.BC_ONE)){ //接收到普通廣播 iShowView.updateText(msg); //回調給HandleBroadcast } } public void setiShowView(IShowView iShowView){ this.iShowView = iShowView; } } IShowView.java 回調到activity更新ui的接口 public interface IShowView { void updateText(String msg);} MainActivity.java publicclassMainActivityextendsAppCompatActivityimplements View.OnClickListener,IShowView{ private Button btOne; private TextView mTvResult; //p層,處理數據 private HandleBroadcast handleBroadcast; @Override protected void onCreate(Bundle savedInstanceState){ super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); handleBroadcast = new HandleBroadcast(this,this); btOne =(Button)findViewById(R.id.bt_one); mTvResult =(TextView)findViewById(R.id.tv_result); btOne.setOnClickListener(this); } @Override public void onClick(View view){ switch(view.getId()){ case R.id.bt_one: handleBroadcast.sendMyBroadcast(TYPE.NORMAL); break; } } /** * 廣播接收處理完畢之后回調更新ui * @param msg 要顯示的文字 */ @Override public void updateText(String msg){ mTvResult.setText(msg); } } activity_main.xml android:layout_width=“match_parent” android:layout_height=“match_parent” android:orientation=“vertical”> android:id=“@+id/bt_one” android:layout_width=“match_parent” android:layout_height=“50dp” android:text=“發送廣播” /> android:id=“@+id/tv_result” android:layout_width=“match_parent” android:layout_height=“60dp” android:text=“結果” android:gravity=“center” /> 如何學習安卓開發?安卓開發學習已經成為IT行業的新潮流。時下,Android也以其創造力、前瞻性、延續性和實現能力成為行業首領,可是怎么學好Android呢?今天,歐柏泰克的老師告訴你如何學好Android。 熟悉Java基礎知識 Android應用的開發語言用的是Java語言,并且在Android中也用到了Java核心類庫的大量的類,因此,在學習Android開發之前,可以先把Java基本語法和Java SE的基礎類庫好好學習一下。Android應用程序開發是以Java語言為基礎的,所以沒有扎實的Java基礎知識,只是機械的照抄別人的代碼,是沒有任何意義的。建議在Android課程前期的Java學習階段中,需要用心的學好。 熟悉一門編程語言 現在大學里面和計算機相關的專業甚至理工類專業一般都會開設C語言課程,只是很多同學在大學期間并沒有好好學習,如果對它掌握的不太好或者很久沒用了,建議先從將其好好復習一下,將其基本的語法再好好回顧一下,最好能搭建一個環境來運行、調試它。如果沒有學過,不妨也提前學習一下。大部分的高校所開設的C語言使用的教材都是用它作為教材,因此無論是購買還是借閱,都容易找到; 熟悉數據結構和算法基礎知識 如果后續有志于游戲方面的開發,最好具備一定的數據結構和算法基礎知識。雖然現代的高級編程語言中,其類庫中已經幫我們實現了大部分的數據結構,一般情況下,我們直接使用即可。但如果能對其原理有所了解,當需要在這些數據結構和算法中間的時候,可以更加的清楚到底應該選擇哪個數據結構或者算法。另外,在圖形圖像處理上面,線性代數的作用也非常重要,如果能掌握一點這方面的基礎知識,無疑也會在后續的學習中如虎添翼。? 大家學習android開發建議首選android開發文檔,該文檔在你下載的sdk中,路徑:/sdk/docs/index.html 目前NFC應用的大的框架上的理解: 我使用的API LEVEL是19,支持的API有三個:android.nfc,android.nfc.cardemulator,android.nfc.tech NFC在手機上的應用大體分為兩類:讀卡器和卡 android.nfc.cardemulator接口是為NFC作為卡應用提供的接口,在較低版本的API上是沒有的 android.nfc.tech,android.nfc接口是為NFC作為讀卡器應用提供的接口 首先說作為卡,nfc有兩種實現方式,一個是使用NFC芯片作為卡,另一個是使用SIM作為卡 Figure 1.NFC card emulation with a secure element.至于從讀卡器發送的指令到底是傳遞到NFC芯片還是SIM由NFC Controler控制,圖中Secure Element是指SIM,Host-CPU指NFC芯片 android提供HostApduService用于NFC芯片,OffHostApduService用于SIM芯片,傳遞方向在res/xml文件中通過AID來控制 ps:Host-Based Card Emulator 簡稱為HCE 代碼實現: AndroidManifest.xml 中 配置service,因為作為卡實現的話,NFC功能是作為service存在的 android:permission=“android.permission.BIND_NFC_SERVICE”> android:resource=“@xml/apduservice”/> res/xml/apduservice.xml 中配置service響應的AID android:requireDeviceUnlock=“false”> android:category=“other”> 配置文件完成后編寫service的處理方法: NFCService需要繼承HostApduService,如果需要與Activity通信,建議采用廣播方式 也可以自己實現觀察者模式,只是這樣就需要持有Activity的引用,感覺不太好 NFCService.java public class NFCService extends HostApduService { private Intent intent = new Intent(“com.example.communication.RECEIVER”); @Override public void onCreate(){ //啟動Acivity Intent i = new Intent(); i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);//需要啟動的Activity不是當前Activity的時候需要用FLAG_ACTIVITY_NEW_TASK i.setAction(“com.apdu.nfc”); getApplication().startActivity(i); Toast.makeText(getApplicationContext(), “Service啟動”, Toast.LENGTH_LONG).show(); } @Override public byte[] processCommandApdu(byte[] commandApdu, Bundle extras){//當注冊的AID被選中后,后續指令被分發到這個處理函數中 byte[] sw = new byte[]{(byte)0x90,(byte)0x00}; byte[] response = new byte[5]; if(commandApdu[0]==(byte)0x00 &&commandApdu[1]==(byte)0xA4&& commandApdu[2]==(byte)0x04 && commandApdu[4]==(byte)0x07&& commandApdu[5]==(byte)0xF0) { return sw; } else { //apdu處理邏輯 switch(commandApdu[1]) { case(byte)0xA8: break; case(byte)0xAE: break; default: return sw; } } intent.putExtra(“command”, commandApdu); intent.putExtra(“response”, response); sendBroadcast(intent);//利用廣播與Activity通信 return response;//SW值需要包含在response中 } @Override public void onDeactivated(int reason){ if(reason==HostApduService.DEACTIVATION_DESELECTED) { Toast.makeText(getApplicationContext(), “已選擇其它應用”, Toast.LENGTH_LONG).show(); } else { Toast.makeText(getApplicationContext(), “連接斷開”, Toast.LENGTH_LONG).show(); } } @Override public void onDestroy() { Toast.makeText(getApplicationContext(), “Service關閉”, Toast.LENGTH_LONG).show();super.onDestroy(); } 框架搭建好剩余的事情就很簡單了,apdu的處理邏輯在processCommandApdu方法中實現即可 以上是Host-CPU方式的實現,SIM方式,API介紹中說該方式沒有提供可供操作的API,也就是說Android不會監聽SIM卡與讀卡器之間的通信 所以NFCOffService 只需要實現onBind接口,這樣綁定該Service的Activity可以對NFCOffService進行有限操作 public class NFCOffService extends OffHostApduService { @Override public IBinder onBind(Intent intent){ // TODO Auto-generated method stub return null;} } 上面沒有提到的就是,如果你需要使用NFC,需要在Manifest中申請NFC權限: android.nfc.tech,android.nfc接口是為NFC作為讀卡器應用提供的接口 接口定義了三種Action Tags:ACTION_NDEF_DISCOVERED,ACTION_TECH_DISCOVERED,ACTION_TAG_DISCOVERED。 當你在Manifest文件中將Activity的action-filter設置為這三個Tag中的一種或幾種時,NFC響應事件會按照如圖流程處理 我的理解是ACTION_NDEF_DISCOVERED 是用于兩臺NFC手機之間傳輸文件的 ACTION_TECH_DISCOVERED,ACTION_TAG_DISCOVERED才是用于NFC與卡進行通訊的 所以開發第一步是在Manifest中配置你的Action: android:resource=“@xml/nfc_tech_filter” /> 當然API中說明你可以將多個tech寫在一個tech-list中,我做了嘗試,這樣做會引出一個問題,在程序未啟動的情況下當手機刷卡時不會自動打開程序 如果想要自動打開需要按照上面這種寫法,tech的個數可以根據你想要支持的卡類型進行調整 配置完成后,可以開始編寫自己的Activity的java代碼了 在onCreate方法中,需要獲取NfcAdapter的引用,從名字可以看出這是一個適配器 NfcAdapter nfcAdapter;PendingIntent pendingIntent;@Override protected void onCreate(Bundle savedInstanceState){ super.onCreate(savedInstanceState); setContentView(R.layout.pos_main); dc =(Button)findViewById(R.id.button4DC); ecc =(Button)findViewById(R.id.button4ECC); qpboc =(Button)findViewById(R.id.button4QPBOC); logWindow=(TextView)findViewById(R.id.communication4Financy); nfcAdapter=NfcAdapter.getDefaultAdapter(this); pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0); onNewIntent(getIntent());} 這里使用PendingIntent,該Intent與普通的Intent不同的是它是有一個延遲啟動的功能,它啟動時會回調onNewIntent函數,這樣能夠實現NFC與Activity的交互 pendingIntent = PendingIntent.getActivity(this, 0, new Intent(this, getClass()).addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP), 0);的含義是將Intent傳遞給this Activity 在onPause與onResume中需要添加代碼 public void onPause(){ super.onPause(); nfcAdapter.disableForegroundDispatch(this);} public void onResume(){ super.onResume(); nfcAdapter.enableForegroundDispatch(this, pendingIntent, FILTERS, TECHLISTS);} enableForegroundDispatch的作用是,當NFC事件發生時如果當前Activity不是注冊了NFC action-filter的Activity,手機會顯示注冊了NFC事件的Activity供用戶選擇 如果當前Activity注冊了NFC action 則將事件優先交由當前Activity處理。onNewIntent實現: @Override public void onNewIntent(Intent intent){ Parcelable p = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); if(p==null) { return; } Tag nfcTag =(Tag)p; final IsoDep isodep = IsoDep.get(nfcTag);// final NfcA isodep = NfcA.get(nfcTag); final byte[] cmd = {(byte)0x00, // CLA Class (byte)0xB4, // INS Instruction (byte)0x04, // P1 Parameter 1 (byte)0x00, // P2 Parameter 2 (byte)0x00, // Le }; try { isodep.connect(); byte[] reaponse=null; logWindow.append(“00B4040000”+'n'); try { reaponse = isodep.getHistoricalBytes(); logWindow.append(Util.bytes2HexString(reaponse)+'n'); reaponse =isodep.transceive(cmd); logWindow.append(Util.bytes2HexString(reaponse)+'n'); } catch(IOException e){ // TODO Auto-generated catch block e.printStackTrace(); } } catch(IOException e1){ // TODO Auto-generated catch block e1.printStackTrace(); }finally{ try { isodep.close(); } catch(IOException e){ // TODO Auto-generated catch block e.printStackTrace(); } } } 首先需要獲取Tag,Tag是操作NFC的基礎 Parcelable p = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); if(p==null) { return; } Tag nfcTag =(Tag)p;再將tag轉換成特定的通訊協議 final IsoDep isodep = IsoDep.get(nfcTag);連接讀卡器: isodep.connect();獲取歷史字節: reaponse = isodep.getHistoricalBytes();響應指令: reaponse =isodep.transceive(cmd);最后斷開連接:(斷開動作應該在Activity關閉或者pause的時候發生)finally{ try { isodep.close(); } catch(IOException e){ // TODO Auto-generated catch block e.printStackTrace(); } } onNewIntent里面可以實現業務邏輯的處理以及UI 本文由麥可網");本地文件存放在:assets文件中 5、如果希望點擊鏈接由自己處理,而不是新開Android的系統browser中響應該鏈接。給WebView添加一個事件監聽對象(WebViewClient) 并重寫其中的一些方法 shouldOverrideUrlLoading:對網頁中超鏈接按鈕的響應。 當按下某個連接時WebViewClient會調用這個方法,并傳遞參數:按下的url onLoadResource onPageStart onPageFinish onReceiveError onReceivedHttpAuthRequest6、如果用webview點鏈接看了很多頁以后,如果不做任何處理,點擊系統“Back”鍵,整個瀏覽器會調用finish()而結束自身,如果希望瀏覽的網頁回退而不是退出瀏覽器,需要在當前Activity中處理并消費掉該Back事件。 覆蓋Activity類的onKeyDown(int keyCoder,KeyEvent event)方法。 public boolean onKeyDown(int keyCoder,KeyEvent event){ if(webView.canGoBack()&& keyCoder == KeyEvent.KEYCODE_BACK){ webview.goBack();//goBack()表示返回webView的上一頁面 return true; } return false; } 上海瑭錦安卓開發心得 對于軟件開發工程師來說學習C語言是一個必經得學習之路,因為他是編程語言中得基礎,如果沒有C語言得編寫能力,那么就談不上是軟件開發工程師,所以學好C語言是成為軟件開發工程師得基礎,那么我們如何了解C語言呢?首先我們要知道C語言是匯編語言中最基礎得編寫語言,它集成了所有匯編語言得大部分優點,能實現低級語言得眾多功能。 C語言把高級語言與低級語言相結合創造出自己得匯編語言風格,塑造出自己得匯編語言工作單元,C語言得結構特點是編寫代碼和相關數據可以分隔化除了程序和相關得必要信息保持彼此獨立得關系,這種關系可以是我們編寫得程序層次化更加清晰,更加便于使用、維護和調試,它會以給用戶方便得調用功能,可以多元化、多種循環、控制編寫語句得控制導向,從而使我們得編寫程序更加結構化、實用化。 這些都可以讓C語言實現決策目得得要求,C語言得應用指針特性使得其應用范圍增加,應用面變大,增加了程序得安全性。不容易被惡意程序所利用,C語言得對其處理標準化、針對于編寫函數得標記、多線程得支持、從新定義新得程序安全定義有了更多得廣泛得選擇。它得編寫靈活和緊湊編寫序列使得書寫程序更加自由化,它可以直接訪問我們常用得物理地址,簡潔得使用操作使得用戶在對硬件上得要求更加簡潔,它可以移植%86得公共代碼在相關得編寫程序上,這一特性使得其在不同環境中運行。它得表達能力使得其在各種系統和相關運行環境下可以自由得發揮它得編寫和使用得能力。第二篇:如何學習安卓開發
第三篇:安卓NFC開發學習筆記
第四篇:安卓應用開發教程WebView總結
第五篇:上海瑭錦安卓開發心得