首先自定义View有三个构造方法:
级联式调用,每一个构造函数调用比它多一个参数的构造函数,最后一个构造函数调用基类的构造函数,最后在做一些额外的初始化工作。
public class CustomView extends View { /** * 第一个构造函数 * @param context */ public CustomView(Context context) { super(context,null); } /** * 第二个构造函数 * @param context * @param attrs */ public CustomView(Context context, @Nullable AttributeSet attrs) { super(context, attrs,0); } /** * 第三个构造函数 * @param context * @param attrs * @param defStyleAttr */ public CustomView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); //获取自定义属性 } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); } }
在代码中的使用时机:
1.直接new一个实例的时候,会调用第一个构造函数
2.在xml布局文件调用的时候,会调用第二个构造函数
3.在xml布局文件中调用,并且标签中还有自定义属性时,这里调用的还是第二个构造函数
也就是说,系统默认只会调用前两个构造函数,至于第三个构造函数的调用,通常都是我们在构造函数中主动调用的。(例如,在第二个构造函数中调用第三个构造函数)
推荐使用这种:
每个构造函数分别调用基类的构造函数,再调用一个公共的初始化方法做额外初始化。
public class CustomView extends View { /** * 在Java代码中直接new出来的时候调用 * @param context */ public CustomView(Context context) { super(context); initCustomView(); } /** * 在XML中不使用自定义属性的时候调用 * @param context * @param attrs */ public CustomView(Context context, @Nullable AttributeSet attrs) { super(context, attrs); initCustomView(); } /** * 在XML中使用自定义属性的时候调用 * @param context * @param attrs * @param defStyleAttr */ public CustomView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); initCustomView(); //获取自定义属性 }比如你自定义的View继承自ListView或者TextView的时候,ListView或者TextView内部的构造函数会有一个默认的defStyle, 第二种方法调用时defStyle会传入0,这将覆盖基类中默认的defStyle,进而导致一系列问题。
本文详细解析了自定义View的三种构造方法及其调用时机,介绍了如何正确使用这些构造方法来实现更灵活的自定义View功能。
2万+

被折叠的 条评论
为什么被折叠?



