Java 自定义 View → Kotlin 自定义 View

Java 自定义 View → Kotlin 自定义 View

老写法(Java)

java 复制代码
public class CircleView extends View {

    private Paint paint;
    private int circleColor;

    public CircleView(Context context) {
        super(context);
        init(null);
    }

    public CircleView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(attrs);
    }

    public CircleView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init(attrs);
    }

    private void init(AttributeSet attrs) {
        paint = new Paint(Paint.ANTI_ALIAS_FLAG);
        circleColor = Color.RED;

        if (attrs != null) {
            TypedArray a = getContext().obtainStyledAttributes(attrs, R.styleable.CircleView);
            circleColor = a.getColor(R.styleable.CircleView_circleColor, Color.RED);
            a.recycle();
        }
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        paint.setColor(circleColor);
        int radius = Math.min(getWidth(), getHeight()) / 2;
        canvas.drawCircle(getWidth() / 2f, getHeight() / 2f, radius, paint);
    }

    public void setCircleColor(int color) {
        this.circleColor = color;
        invalidate();
    }
}

问题在哪里

三个重载构造函数纯属样板代码。每个自定义 View 都要写一遍,十几个 View 的项目光构造函数就占上百行。init() 方法的存在也只是为了绕过构造函数不能互相调用的限制。

新写法(Kotlin)

kotlin 复制代码
class CircleView @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = 0
) : View(context, attrs, defStyleAttr) {

    private val paint = Paint(Paint.ANTI_ALIAS_FLAG)
    private var circleColor = Color.RED

    init {
        if (attrs != null) {
            val typedArray = context.obtainStyledAttributes(attrs, R.styleable.CircleView)
            circleColor = typedArray.getColor(R.styleable.CircleView_circleColor, Color.RED)
            typedArray.recycle()
        }
    }

    override fun onDraw(canvas: Canvas) {
        super.onDraw(canvas)
        paint.color = circleColor
        val radius = minOf(width, height) / 2
        canvas.drawCircle(width / 2f, height / 2f, radius.toFloat(), paint)
    }

    fun setCircleColor(color: Int) {
        circleColor = color
        invalidate()
    }
}

一句话注意

@JvmOverloads 配合 Kotlin 默认参数,会自动生成 Java 调用方需要的三个重载构造函数。如果自定义 View 需要在 XML 中使用,这个注解不能省,否则布局填充器找不到对应签名的构造函数会直接崩溃。

paint.color = circleColor 是 Kotlin 的属性访问语法,等价于 Java 的 paint.setColor(circleColor)。Android SDK 里所有 getXxx()/setXxx() 方法对在 Kotlin 中都会自动映射为属性,不需要像 Java 那样显式调用 setter。


Java Android 老项目迁移系列,持续更新中。

相关推荐
用户128526116026 小时前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java
通玄6 小时前
Jetpack Compose 入门系列(六):Navigation 3 页面导航
android
Linsk6 小时前
组件 = 模板 + 业务逻辑
java·前端·vue.js
星沉远浦7 小时前
用Gemini高效解决Java代码报错难以定位的问题
java
rocpp8 小时前
Android 多语言切换实战:从 Context 到 Android 13 应用语言适配
android·kotlin
释然小师弟9 小时前
Android开发十年:反思与回顾
android·后端·嵌入式
用户2986985301410 小时前
Word 文档字符级格式化:Java 实现方案详解
java·后端
笨鸟飞不快11 小时前
从单个服务到集群:一次完整的性能排查复盘
java·前端
荣码11 小时前
用Streamlit给AI应用套个界面,10行代码出Web页面
java·python
SamDeepThinking11 小时前
Java微服务练习方式
java·后端·微服务