正常使用ClassicsHeader没有问题,在自定义header中遇到的两个问题总结下。
现象描述:
APP打开,放置2分钟,APP内存陆续飙升,最后OOM。
原因分析:
在引入SmartRefreshLayout库之后自定义header,继承自RefreshHeader,重写对应的抽象方法,之后会出现内存溢出现象。
通过Android studio profile分析memory自定义的RefreshLayout extends SmartRefreshLayou内存越来越大,通过MAT进一步分析MyHeaderView存在问题。原因如下:
MyHeaderView在getView中返回布局,如下写法:
public class MyHeaderView extends LinearLayout implements RefreshHeader {
****************省略****************
@Override
public View getView() {
View view = LayoutInflater.from(context).inflate(R.layout.refresh_layout, null);
view1 = view.findViewById(R.id.iv_view1);
view2 = view.findViewById(R.id.iv_view2);
view3 = view.findViewById(R.id.iv_view3);
return view;
}
****************省略****************
}
这种写法是导致OOM的元凶,getView方法只是返回View对象,不能findVideById多次,初始化View需要在构造方法中进行,只需要初始化一次即可,经过优化,改为如下即可:
解决方案:
public class MyHeaderView extends LinearLayout implements RefreshHeader {
****************省略****************
private void init(Context context) {
View view = LayoutInflater.from(context).inflate(R.layout.refresh_layout, this);
view1 = view.findViewById(R.id.iv_view1);
view2 = view.findViewById(R.id.iv_view2);
view3 = view.findViewById(R.id.iv_view3);
}
@NonNull
@Override
public View getView() {
return this;
}
****************省略****************
}
然后在构造方法中调用init进行初始化,就不会出现内存溢出问题。
现象描述:
自定义header,首次进入APP,禁用下拉刷新,滑动webview到超过一屏,到点击button,让webview滑动到顶部,触发autoRefresh,这时候发现刷新出现,但是MyHeader并没有被绘制出来getView的布局没有被加载,但是用手指稍微滑动一下,就正常了。如果首次通过触摸这一切也正常。
原因分析:
通过设置MyHeader的属性,重写onMeasure,onLayout,延时加载等等方法依然无法解决,网上搜到一些解决方案:
上面的方案中都没有效果,最后一个人的现象和我的情况一样,没有现成的方案,下载了SmartRefreshLayout推荐的几个demo,发现自定义和我的header一样。最后只能一步步看源码了,最后发现ClassicsAbstract有一段这样代码:
@Override
public void onInitialized(@NonNull RefreshKernel kernel, int height, int maxDragHeight) {
mRefreshKernel = kernel;
mRefreshKernel.requestDrawBackgroundFor(this, mBackgroundColor);
}
然后在自定义的header加上,瞬间自定义的header头就出来了,哈哈,就是requestDrawBackgroundFor在作祟。出来是出来了,但是刷新时候的主题\强调颜色是系统默认的:
#6200EE
#3700B3
#03DAC5
但是我并么设置该属性,领外把colors的colorPrimary删掉或者改成其他颜色,还是不行,究竟哪里设置了,还没查出来,通过setPrimaryColors也无效,最后又继续查看ClassicsAbstract源码:
@Override
public void setPrimaryColors(@ColorInt int... colors) {
if (colors.length > 0) {
final View thisView = this;
if (!(thisView.getBackground() instanceof BitmapDrawable) && !mSetPrimaryColor) {
setPrimaryColor(colors[0]);
mSetPrimaryColor = false;
}
if (!mSetAccentColor) {
if (colors.length > 1) {
setAccentColor(colors[1]);
}
mSetAccentColor = false;
}
}
}
public T setPrimaryColor(@ColorInt int primaryColor) {
mSetPrimaryColor = true;
mBackgroundColor = primaryColor;
if (mRefreshKernel != null) {
mRefreshKernel.requestDrawBackgroundFor(this, primaryColor);
}
return self();
}
解决方案:
在MyHeader又实现了上面两个方法,通过SmartRefreshLayout设置setPrimaryColors设置的颜色就生效了,终于完美解决,首次autoRefresh正常了,header出来了,主题颜色也正常了。接入SmartRefreshLayout很方便,但是坑也有啊。