第一篇:[安卓開發(fā)] Broadcast 三種廣播的使用總結
[安卓開發(fā)] Broadcast 三種廣播的使用
總結
1、什么是安卓的Broadcast?
安卓的四大組件之一,是一種廣泛應用在應用程序之間傳輸信息的機制。
2、什么是安卓的BroadcastReceiver?
是對發(fā)送出來的廣播進行過濾接收并響應的一類組件,它就是用來接收來自系統(tǒng)和應用中的廣播。例如系統(tǒng)的廣播有開機廣播: 系統(tǒng)在開機時候會發(fā)送開機廣播,程序接收到之后,能進行開機自啟動。網(wǎng)絡狀態(tài)改變廣播: 3g變wifi、網(wǎng)絡斷開等。電量改變廣播等等。。
3、Anroid為什么要這樣設計?
大大減少開發(fā)工作量和開發(fā)周期
作為開發(fā)者,只需要掌握BroadcastReceiver
4、怎么理解Broadcast和BroadcastReceiver ?
Broadcast就像現(xiàn)實中的廣播電臺,他發(fā)廣播信號來,然后我們用收音機來接收,然后處理,并且播放出聲音,BroadcastReceiver就相當于那臺收音機。
5、使用方法
發(fā)送:
把信息裝入一個Intent對象(如:Action、Category),通過調相應的方法將Intent對象以廣播的方式發(fā)送出去: sendBroadcast();
sendOrederBroadcast();sendStickyBroadcast();接收:
當Intent發(fā)送之后,所有已經(jīng)注冊receivedBroadcastReceiver會檢查注冊時的IntentFilter是否與發(fā)送的Intent相匹配,若匹配則就會調用BroadcastReceiver的onReceiver()方法。所以當我們定義一個BroadcastReceiver的時候,都需要實現(xiàn)onReceiver()方法。注意:
BroadcastReceiver需要注冊
靜態(tài)注冊
代碼動態(tài)注冊
6、注意??!
BroadReceiver生命周期只有十秒左右,不能直接執(zhí)行耗時操作,不然會出現(xiàn)ANR(應用程序無響應),也不能用子線程來做,因為每次廣播來的時候都會創(chuàng)建一個Reveiver對象,并且調用onReceiver,執(zhí)行完之后,對象會立刻被銷毀,子線程也沒了
要做耗時操作的話,應該通過發(fā)送Intent給Service,由Service來完成。
動態(tài)注冊廣播接受者的話要在Destory回調事件進行unregister
7、廣播的分類
普通廣播(Normal broadcast)所有監(jiān)聽該廣播接受者都可以監(jiān)聽到該廣播 同級別接收先后順序是隨機的(無序)級別低的后收到廣播
接收器不能截斷廣播的繼續(xù)傳播,也不能處理廣播 同級別動態(tài)注冊高于靜態(tài)注冊 有序廣播(Oredered broadcast)按照接收者的優(yōu)先順序來接收廣播,優(yōu)先級別在intent-filter中的priority中聲明,-1000到1000之間,值越大優(yōu)先級越高,可以終止廣播的繼續(xù)傳播,接受者可以修改intent的內(nèi)容。同級別接收順序是隨機的 級別低的后收到
能截斷廣播的繼續(xù)傳播,高級別的廣播接收器接收廣播后能決定時候截斷。能處理廣播
同級別動態(tài)注冊高于靜態(tài)注冊
異步廣播(粘滯性滯留廣播)ps:已被棄用
不能處理結果給下一個接收者,無法終止廣播。一直存在
可以先發(fā)送廣播,再注冊接收器
需要在清單文件添加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);
//動態(tài)注冊,在當前activity的生命周期內(nèi)運行
/*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:
//發(fā)送普通廣播
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);//其中第二個參數(shù)是設置權限,即接收器必須具有相應的權限才能正常接收到廣播。
break;
case R.id.bt_three:
//異步廣播
intent.setAction(Config.BC_THREE_ACTION);
intent.putExtra(“msg”,“這是異步廣播”);
sendStickyBroadcast(intent);
//可以先發(fā)送 后注冊
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代表如果前面的接收器沒有存放數(shù)據(jù),則自動創(chuàng)建一個空的Bundle對象,false則表示如果前面的接收器如果沒有存放任何數(shù)據(jù)則返回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();
//截斷廣播,不讓別的接收器繼續(xù)接收,有序廣播才能成功攔截
//處理廣播
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層,處理數(shù)據(jù) 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=“發(fā)送廣播” /> android:id=“@+id/tv_result” android:layout_width=“match_parent” android:layout_height=“60dp” android:text=“結果” android:gravity=“center” /> 如何學習安卓開發(fā)?安卓開發(fā)學習已經(jīng)成為IT行業(yè)的新潮流。時下,Android也以其創(chuàng)造力、前瞻性、延續(xù)性和實現(xiàn)能力成為行業(yè)首領,可是怎么學好Android呢?今天,歐柏泰克的老師告訴你如何學好Android。 熟悉Java基礎知識 Android應用的開發(fā)語言用的是Java語言,并且在Android中也用到了Java核心類庫的大量的類,因此,在學習Android開發(fā)之前,可以先把Java基本語法和Java SE的基礎類庫好好學習一下。Android應用程序開發(fā)是以Java語言為基礎的,所以沒有扎實的Java基礎知識,只是機械的照抄別人的代碼,是沒有任何意義的。建議在Android課程前期的Java學習階段中,需要用心的學好。 熟悉一門編程語言 現(xiàn)在大學里面和計算機相關的專業(yè)甚至理工類專業(yè)一般都會開設C語言課程,只是很多同學在大學期間并沒有好好學習,如果對它掌握的不太好或者很久沒用了,建議先從將其好好復習一下,將其基本的語法再好好回顧一下,最好能搭建一個環(huán)境來運行、調試它。如果沒有學過,不妨也提前學習一下。大部分的高校所開設的C語言使用的教材都是用它作為教材,因此無論是購買還是借閱,都容易找到; 熟悉數(shù)據(jù)結構和算法基礎知識 如果后續(xù)有志于游戲方面的開發(fā),最好具備一定的數(shù)據(jù)結構和算法基礎知識。雖然現(xiàn)代的高級編程語言中,其類庫中已經(jīng)幫我們實現(xiàn)了大部分的數(shù)據(jù)結構,一般情況下,我們直接使用即可。但如果能對其原理有所了解,當需要在這些數(shù)據(jù)結構和算法中間的時候,可以更加的清楚到底應該選擇哪個數(shù)據(jù)結構或者算法。另外,在圖形圖像處理上面,線性代數(shù)的作用也非常重要,如果能掌握一點這方面的基礎知識,無疑也會在后續(xù)的學習中如虎添翼。? 大家學習android開發(fā)建議首選android開發(fā)文檔,該文檔在你下載的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有兩種實現(xiàn)方式,一個是使用NFC芯片作為卡,另一個是使用SIM作為卡 Figure 1.NFC card emulation with a secure element.至于從讀卡器發(fā)送的指令到底是傳遞到NFC芯片還是SIM由NFC Controler控制,圖中Secure Element是指SIM,Host-CPU指NFC芯片 android提供HostApduService用于NFC芯片,OffHostApduService用于SIM芯片,傳遞方向在res/xml文件中通過AID來控制 ps:Host-Based Card Emulator 簡稱為HCE 代碼實現(xiàn): AndroidManifest.xml 中 配置service,因為作為卡實現(xiàn)的話,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通信,建議采用廣播方式 也可以自己實現(xiàn)觀察者模式,只是這樣就需要持有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被選中后,后續(xù)指令被分發(fā)到這個處理函數(shù)中 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方法中實現(xiàn)即可 以上是Host-CPU方式的實現(xiàn),SIM方式,API介紹中說該方式?jīng)]有提供可供操作的API,也就是說Android不會監(jiān)聽SIM卡與讀卡器之間的通信 所以NFCOffService 只需要實現(xiàn)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與卡進行通訊的 所以開發(fā)第一步是在Manifest中配置你的Action: android:resource=“@xml/nfc_tech_filter” /> 當然API中說明你可以將多個tech寫在一個tech-list中,我做了嘗試,這樣做會引出一個問題,在程序未啟動的情況下當手機刷卡時不會自動打開程序 如果想要自動打開需要按照上面這種寫法,tech的個數(shù)可以根據(jù)你想要支持的卡類型進行調整 配置完成后,可以開始編寫自己的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函數(shù),這樣能夠實現(xiàn)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事件發(fā)生時如果當前Activity不是注冊了NFC action-filter的Activity,手機會顯示注冊了NFC事件的Activity供用戶選擇 如果當前Activity注冊了NFC action 則將事件優(yōu)先交由當前Activity處理。onNewIntent實現(xiàn): @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轉換成特定的通訊協(xié)議 final IsoDep isodep = IsoDep.get(nfcTag);連接讀卡器: isodep.connect();獲取歷史字節(jié): reaponse = isodep.getHistoricalBytes();響應指令: reaponse =isodep.transceive(cmd);最后斷開連接:(斷開動作應該在Activity關閉或者pause的時候發(fā)生)finally{ try { isodep.close(); } catch(IOException e){ // TODO Auto-generated catch block e.printStackTrace(); } } onNewIntent里面可以實現(xiàn)業(yè)務邏輯的處理以及UI 本文由麥可網(wǎng)");本地文件存放在:assets文件中 5、如果希望點擊鏈接由自己處理,而不是新開Android的系統(tǒng)browser中響應該鏈接。給WebView添加一個事件監(jiān)聽對象(WebViewClient) 并重寫其中的一些方法 shouldOverrideUrlLoading:對網(wǎng)頁中超鏈接按鈕的響應。 當按下某個連接時WebViewClient會調用這個方法,并傳遞參數(shù):按下的url onLoadResource onPageStart onPageFinish onReceiveError onReceivedHttpAuthRequest6、如果用webview點鏈接看了很多頁以后,如果不做任何處理,點擊系統(tǒng)“Back”鍵,整個瀏覽器會調用finish()而結束自身,如果希望瀏覽的網(wǎng)頁回退而不是退出瀏覽器,需要在當前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; } 上海瑭錦安卓開發(fā)心得 對于軟件開發(fā)工程師來說學習C語言是一個必經(jīng)得學習之路,因為他是編程語言中得基礎,如果沒有C語言得編寫能力,那么就談不上是軟件開發(fā)工程師,所以學好C語言是成為軟件開發(fā)工程師得基礎,那么我們?nèi)绾瘟私釩語言呢?首先我們要知道C語言是匯編語言中最基礎得編寫語言,它集成了所有匯編語言得大部分優(yōu)點,能實現(xiàn)低級語言得眾多功能。 C語言把高級語言與低級語言相結合創(chuàng)造出自己得匯編語言風格,塑造出自己得匯編語言工作單元,C語言得結構特點是編寫代碼和相關數(shù)據(jù)可以分隔化除了程序和相關得必要信息保持彼此獨立得關系,這種關系可以是我們編寫得程序層次化更加清晰,更加便于使用、維護和調試,它會以給用戶方便得調用功能,可以多元化、多種循環(huán)、控制編寫語句得控制導向,從而使我們得編寫程序更加結構化、實用化。 這些都可以讓C語言實現(xiàn)決策目得得要求,C語言得應用指針特性使得其應用范圍增加,應用面變大,增加了程序得安全性。不容易被惡意程序所利用,C語言得對其處理標準化、針對于編寫函數(shù)得標記、多線程得支持、從新定義新得程序安全定義有了更多得廣泛得選擇。它得編寫靈活和緊湊編寫序列使得書寫程序更加自由化,它可以直接訪問我們常用得物理地址,簡潔得使用操作使得用戶在對硬件上得要求更加簡潔,它可以移植%86得公共代碼在相關得編寫程序上,這一特性使得其在不同環(huán)境中運行。它得表達能力使得其在各種系統(tǒng)和相關運行環(huán)境下可以自由得發(fā)揮它得編寫和使用得能力。第二篇:如何學習安卓開發(fā)
第三篇:安卓NFC開發(fā)學習筆記
第四篇:安卓應用開發(fā)教程WebView總結
第五篇:上海瑭錦安卓開發(fā)心得