面試思維導圖
以下所有內容以及實用的知識點已被整理app,歡迎下載
下載方式一:酷安商城搜索Android_程序猿即可下載
下載方式二:掃描下載二維碼進行下載
ListView系列問題
1.1 ListView卡頓的原因與性能優化,越多越好
重用converView: 通過復用converview來減少不必要的view的創建,另外Infalte操作會把xml文件實例化成相應的View實例,屬于IO操作,是耗時操作。
減少findViewById()操作: 將xml文件中的元素封裝成viewholder靜態類,通過converview的setTag和getTag方法將view與相應的holder對象綁定在一起,避免不必要的findviewbyid操作
避免在 getView 方法中做耗時的操作: 例如加載本地 Image 需要載入內存以及解析 Bitmap ,都是比較耗時的操作,如果用戶快速滑動listview,會因為getview邏輯過于復雜耗時而造成滑動卡頓現象。用戶滑動時候不要加載圖片,待滑動完成再加載,可以使用這個第三方庫glide
Item的布局層次結構盡量簡單,避免布局太深或者不必要的重繪
盡量能保證 Adapter 的 hasStableIds() 返回 true 這樣在 notifyDataSetChanged() 的時候,如果item內容并沒有變化,ListView 將不會重新繪制這個 View,達到優化的目的
在一些場景中,ScollView內會包含多個ListView,可以把listview的高度寫死固定下來。 由于ScollView在快速滑動過程中需要大量計算每一個listview的高度,阻塞了UI線程導致卡頓現象出現,如果我們每一個item的高度都是均勻的,可以通過計算把listview的高度確定下來,避免卡頓現象出現
使用 RecycleView 代替listview: 每個item內容的變動,listview都需要去調用notifyDataSetChanged來更新全部的item,太浪費性能了。RecycleView可以實現當個item的局部刷新,并且引入了增加和刪除的動態效果,在性能上和定制上都有很大的改善
ListView 中元素避免半透明: 半透明繪制需要大量乘法計算,在滑動時不停重繪會造成大量的計算,在比較差的機子上會比較卡。 在設計上能不半透明就不不半透明。實在要弄就把在滑動的時候把半透明設置成不透明,滑動完再重新設置成半透明。
盡量開啟硬件加速: 硬件加速提升巨大,避免使用一些不支持的函數導致含淚關閉某個地方的硬件加速。當然這一條不只是對 ListView。
1.2 怎么實現一個部分更新的 ListView?(了解)
1.先通過listView.getChildAt(position)拿到要更新的對應的item布局文件,然后再通過findViewById找到對應的控件進行設置。
2.通過Item找出對應的ViewHolder,然后通過ViewHolder去設置值
3.調用適配器對應的getView方法,用它里面的代碼對界面進行刷新。這也是google在IO大會上推薦的做法
private void updateView(int itemIndex) {
//得到第一個可顯示控件的位置,
int visiblePosition = mListView.getFirstVisiblePosition();
//只有當要更新的view在可見的位置時才更新,不可見時,跳過不更新
if (itemIndex - visiblePosition >= 0) {
//得到要更新的item的view
View view = mListView.getChildAt(itemIndex - visiblePosition);
//調用adapter更新界面
mAdapter.updateView(view, itemIndex);
}
}
1.3 當 ListView 數據集改變后,如何更新 ListView?
使用該 ListView 的 adapter 的 notifyDataSetChanged()方法。該方法會 使 ListView 重新繪制。
1.4 ListView 如何實現分頁加載(理解)
1.設 置 ListView 的 滾 動 監 聽 器 : setOnScrollListener(new OnScrollListener{….})
在監聽器中有兩個方法: 滾動狀態發生變化的方法 (onScrollStateChanged)和 listView 被滾動時調用的方法(onScroll)-
2.在滾動狀態發生改變的方法中,有三種狀態:
- 手指按下移動的狀態: SCROLL_STATE_TOUCH_SCROLL: // 觸摸滑動
- 慣性滾動(滑翔(flgin)狀態): SCROLL_STATE_FLING: // 滑翔 靜止狀態:
- SCROLL_STATE_IDLE: // 靜止
3.對不同的狀態進行處理:
分批加載數據,只關心靜止狀態:關心最后一個可見的條目,如果最后一個 可見條目就是數據適配器(集合)里的最后一個,此時可加載更多的數據。在每 次加載的時候,計算出滾動的數量,當滾動的數量大于等于總數量的時候,可以 提示用戶無更多數據了。
1.5 怎么實現ListView多種布局?(了解)
ListView 顯示的每個條目都是通過 baseAdapter 的 getView(int position, View convertView, ViewGroup parent)來展示的,理 論上我們完全可以讓每個條目都是不同類型的 view。
比如:從服務器拿回一個標識為 id=1,那么當 id=1 的時候,我們就加載類 型一的條目,當 id=2 的時候,加載類型二的條目。常見布局在資訊類客戶端中 可以經常看到。
除此之外 adapter 還提供了 getViewTypeCount()和 getItemViewType(int position)兩個方法。在 getView 方法中我們可以根據不 同的 viewtype 加載不同的布局文件。
1.6 ListView 如何定位到指定位置?
可以通過 ListView 提供的 lv.setSelection(listView.getPosition());方法。
1.7 在ListView中設置Selector為null會報空指針? (知道)
mListView.setSelector(null);//空指針
試試下面這個:
mListView.setSelector(new ColorDrawable(Color.TRANSPARENT));
1.8 ListView中主要有2種監聽器:
- OnItemClickListener: 處理視圖中單個條目的點擊事件。 (OnItemSelectedListener,參數與其一致,只是監聽用戶手指行為不同)
- OnScrollListener:監聽滾動變化,可以用于視圖滾動中加載數據。
1.9 ListView中圖片錯位問題
每次getView能給對象一個標識,在異步加載完成時比較標識與當前行item的標識是否一致,一致則顯示,否則不做處理即可
// 給 ImageView 設置一個 tag
holder.img.setTag(imgUrl);
// 預設一個圖片
holder.img.setImageResource(R.drawable.ic_launcher);
// 通過 tag 來防止圖片錯位
if (imageView.getTag() != null &&imageView.getTag().equals(imageUrl)) {
imageView.setImageBitmap(result);
}
2.0 ListView與數據庫綁定的實現(了解)
第一種是用SimpleAdapter創建
第二種是用SimpleCursorAdapter創建