Adapter源碼簡析及自定義實戰(zhàn)

今天來分享下做導航欄的另外一種方法,導航欄可以放在頂部,也可以放在底部,之前分享過一片底部導航欄的實現(xiàn)方式一行代碼實現(xiàn)底部導航欄TabLayout,用的是Android自帶的控件FragmentTabLayout。今天我們用的一種更為靈活的方式,采用國際慣例Adapter來自定義一個導航欄,可以自己定義每個Tab的布局,可以方便的改變導航欄里面標簽的個數(shù)。
本文會分享到的內(nèi)容:

1.ListView Adapter源碼分析

2.自定義Adapter

3.自定義控件

4.觀察者設計模式

看下Demo:


初始頁.png

點擊添加按鈕可以添加標簽:


添加.png

點擊刪除按鈕可以刪除標簽:

刪除.png

接下來我們正式開車~~~

1.使用

我們先來看下怎么使用,首先看下布局文件,很簡單就是,在底部放置自定義控件TabLinearLayout。

    <Button
        android:text="點擊添加"
        android:id="@+id/add"
        android:layout_centerInParent="true"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <Button
        android:text="點擊刪除"
        android:id="@+id/delete"
        android:layout_below="@id/add"
        android:layout_centerInParent="true"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>

    <com.example.juexingzhe.adapterbottomtab.TabLinearLayout
        android:background="@color/colorPrimary"
        android:layout_width="match_parent"
        android:layout_centerVertical="true"
        android:id="@+id/bottom_tab"
        android:layout_alignParentBottom="true"
        android:layout_height="70dp"/>

然后在Activity中就是構造Adapter傳進去數(shù)據(jù),然后將Adapter設置給tabLinearLayout。


defaultTabAdapter = new DefaultTabAdapter(datas);
tabLinearLayout.setTabAdapter(defaultTabAdapter);

那么增加或者刪除Tab

defaultTabAdapter.notifyChanged();

以上,是不是so easy?接下來我們看看背后的邏輯。

2.ListView Adapter背后邏輯

ListView地球人都知道,平時我們在用的時候基本有下面兩句話:

listView.setAdapter(adapter);
adapter.notifyDataSetChanged();

這兩句話背后發(fā)生了什么使得ListView可以根據(jù)數(shù)據(jù)做出顯示?下面看下源碼,為了看了方便做了些調整。對adapter通過registerDataSetObserver注冊了觀察者mDataSetObserver。

public void setAdapter(ListAdapter adapter) {
        if (mAdapter != null && mDataSetObserver != null) {
            mAdapter.unregisterDataSetObserver(mDataSetObserver);
        }

        // AbsListView#setAdapter will update choice mode states.
        super.setAdapter(adapter);

        if (mAdapter != null) {
            mDataSetObserver = new AdapterDataSetObserver();
            mAdapter.registerDataSetObserver(mDataSetObserver);
}

這個觀察者mDataSetObserver在ListView中找不到,我們到他父類AbsListView中找,定義了一個內(nèi)部類AdapterDataSetObserver。

class AdapterDataSetObserver extends AdapterView<ListAdapter>.AdapterDataSetObserver {
        @Override
        public void onChanged() {
            super.onChanged();
            if (mFastScroll != null) {
                mFastScroll.onSectionsChanged();
            }
        }

        @Override
        public void onInvalidated() {
            super.onInvalidated();
            if (mFastScroll != null) {
                mFastScroll.onSectionsChanged();
            }
        }
    }

setAdapter中的邏輯就先到這,ListView中的源碼還是比較多的,這個不是我們今天的重點。我們今天只要知道setAdapter中做了一件事就是給adapter注冊了一個觀察者,這個就是在數(shù)據(jù)變化的時候adapter可以通知ListView。

我們簡單說下觀察者模式,這個模式可謂是無處不在,主要對象可以簡單分為觀察者和被觀察者。打個比喻,辦公室分成三類人,一類是產(chǎn)品經(jīng)理,一類是前臺秘書,一類是程序猿。上班時間程序猿不想擼代碼,就委托前臺秘書如果產(chǎn)品經(jīng)理過來了就給大家發(fā)個消息,準備好開撕啊。這里程序猿就是觀察者,產(chǎn)品經(jīng)理就是被觀察者,當被觀察者有動靜時,前臺秘書就通知觀察者。

再說個我們常用的點擊事件,Button就是被觀察者,OnClickListener就是觀察者,二者通過setOnClickListener達成關系,view在狀態(tài)變化的時候自動通知(當然這里是Android系統(tǒng)做的工作)觀察者OnClickListener。所以這里Button就對應上面栗子中的產(chǎn)品經(jīng)理,OnClickListener就對應程序猿,setOnClickListener就對應的前臺秘書的作用。

觀察者模式.png

對應到ListView就是在ListView內(nèi)部聲明了一個觀察者AdapterDataSetObserver,在上面setAdapter中AdapterDataSetObserver注冊到adapter中。adapter就起到中介者的作用,在數(shù)據(jù)(被觀察者)變化再通知ListView。

接著看另外一句話adapter.notifyDataSetChanged()的源碼,很簡單一句話,就是調用mDataSetObservablenotifyChanged

private final DataSetObservable mDataSetObservable = new DataSetObservable();

public void registerDataSetObserver(DataSetObserver observer) {
        mDataSetObservable.registerObserver(observer);
    }
public void notifyDataSetChanged() {
        mDataSetObservable.notifyChanged();
}

接著跟到mDataSetObservablenotifyChanged中,就是調用觀察者的onChanged方法。

public class DataSetObservable extends Observable<DataSetObserver> {
    public void notifyChanged() {
        synchronized(mObservers) {
            for (int i = mObservers.size() - 1; i >= 0; i--) {
                mObservers.get(i).onChanged();
            }
        }
    }

對于ListView,觀察者就是AdapterDataSetObserver,這個是在AbdListView中,super就是AdapterView

public void onChanged() {
            super.onChanged();
}

我們進去看到了很熟悉的一句話requestLayout,就是進行ListView的重繪。

class AdapterDataSetObserver extends DataSetObserver {
        @Override
        public void onChanged() {
            ……
            requestLayout();
        }
}

到這里就恍然大悟了,就是在ListView中注冊觀察者,要實現(xiàn)抽象類DataSetObserver

public abstract class DataSetObserver {
    public void onChanged() {
        // Do nothing
    }

    public void onInvalidated() {
        // Do nothing
    }
}

在adapter中含有一個DataSetObservable,類似于中介作用,在數(shù)據(jù)變化時通知觀察者,也就是DataSetObserver,然后onChanged就會被回調了。

public class DataSetObservable extends Observable<DataSetObserver> {
    public void notifyChanged() {
        synchronized(mObservers) {
            for (int i = mObservers.size() - 1; i >= 0; i--) {
                mObservers.get(i).onChanged();
            }
        }
    }
}

3.自定義Adapter

前面源碼分析有木有很枯燥?我們接著來點實戰(zhàn)提提精神。這里adapter我們就不包含,直接繼承DataSetObservable。然后擴展三個方法,都比較簡單,見名知意就不多說了。

public abstract class TabAdapter<T> extends DataSetObservable {
    public abstract int getCount();

    public abstract View getView(ViewGroup parent, View convertView, int position);

    public abstract T getItem(int position);
}

為了方便用戶使用,我們實現(xiàn)一個默認的TabAdapter.為了內(nèi)存優(yōu)化,這里也是使用了ViewHolder進行重用。布局也很簡單就是上面一個ImageView,底下是一個TextView。

public class DefaultTabAdapter extends TabAdapter {
    private List<TabBean> mData = new ArrayList<>();
    private ViewHolder viewHolder;

    DefaultTabAdapter(ArrayList<TabBean> data) {
        mData = data;
    }

    @Override
    public int getCount() {
        return mData.size();
    }

    @Override
    public View getView(ViewGroup parent, View convertView, final int position) {

        if (convertView == null){
            viewHolder = new ViewHolder();
            LayoutInflater inflater = LayoutInflater.from(parent.getContext());
            convertView =  inflater.inflate(R.layout.tab_item, parent, false);
            viewHolder.imageView = (ImageView) convertView.findViewById(R.id.tab_img);
            viewHolder.textView = (TextView) convertView.findViewById(R.id.tab_txt);
            convertView.setTag(viewHolder);
        }else {
            viewHolder = (ViewHolder) convertView.getTag();
        }

        viewHolder.imageView.setBackgroundResource(mData.get(position).tabImgSourceUnSelect);
        viewHolder.textView.setText(mData.get(position).tabTxt);
        LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.MATCH_PARENT, 1);
        convertView.setLayoutParams(layoutParams);

        return convertView;
    }

    @Override
    public TabBean getItem(int position) {
        return mData.get(position);
    }

    private static class ViewHolder{
        ImageView imageView;
        TextView textView;
    }
}

4.自定義TabLinearLayout

我們先看下adapter的觀察者模式的邏輯實現(xiàn)。
1.首先聲明一個觀察者,實現(xiàn)DataSetObserver的兩個方法。

private DataSetObserver mTabDataSetObserver = new DataSetObserver() {
        @Override
        public void onChanged() {
            super.onChanged();
            tabOnChanged();
        }

        @Override
        public void onInvalidated() {
            super.onInvalidated();
            removeAllViews();
        }
};

2.接著,在setTabAdapter中將觀察者mTabDataSetObserver注冊到mTabAdapter中。

public void setTabAdapter(TabAdapter tabAdapter) {
        this.mTabAdapter = tabAdapter;
        removeAllViews();
        mTabAdapter.registerObserver(mTabDataSetObserver);
        mTabAdapter.notifyChanged();
}

3.最后,在用戶調用defaultTabAdapter.notifyChanged();后會調用我們觀察者的onChanged方法,我們在里面實現(xiàn)更新。

以上就是adapter的觀察者邏輯,和Android源碼略微不一樣的地方就是我們這里adapter就是一個DataSetObservable,源碼中是adapter中包含了一個DataSetObservable,就這樣。

從上面可以看出,增加和刪除的邏輯主要是在tabOnChanged()方法中。首先從adapter中getView獲得布局,然后動態(tài)添加到LinearLayour中。

private void tabOnChanged() {
        removeAllViews();
        mContainer.clear();
        int count = mTabAdapter.getCount();

        for (int i = 0; i < count; i++) {
            LinearLayout layout = (LinearLayout) mTabAdapter.getView(this, null, i);
            final int finalI = i;
            layout.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
                    notifyClickEvent(finalI);
                }
            });
            addView(layout);
            mContainer.add(layout);
        }
        mContainer.get(0).getChildAt(0).setBackgroundResource(mTabAdapter.getItem(0).tabImgSourceSelected);
}

同時也將布局文件添加到數(shù)組中,這個是為了點擊事件的處理,在點擊其中一個Tab時需要更新剩下的Tab。

private void notifyClickEvent(int finalI) {
        for (int i = 0; i < mContainer.size(); i++) {
            if (i == finalI) {
                mContainer.get(i).getChildAt(0).setBackgroundResource(mTabAdapter.getItem(i).tabImgSourceSelected);
                continue;
            }
            mContainer.get(i).getChildAt(0).setBackgroundResource(mTabAdapter.getItem(i).tabImgSourceUnSelect);
        }
}

以上就是自定義TabLinearLayout的內(nèi)容了。

5.總結

我們通過分析Android源碼的Adapter實現(xiàn)原理,結合觀察者設計模式的講解,應該是比較清晰的。學習并實踐,動手實現(xiàn)了一個Adapter,用來充當被觀察者,自定義一個TabLinearLayout導航欄作為觀察者,從而實現(xiàn)動態(tài)添加Tab和刪除Tab。

到這里我們的實現(xiàn)導航欄的第二種方式已經(jīng)分享完畢,大家可以下車嘍,希望我有說清楚讓大家有點收獲。

感謝@右傾傾的支持與理解!

你們的贊是我最大的動力,謝謝!

歡迎關注公眾號:JueCode

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,443評論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,530評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 176,407評論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,981評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,759評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,204評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,263評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,415評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,955評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,782評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,983評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,222評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,650評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,892評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,675評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,967評論 2 374

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,658評論 25 708
  • 這是我,見過第二次神奇的數(shù)字。一個小時過的很慢,一分鐘過的很快,日常生活里,我們錯過太多的歲月。 想要提高做某...
    沐星之星星閱讀 286評論 0 0
  • 我與朱自清初識,是在中學課本,就是借助初中時的《春》、高中時的《荷塘月色》和《綠》三篇經(jīng)典課文,以及后來讀到的《匆...
    阿YAO閱讀 4,180評論 10 14
  • socket服務器可通過python來搭建 源程序分享在百度云,下載鏈接如下:鏈接: http://pan.bai...
    dibadalu閱讀 6,064評論 12 12