Android 自定义 MarqueeView 实现跑马灯 —— 原理篇,大厂面试参考指南v1.0

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip204888 (备注Android)
img

正文

throw new IllegalArgumentException(“No ItemViewDelegate added that matches position=” +

position + " in data source");

}

MultiItemTypeAdapter 讲解

主要有几个重要的方法:

public View createItemView(ItemViewDelegate itemViewDelegate, ViewGroup parent) {

int layoutId = itemViewDelegate.getItemViewLayoutId();

ViewHolder viewHolder = null;

View convertView = LayoutInflater.from(mContext).inflate(layoutId, parent, false);

viewHolder = new ViewHolder(mContext, convertView, parent, -1);

viewHolder.mLayoutId = layoutId;

onViewHolderCreated(viewHolder, viewHolder.getConvertView());

return convertView;

}

public View createItemView(int position, View convertView, ViewGroup parent) {

ItemViewDelegate itemViewDelegate = mItemViewDelegateManager.getItemViewDelegate(mDatas

.get(position), position);

int layoutId = itemViewDelegate.getItemViewLayoutId();

ViewHolder viewHolder = null;

if (convertView == null) {

convertView = LayoutInflater.from(mContext).inflate(layoutId, parent, false);

viewHolder = new ViewHolder(mContext, convertView, parent, position);

viewHolder.mLayoutId = layoutId;

onViewHolderCreated(viewHolder, viewHolder.getConvertView());

} else {

viewHolder = (ViewHolder) convertView.getTag();

viewHolder.mPosition = position;

}

convert(viewHolder, getItem(position), position);

return convertView;

}

private void convert(ViewHolder viewHolder, T item, int position) {

mItemViewDelegateManager.convert(viewHolder, item, position);

}

public SparseArrayCompat getAllTyeView(ViewGroup parent) {

SparseArrayCompat<ItemViewDelegate> itemViewDelegates = getItemViewDelegate();

int size = itemViewDelegates.size();

SparseArrayCompat viewSparseArrayCompat = new SparseArrayCompat<>();

for (int i = 0; i < size; i++) {

ItemViewDelegate delegate = itemViewDelegates.valueAt(i);

View itemView = createItemView(delegate, parent);

int itemViewType = getItemViewType(itemViewDelegates, i);

Log.i(TAG, "getAllTyeView: itemViewType = " + itemViewType);

viewSparseArrayCompat.put(itemViewType, itemView);

}

return viewSparseArrayCompat;

}

  • 第一个方法: createItemView(ItemViewDelegate itemViewDelegate, ViewGroup parent),会根据传递的 itemViewDelegate 创建相应的 convertView,并调用 onViewHolderCreated() 方法

  • 第二个方法:createItemView 会根据传递进来的 position 创建相应的 convertView

  • 若 convertView 为 null,从布局中 load 进来

  • 若 convertView 不为空,取出来 viewHolder,并刷新 viewHolder 里面的 position

最后调用 convert 方法去刷新界面数据。

而这个 convertView 什么时候为 null,什么时候不为 null,这个必须要外部调用来管理,MultiItemTypeAdapter 管理不了,也不应该管理。

  • 第三个方法: getAllTyeView ,这个方法会遍历所有的 itemViewDelegate 并创建相应的 View 及 ViewHolder

接下来我们来看一下在 MarqueeView 里面是怎样实现 convertView 的缓存的,标重点了。


MarqueeView


首先我们来看一下 getItemView

private SparseArray mViews;

private View getItemView(int index) {

int itemViewType = mMultiItemTypeAdapter.getItemViewType(index);

// 获取缓存的 convertView

View convertView = mViews.get(itemViewType);

View itemView = mMultiItemTypeAdapter.createItemView(index, convertView, MarqueeView.this);

return itemView;

}

从代码中可以看出我们是从 mViews 里面根据当前位置 index 的 itemViewType 取出 convertView 的。那我们的 mViews 是什么时候赋值的呢?

是在 addAllTypeView 方法中

private void addAllTypeView() {

int viewTypeCount = mMultiItemTypeAdapter.getViewTypeCount();

if (viewTypeCount < 1) {

return;

}

mViews.clear();

SparseArrayCompat allTyeView = mMultiItemTypeAdapter.getAllTyeView(MarqueeView.this);

int curItemViewType = mMultiItemTypeAdapter.getItemViewType(mPosition);

for (int i = 0; i < allTyeView.size(); i++) {

int key = allTyeView.keyAt(i);

View view = allTyeView.valueAt(i);

mViews.put(key, view);

LayoutParams layoutParams = new FrameLayout.LayoutParams(LayoutParams.MATCH_PARENT,

LayoutParams.WRAP_CONTENT);

layoutParams.gravity = mGravity;

addView(view, layoutParams);

// 设置当前 itemView 可见,其他不可见

if (key == curItemViewType) {

view.setVisibility(View.VISIBLE);

} else {

view.setVisibility(View.INVISIBLE);

}

}

}

在 addAllTypeView 的时候,会调用 mMultiItemTypeAdapter.getAllTyeView 初始化所有类型的 itemView,并添加到 mViews 缓存,key 为 viewType,value 为 itemView。

MarqueeView 是怎样与 MultiItemTypeAdapter 建立关联的

我们来看一下 setAdapter 这个方法:

有一个参数,MultiItemTypeAdapter ,这个 MultiItemTypeAdapter 主要是用来实现 View 的复用以及根据不同的 viewType 添加不同的 View 的。这里先大概有个印象。下面会讲解到。

public void setAdapter(MultiItemTypeAdapter multiItemTypeAdapter) {

if (multiItemTypeAdapter == null) {

return;

}

mMultiItemTypeAdapter = multiItemTypeAdapter;

start(mInAnimResId, mOutAnimResId);

}

private void start(final @AnimRes int inAnimResId, final @AnimRes int outAnimResID) {

// 第一步:做一些重置的工作,mPosition 终止,清除所有 View,清除动画;

mPosition = 0;

clearAnimation();

removeAllViews();

// 第二步:根据 MultiItemTypeAdapter ,把所有类型的 typeView 加载进来,并根据 mPosition 设置可见性

addAllTypeView();

// 第三步:初始化当前 position 的 View,并调用 mMultiItemTypeAdapter 的相关方法

int itemViewType = mMultiItemTypeAdapter.getItemViewType(mPosition);

View convertView = mViews.get(itemViewType);

View itemView = mMultiItemTypeAdapter.createItemView(mPosition, convertView, MarqueeView.this);

mCurView = itemView;

mLastView = mCurView;

// 利用 handle 发送消息,执行动画

post(new Runnable() {

@Override

public void run() {

sendAppear();

}

});

}

在 setAdapter 方法中,会先用 start 方法。而在 start 方法中主要做即将事情

  • 第一步:做一些重置的工作,mPosition 终止,清除所有 View,清除动画;

  • 第二步:根据 MultiItemTypeAdapter ,把所有类型的 typeView 加载进来,并根据 mPosition 设置可见性

  • 第三步:初始化当前 position 的 View,并调用 mMultiItemTypeAdapter 的 createItemView 去初始化对应 postion 的 View

int itemViewType = mMultiItemTypeAdapter.getItemViewType(mPosition);

View convertView = mViews.get(itemViewType);

View itemView = mMultiItemTypeAdapter.createItemView(mPosition, convertView, MarqueeView.this);

mCurView = itemView;

mLastView = mCurView;

  • 第四步:利用 handle 发送消息,执行进场动画

post(new Runnable() {

@Override

public void run() {

sendAppear();

}

});

private void sendAppear() {

mHandler.removeMessages(APPEAR);

if (!isStart) {

return;

}

mHandler.sendEmptyMessageDelayed(APPEAR, 0);

}


MarqueeView 是怎样轮询执行动画的

实质是用 hanlde 不断发送消息

接受到 APPEAR 消息的时候:

首先获取当前位置的 ItemView,接着执行动画,执行完动画之后,mLastView = mCurView; 。接着,判断当前是否还需要执行 flip 动画,如果需要的话,会发送并发送延时消息,告诉下一次执行小时动画的时间。如果,不需要,则不会发送 DIS_APPEAR 消息

private Handler mHandler = new Handler() {

@Override

public void handleMessage(Message msg) {

super.handleMessage(msg);

switch (msg.what) {

case APPEAR:

handleAppearMes();

break;


}

private void handleAppearMes() {

mLastView = mCurView;

mCurView = getItemView(mPosition);

Animation inAnimation = getInAnimation();

inAnimation.setAnimationListener(new Animation.AnimationListener() {

@Override

public void onAnimationStart(Animation animation) {

mLastView.setVisibility(View.GONE);

mCurView.setVisibility(View.VISIBLE);

if (mIFlipListener != null) {

mIFlipListener.onFilpStart(mPosition, mCurView);

}

}

@Override

public void onAnimationEnd(Animation animation) {

mLastView = mCurView;

mCurView = getItemView(mPosition);

if (mIFlipListener != null) {

mIFlipListener.onFilpSelect(mPosition, mCurView);

}

sendDisappear();

}

@Override

public void onAnimationRepeat(Animation animation) {

总结

其实要轻松掌握很简单,要点就两个:

  1. 找到一套好的视频资料,紧跟大牛梳理好的知识框架进行学习。
  2. 多练。 (视频优势是互动感强,容易集中注意力)

你不需要是天才,也不需要具备强悍的天赋,只要做到这两点,短期内成功的概率是非常高的。

对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。

以上就是总结的关于在面试的一些总结,希望对大家能有些帮助,除了这些面试中需要注意的问题,当然最重要的就是刷题了,这里放上我之前整理的一份超全的面试专题PDF

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

这里只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注Android)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

中需要注意的问题,当然最重要的就是刷题了,这里放上我之前整理的一份超全的面试专题PDF

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

[外链图片转存中…(img-7tiU8FZ2-1713684902831)]

这里只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注Android)
[外链图片转存中…(img-e1ramCFG-1713684902832)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 13
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值