av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

您的位置:首頁技術(shù)文章
文章詳情頁

Android Filterable實現(xiàn)Recyclerview篩選功能的示例代碼

瀏覽:7日期:2022-09-20 18:44:48

原先碰到篩選這種功能時,后端的接口都會讓上傳一個字段,根據(jù)字段來返回相應(yīng)的數(shù)據(jù)。后來一次和別人對接時,接口直接返回全部數(shù)據(jù),而且還要實現(xiàn)篩選功能。我...我說不就是一條sql語句的事,改接口多方便,我苦心勸導(dǎo),然后被懟回來,切,不就是篩選嘛,求人不如自己搞。

1. 效果圖

Android Filterable實現(xiàn)Recyclerview篩選功能的示例代碼

2. 思路

既然是篩選,那就少不了比較。也沒有什么好的辦法,無非就是循環(huán)對比,然后將適配器進(jìn)行數(shù)據(jù)更新。頁面刷新即可。但篩選的調(diào)用要方便,怎么比較才方便我們調(diào)用呢?偶然間看到了Filterable,使Adapter繼承自該接口,實現(xiàn)getFilter()方法,在該方法里實現(xiàn)具體的過濾邏輯即可。

3. 實現(xiàn)步驟3.1 數(shù)據(jù)Bean類

class MyBean(var type:String,var name:String,var deliverType:String)

這里我們簡單的創(chuàng)建個數(shù)據(jù)Bean類,后面我們的篩選字段是根據(jù)type和deliverType來進(jìn)行篩選。

3.2 創(chuàng)建適配器

class MyAdapter(data: MutableList<MyBean>) : RecyclerView.Adapter<MyAdapter.MyViewHolder>(), Filterable { //存放原數(shù)據(jù) private var mSourceList = mutableListOf<MyBean>() //存放過濾后的數(shù)據(jù) private var mFilterList = mutableListOf<MyBean>() init { mSourceList = data } override fun onBindViewHolder(holder: MyViewHolder, position: Int) { holder.name.text = mFilterList[position].name holder.deliverType.text = mFilterList[position].deliverType } override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): MyViewHolder { var view = LayoutInflater.from(parent.context).inflate(R.layout.item, parent, false) return MyViewHolder(view) } /** * 注意:這里返回過濾后的集合大小 */ override fun getItemCount(): Int { return mFilterList.size } inner class MyViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) { //商品名稱 var name: TextView = itemView.findViewById(R.id.tvName) //配送方式 var deliverType: TextView = itemView.findViewById(R.id.tvDeliverType) }}

和我們平時創(chuàng)建的Adapter沒什么兩樣。但要注意以下幾點

1、這里我們創(chuàng)建了兩個集合mSourceList和mFilterList,mFilterList主要是用來存放過濾后的數(shù)據(jù),而mSourceList主要是用來在篩選后數(shù)據(jù)恢復(fù)時使用,使得不用再去請求一次數(shù)據(jù)。

2、getItemCount()方法返回過濾后的集合的大小。

有個疑問:假如我們沒有進(jìn)行過濾,而因為我們的mFilterList默認(rèn)為空,且getItemCount()返回的是它的大小0,那我們默認(rèn)是不是就顯示不出數(shù)據(jù)?

3.3 繼承Filterable接口

1、繼承Filterable接口后,實現(xiàn)其getFilter()方法,該方法需要我們返回一個Filter過濾器對象。

2、我們重寫Filter的performFiltering()方法和publishResults()方法,performFiltering()用來實現(xiàn)我們具體過濾的邏輯操作,publishResults()用來將我們過濾后的數(shù)據(jù)進(jìn)行更新。

3、因為performFiltering()傳來的過濾條件是一段字符串,而我們的過濾條件有兩個,所以我們將過濾的條件轉(zhuǎn)化為Json對象傳過來,那樣就可以得到多個過濾條件的字符串了。

4、這里我們的具體過濾操作是使用Collection的filter()方法進(jìn)行過濾

(1)當(dāng)condition1和condition2為空時,返回原數(shù)據(jù)mSourceList

(2)否則使用filter()方法按條件進(jìn)行過濾,最后將過濾后的集合賦值給FilterResults()對象的value字段,并將其返回

5、publishResults(charSequence: CharSequence,filterResults: FilterResults)方法中filterResults對象內(nèi)的value字段是我們performFiltering()方法返回的過濾后的集合,在這里我們將RecyclerView進(jìn)行更新。

具體實現(xiàn)見以下代碼:

class MyAdapter(data: MutableList<MyBean>) : RecyclerView.Adapter<MyAdapter.MyViewHolder>(), Filterable { /** * 具體的執(zhí)行過濾的操作 * 創(chuàng)建適配器后會默認(rèn)的執(zhí)行一次 */ override fun getFilter(): Filter { return object : Filter() { //執(zhí)行過濾操作 override fun performFiltering(charSequence: CharSequence): FilterResults { val charString = charSequence.toString() Log.i(TAG, 'performFiltering: 執(zhí)行過濾操作,過濾字段為:$charString') val jsonObject = JSONObject(charString) //篩選條件一 var condition1 = jsonObject.getString('condition1') //篩選條件二 var condition2 = jsonObject.getString('condition2') //存放已過濾的數(shù)據(jù) var theFilterList = if (condition1.isEmpty() && condition2.isEmpty()) { //沒有過濾的內(nèi)容,則使用源數(shù)據(jù) mSourceList } else if (condition2.isEmpty()) { mSourceList.filter { it.type == condition1 } } else if (condition1.isEmpty()) { mSourceList.filter { it.deliverType == condition2 } } else { mSourceList.filter { it.type == condition1 && it.deliverType == condition2 } } val filterResults = FilterResults() filterResults.values = theFilterList return filterResults } //把過濾后的值返回出來并進(jìn)行更新 override fun publishResults( charSequence: CharSequence, filterResults: FilterResults ) { mFilterList = filterResults.values as MutableList<MyBean> notifyDataSetChanged() } } }}3.4 過濾調(diào)用

class MainActivity : AppCompatActivity() { //過濾條件1 var condition1 = '' //過濾條件2 var condition2 = '' //總的過濾條件 var jsonObject = JSONObject() private var dataList = mutableListOf<MyBean>() var myAdapter = MyAdapter(dataList) override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) mRecyclerView.layoutManager = LinearLayoutManager(this) mRecyclerView.adapter = myAdapter jsonObject.put('condition1','過濾條件一') jsonObject.put('condition2','過濾條件二') myAdapter.filter.filter(jsonObject.toString()) }}

如果想恢復(fù)數(shù)據(jù)不篩選,直接將jsonObject對象內(nèi)的condition1和condition2字段設(shè)為空,然后調(diào)用myAdapter.filter.filter(jsonObject.toString())即可。

具體見代碼

4. 優(yōu)化

其實我們getFilter()內(nèi)的過濾操作還可以優(yōu)化下

var theFilterList = if (condition1.isEmpty() && condition2.isEmpty()) { //沒有過濾的內(nèi)容,則使用源數(shù)據(jù) mSourceList} else if (condition2.isEmpty()) { mSourceList.filter { it.type == condition1 }} else if (condition1.isEmpty()) { mSourceList.filter { it.deliverType == condition2 }} else { mSourceList.filter { it.type == condition1 && it.deliverType == condition2 }}

可以看到else{}下是當(dāng)condition1和condition2都不為空的情況下進(jìn)行的篩選,但是如果我們使用下拉框進(jìn)行篩選時,選擇第一個條件condition1后就已經(jīng)進(jìn)行了一次篩選,即condition1不為空condition2為空,在 else if (condition2.isEmpty()){} 里對源數(shù)據(jù)進(jìn)行了篩選;再選擇第二個條件時,又進(jìn)行了一次篩選,即condition1不為空condition2不為空,在else{}里又是對源數(shù)據(jù)進(jìn)行篩選,其實我們應(yīng)該是在第一次的結(jié)果下進(jìn)行篩選是最優(yōu)的辦法。

想法很好,但實現(xiàn)起來困難挺多,在兩個條件都不為空時,我們需要判斷第一次刪選下來的數(shù)據(jù)是以哪個篩選條件為依據(jù)的,在兩個條件都不為空篩選后,再次更改其中一個篩選條件,我們需要先將另外一個篩選條件下的數(shù)據(jù)給篩選出來,越來越麻煩,暫時不考慮了,有好的方案的麻煩給個思路。

5. 注意

因為Adapter默認(rèn)返回的大小是篩選后的尺寸,而我們默認(rèn)是沒有篩選的,導(dǎo)致上來會沒有數(shù)據(jù),所以我們需要設(shè)置適配器后,人為的調(diào)用一下篩選才好:myAdapter.filter.filter(jsonObject.toString())。而我在項目中沒有寫因為AppCompatSpinner會默認(rèn)的選擇第0項,我在其onItemSelected()回調(diào)里調(diào)用了篩選功能。

6. 總結(jié)

總的來說并不難,還是更新數(shù)據(jù)更新布局的那一套,不同的是用了Filterable接口實現(xiàn),使得篩選調(diào)用的方式更簡單。但是這種實現(xiàn)更多的是適用于數(shù)據(jù)量小或者固定的數(shù)據(jù),如果數(shù)據(jù)量大,或者數(shù)據(jù)會一直上拉加載擴(kuò)充,使用這種方式只會讓效率隨著數(shù)據(jù)量的增大而越來越低,顯然不合適,下次后端還強硬不改,那就只能開懟了。Github項目地址 https://github.com/myfittinglife/RecyclerViewFilterable

7. 參考文章

集合過濾操作 Google文檔

到此這篇關(guān)于Android Filterable實現(xiàn)Recyclerview篩選功能的示例代碼的文章就介紹到這了,更多相關(guān)Android Recyclerview篩選內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Android
相關(guān)文章:
主站蜘蛛池模板: 国产精彩视频 | 91免费国产 | 黄色小视频在线观看免费 | 日韩视频在线免费观看 | 欧洲精品一区二区 | 丁香激情网 | 专业操老外| 青青草国产成人av片免费 | 日韩精品少妇 | 3d动漫精品h区xxxxx区 | 九九免费视频 | 午夜在线视频观看 | 能看的毛片 | av小说在线观看 | 日韩欧美一级片 | 在线国产91| 操日本老女人 | 国模一区二区 | 丁香九月婷婷 | 日本少妇网站 | 成年人网站在线免费观看 | 欧美亚洲视频 | 国产免费黄色 | 日韩中文在线视频 | 精品少妇v888av | 日韩黄色一级 | 九九九色| 免费在线看a | 日本国产精品 | 丁香婷婷网 | 欧美日韩激情 | 在线观看黄色片 | 亚洲成人一区二区三区 | 久久久精品一区二区三区 | 久久精品伊人 | 青青草手机视频 | 亚洲福利片 | 欧美日韩在线一区二区三区 | 美女一级毛片 | 黄色小视频在线观看 | 亚洲天天干 |