Android开发(2) | 广播 Broadcast 的应用——强制下线功能
文章目錄
- 功能簡介
- 關(guān)閉所有活動(dòng)
- 登陸界面
- 發(fā)送強(qiáng)制下線的廣播
- 廣播接收器
- AndroidManifest.xml
- 運(yùn)行結(jié)果
功能簡介
強(qiáng)制下線功能只需要彈出一個(gè)對(duì)話框,讓用戶只能點(diǎn)擊確定按鈕,回到登錄界面。
如果在每一個(gè)活動(dòng)中添加一個(gè)對(duì)話框的話太過繁瑣,用廣播實(shí)現(xiàn)是一個(gè)好辦法。下面逐步進(jìn)行實(shí)現(xiàn):
關(guān)閉所有活動(dòng)
強(qiáng)制下線功能需要先關(guān)閉所有的活動(dòng),我們只需要用 AcitivityCollector 類來管理所有的活動(dòng),然后用 BaseActivity 類作為所有活動(dòng)的父類,如此一來即可通過 ActivityCollector.finishAll() 退出所有程序。
AcitivityCollector.java 和 BaseActivity.java 我們?cè)谠撈┛椭袑?shí)現(xiàn)過(生產(chǎn)環(huán)境中關(guān)于 Activity 的小技巧),這里直接拿過來用:
public class ActivityCollector {public static List<Activity> activities = new ArrayList<>();public static void addActivity(Activity activity){activities.add(activity);}public static void removeActivity(Activity activity) {activities.remove(activity);}public static void finishAll(){for(Activity activity : activities){if(!activity.isFinishing()){activity.finish();}}} } public class BaseActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityCollector.addActivity(this);}@Overrideprotected void onDestroy() {super.onDestroy();ActivityCollector.removeActivity(this);} }登陸界面
首先我們創(chuàng)建一個(gè) LoginActivity 作為登錄界面。
布局文件 login_layout.xml:
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><LinearLayoutandroid:orientation="horizontal"android:layout_width="match_parent"android:layout_height="60dp"><TextViewandroid:layout_width="90dp"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:textSize="18sp"android:text="Account:"/><EditTextandroid:id="@+id/account"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:hint="Please input a account"android:layout_gravity="center_vertical"/></LinearLayout><LinearLayoutandroid:orientation="horizontal"android:layout_width="match_parent"android:layout_height="60dp"><TextViewandroid:layout_width="90dp"android:layout_height="wrap_content"android:layout_gravity="center_vertical"android:textSize="18sp"android:text="Password:"/><EditTextandroid:id="@+id/password"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:layout_gravity="center_vertical"android:hint="Please input a password"android:inputType="textPassword" /></LinearLayout><Buttonandroid:id="@+id/button_login"android:layout_width="match_parent"android:layout_height="60dp"android:text="login"/></LinearLayout>- android:inputType :讓輸入的內(nèi)容變成我們熟知的小圓點(diǎn)形式(見下圖
運(yùn)行結(jié)果:
活動(dòng)文件 LoginActivity.java:
// 繼承自自定義的父類BaseActivity public class LoginActivity extends BaseActivity {private EditText accountEdit;private EditText passwordEdit;private Button login;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.login_layout);accountEdit = findViewById(R.id.account);passwordEdit = findViewById(R.id.password);login = findViewById(R.id.button_login);login.setOnClickListener((View v)->{String account = accountEdit.getText().toString();String password = passwordEdit.getText().toString();if(account.equals("cmy") && password.equals("201314")){Intent intent = new Intent(LoginActivity.this, ForceQuitActivity.class);startActivity(intent);finish();}else{if (account.isEmpty() || password.isEmpty()){Toast.makeText(this, "賬號(hào)或密碼不不能為空", Toast.LENGTH_LONG).show();}Toast.makeText(this, "賬號(hào)或密碼不正確", Toast.LENGTH_LONG).show();}});} }發(fā)送強(qiáng)制下線的廣播
登陸界面賬號(hào)密碼正確后,跳轉(zhuǎn)到 ForceQuitActivity.java 活動(dòng),在 onCreate 方法中初始化發(fā)送廣播的按鈕,并實(shí)現(xiàn)發(fā)送廣播的過程:
public class ForceQuitActivity extends BaseActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.force_quit_layout);Button forceOffline = findViewById(R.id.force_offline);forceOffline.setOnClickListener((View v)->{// 將要發(fā)送的廣播植入IntentIntent intent = new Intent("com.example.activitytest.Activity.FORCE_OFFLINE");sendBroadcast(intent);});} }布局文件 force_quit_layout.xml:
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><Buttonandroid:id="@+id/force_offline"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="send force offline broadcast"/> </LinearLayout>發(fā)送廣播的按鈕:
廣播接收器
接下來就應(yīng)該創(chuàng)建一個(gè)廣播接收器來接收強(qiáng)制下線廣播了,問題是應(yīng)該創(chuàng)建在哪里?
- 靜態(tài)注冊(cè)的廣播接收器無法在 onReceive() 方法中彈出對(duì)話框這種 UI 控件。
- 也不可能在每個(gè)活動(dòng)中都注冊(cè)一個(gè)動(dòng)態(tài)的廣播接收器。
因此可以在 BaseActivity 中動(dòng)態(tài)注冊(cè)一個(gè)廣播接收器,供其他子類調(diào)用:
public class BaseActivity extends AppCompatActivity {private ForceOfflineReceiver receiver;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityCollector.addActivity(this);}// 活動(dòng)位于棧頂且準(zhǔn)備好與用戶交互@Overrideprotected void onResume() {super.onResume();IntentFilter intentFilter = new IntentFilter();intentFilter.addAction("com.example.activitytest.Activity.FORCE_OFFLINE");receiver = new ForceOfflineReceiver();// 動(dòng)態(tài)注冊(cè)registerReceiver(receiver, intentFilter);}// 在系統(tǒng)準(zhǔn)備去啟動(dòng)或者恢復(fù)另一個(gè)活動(dòng)時(shí)調(diào)用,通常會(huì)釋放一些占用 CPU 的資源,保存一些關(guān)鍵數(shù)據(jù);@Overrideprotected void onPause() {super.onPause();// 動(dòng)態(tài)注冊(cè)要在結(jié)束時(shí)取消注冊(cè)// 本程序中點(diǎn)擊提示框中的OK按鈕后會(huì)跳轉(zhuǎn)到登陸界面,此時(shí)就是強(qiáng)制下線廣播職責(zé)結(jié)束的時(shí)候if(receiver != null){unregisterReceiver(receiver);receiver = null;}}@Overrideprotected void onDestroy() {super.onDestroy();ActivityCollector.removeActivity(this);}class ForceOfflineReceiver extends BroadcastReceiver{@Overridepublic void onReceive(final Context context, Intent intent) {AlertDialog.Builder dialog = new AlertDialog.Builder(context);dialog.setTitle("Warning");dialog.setMessage("You are forced to be offline. Please try to login again.");dialog.setCancelable(false);dialog.setPositiveButton("OK", (DialogInterface dialogInterface, int which)->{ActivityCollector.finishAll(); // 銷毀所有活動(dòng)// 重新啟動(dòng)LoginActivityIntent intent1 = new Intent(context, LoginActivity.class);context.startActivity(intent1);});dialog.show();}} }以往注冊(cè)和取消注冊(cè)廣播接收器都是在 onCreate() 和 onDestroy() 里進(jìn)行的,而這里我們?cè)?onResume() 和 onPause() 中注冊(cè)/取消注冊(cè),這是因?yàn)槲覀冃枰WC只有處于棧頂?shù)幕顒?dòng)才能收到這條廣播,當(dāng)活動(dòng)失去棧頂位置時(shí)應(yīng)該取消廣播接收器的注冊(cè)。
AndroidManifest.xml
將登錄界面 LoginActivity 設(shè)為程序主界面。
運(yùn)行結(jié)果
啟動(dòng)程序,展示登陸界面:
賬號(hào)密碼正確,跳轉(zhuǎn)到發(fā)送廣播界面:
點(diǎn)擊按鈕發(fā)送廣播,彈出強(qiáng)制下線對(duì)話框:
點(diǎn)擊OK,返回登陸界面:
總結(jié)
以上是生活随笔為你收集整理的Android开发(2) | 广播 Broadcast 的应用——强制下线功能的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 「重点」此去经年一别两宽什么意思
- 下一篇: 左脑型思维者(你是左脑思维还是右脑思维的