前言
如果你不了解位運算,那你應該要補習一下功課,這里就不對位移運算做補習,因為今天用到的是與、或、非在二進制運算的巧用
復習
1、&運算
有兩個操作數a、b,a上的第x位和b上的第x位都是1,那x也是1,否則x為0
a:
0000 0001 0000 1010 0000 0000 0000 0110
a:
0000 0001 0000 1000 0000 0110 0100 0000
a&b
0000 0001 0000 1000 0000 0000 0000 00002、|運算
有兩個操作數a、b,a上的第x位和b上的第x位只要有一個是1,那x是1,否則x為0
a:
0000 0001 0000 1010 0000 0000 0000 0110
a:
0000 0001 0000 1000 0000 0110 0100 0000
a|b
0000 0001 0000 1000 0000 0110 0100 01103、^異或
有兩個操作數a、b,a上的第x位和b上的第x位不同,那x是1,否則x為0
a:
0000 0001 0000 1010 0000 0000 0000 0110
a:
0000 0001 0000 1000 0000 0110 0100 0000
a^b
0000 0000 0000 0010 0000 0110 0100 01103、~運算
有一個操作數a,a的第x位上的數為1,那么結果的第x位則為0,否則為1
a:
0000 0001 0000 1010 0000 0000 0000 0110
~a:
1111 1110 1111 0101 1111 1111 1111 1001
應用場景
情景1:我們在啟動頁往往會有一個啟動動畫,一般都是啟動動畫結束后才會進入Activity,而如果這個時候恰好在啟動頁需要啟動一個網絡請求,同時滿足動畫結束以及網路請求完成(可能是成功回調也可能是失敗回調)才能進入HomeActivity,那這個時候一般我們會用兩個boolean類型的值去判斷不同的地方是否完成,在條件不多的時候還好,如果條件增加了,那這種利用boolean值去判斷就顯得非常凌亂
情景2:當你需要一系列流程的執行的時候我們往往會使用責任鏈模式去解決這個問題,如果中途出現問題了我們會把出問題的步驟返回出去,那對于不同的環節我們都可能對其進行做一個標記,這樣在具有耗時環節的場景下,我們可以通過標記符來判斷這個環節是否已經走過了,可以避免二次執行同一個耗時操作,那這個時候我們如何給每個環節設置一個標記呢
以上情景均適用于用位運算給一個數做多種標記
在簡物中有一個應用場景是這樣的,注冊環節中要對手機、驗證碼、密碼進行逐個判斷,中途還可以修改驗證碼,需要二次判斷,并且每一步執行合法后是通過啟動一個交互動畫來提供下一個參數的設置入口,那這里就涉及到判斷每個參數的合法性問題了,因為在最后注冊的步驟,我需要判斷每一步是否均合法,當然我們可以通過好幾個if去判斷每一步流程,每次點擊注冊都走一遍每個if判斷,但是這樣顯得非常啰嗦,明明已經判斷過的了,為什么每次點擊注冊都要判斷條件呢,是不是可以給判斷過的步驟加上標記符,然后每次點擊注冊只要判斷每個標記符是否在當前步驟變量上不就好了?
第一步,給每個參數步驟設置一個操作數
/**
* 設置手機號碼
*/
public static final int SET_MOBILE = 0x00010;
/**
* 設置驗證碼
*/
public static final int SET_SMS_CODE = 0x00020;
/**
* 設置密碼
*/
public static final int SET_PASSWORD = 0x00040;
/**
* 注冊
*/
public static final int REGISTER = 0x00080;
建一個當前步驟的變量值,用于和參數步驟進行位運算
int mRegisterState;
首次初始化,將步驟移到設置手機號碼
mRegisterState |= RegisterState.SET_MOBILE;
那么這個時候你可以開始輸入手機號碼,輸完了之后點擊下一步,會調用網絡請求判斷手機號是否合法,如果不合法,在當前界面反饋給用戶,如果合法,將當前操作步驟移到填寫驗證碼
/**
* 用戶可注冊
* @param message
*/
@Override
public void userUnavailable(String message) {
mRegisterState |= RegisterState.SET_SMS_CODE;
}
然后填寫驗證碼,如果驗證碼符合條件(如6位數字),則把操作步驟移到設置密碼
mRegisterState |= RegisterState.SET_PASSWORD;
設置的密碼合法,那就可以移動到注冊步驟了
mRegisterState |= RegisterState.REGISTER;
那么是如何判斷當前步驟可以執行呢,大概類似以下方法
if((mRegisterState & RegisterState.REGISTER) == RegisterState.REGISTER){
//注冊
}
剛剛說了,如果我在添加參數過程中修改了前面的操作,比如驗證碼改成了5位(驗證碼不合法了),那我如何撤銷步驟呢,很簡單
public void onTextChanged(int id){
switch (id){
case R.id.sms_code:
/**
* 驗證碼不合法
*/
if(isEmpty(getText(mSmsCode)) || isSmsCodeIllegal()){
/**
* 取消設置密碼步驟
*/
mRegisterState &= ~RegisterState.SET_PASSWORD;
}
....
break;
}
}
對就是通過以下代碼來撤銷標記
mRegisterState &= ~RegisterState.SET_PASSWORD;
可能這么寫不是很明了,我寫一個java小函數來給大家演示一下,有興趣的同學可以拷貝運行一下
public class Register {
/**
* 設置手機號碼
*/
public static final int SET_MOBILE = 0x00010;
/**
* 設置驗證碼
*/
public static final int SET_SMS_CODE = 0x00020;
/**
* 設置密碼
*/
public static final int SET_PASSWORD = 0x00040;
/**
* 注冊
*/
public static final int REGISTER = 0x00080;
public static void main(String[] args) {
int state = 0;
/**
* 設置手機號
*/
state |= SET_MOBILE;
/**
* 手機號合法了,可以設置驗證碼了
*/
state |= SET_SMS_CODE;
/**
* 驗證碼合法,可以設置密碼
*/
state |= SET_PASSWORD;
/**
* 密碼合法,可以注冊了
*/
state |= REGISTER;
if((state & SET_MOBILE) == SET_MOBILE){
System.out.println("手機號設置合法!");
}else {
System.out.println("請設置手機號");
return;
}
if((state & SET_SMS_CODE) == SET_SMS_CODE){
System.out.println("驗證碼設置合法!");
}else {
System.out.println("驗證碼不合法");
return;
}
if((state & SET_PASSWORD) == SET_PASSWORD){
System.out.println("密碼設置合法!");
}else {
System.out.println("密碼不合法");
return;
}
if((state & REGISTER) == REGISTER){
System.out.println("注冊成功");
}else {
System.out.println("參數不合法");
return;
}
}
}
輸出結果是
手機號設置合法!
驗證碼設置合法!
密碼設置合法!
注冊成功
那假如注冊途中修改了驗證碼,導致驗證碼不合法,執行結果會怎樣呢?注意看一下注釋
public static void main(String[] args) {
int state = 0;
/**
* 設置手機號
*/
state |= SET_MOBILE;
/**
* 手機號合法了,可以設置驗證碼了
*/
state |= SET_SMS_CODE;
/**
* 驗證碼合法,可以設置密碼
*/
state |= SET_PASSWORD;
/**
* 密碼合法,可以注冊了
*/
state |= REGISTER;
/**
* 用戶又修改了輸入的驗證碼,導致驗證碼不合法
*/
state &= ~SET_SMS_CODE;
if((state & SET_MOBILE) == SET_MOBILE){
System.out.println("手機號設置合法!");
}else {
System.out.println("請設置手機號");
return;
}
if((state & SET_SMS_CODE) == SET_SMS_CODE){
System.out.println("驗證碼設置合法!");
}else {
System.out.println("驗證碼不合法");
return;
}
if((state & SET_PASSWORD) == SET_PASSWORD){
System.out.println("密碼設置合法!");
}else {
System.out.println("密碼不合法");
return;
}
if((state & REGISTER) == REGISTER){
System.out.println("注冊成功");
}else {
System.out.println("參數不合法");
return;
}
}
運行結果
手機號設置合法!
驗證碼不合法
這樣可能就比較好理解了,當然具體這個環節是如何判斷參數的合法性,并不是這么簡單,簡物里面用到責任鏈模式,可以移步到這篇文章看看Jianwoo中的設計模式(7) — 責任鏈模式
以上就是使用位運算來給一個變量做多種標記的小技巧,如果喜歡請不要吝嗇給個Like!