kotlin在鸿蒙开发中的实践 原创 精华

没用的喵叔
发布于 2022-1-2 11:37
浏览
4收藏

【本文正在参与优质创作者激励】

先说一说kotlin

我们知道:

  1. kotlin目前是安卓首选的编程语言。

    安卓逐渐抛弃java,拥抱kotlin这是大的趋势。

  2. kotlin的最大优点就是与java的互操作性。

  3. kotlin编译的产物和java一样是bytecode(不抬杠,本文只说面向jvm的kotlin)。

  4. kotlin是一门现代高级语言。

    java也是高级语言,但开发效率很低。

    kotlin作为一门现代的语言,语法更简洁,而且具有很多高级特性,比如:Null Safe、Data Class、扩展、操作符重载、lambda表达式、闭包等

第一次尝试

我使用的鸿蒙ide是DevEco Studio 3.0.0.800。要使用kotlin,肯定要把相应的gradle插件、ide插件等配置上。

  1. gradle插件(作用:编译.kt文件)
    project的gradle.build中配置

    buildscript {
        dependencies {
         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:1.6.10"
        }
    }
    
    • 1.
    • 2.
    • 3.
    • 4.
    • 5.

    entry的gradle.build中配置

    apply plugin: 'kotlin'
    
    • 1.
  2. ide插件(作用:让ide识别.kt文件、高亮显示、语法提示等)

    DevEco Studio 3.0.0.800是基于IntelliJ IDEA212.5457.46,所以我们找到相应的版本并下载https://plugins.jetbrains.com/plugin/6954-kotlin/versions/stable/150173

    把下载的zip包直接拖拽到DevEco界面上就会自动安装。
    kotlin在鸿蒙开发中的实践-鸿蒙开发者社区

  3. 建一个kotlin的Data Class, User

    data class User(val name: String, val age: Int)
    
    • 1.

    在MainAbilitySlice中使用User

    public class MainAbilitySlice extends AbilitySlice {
        @Override
        public void onStart(Intent intent) {
            super.onStart(intent);
            super.setUIContent(ResourceTable.Layout_ability_main);
            User andy = new User("andy", 28);//使用Data Class ,就像java的类一样
    
            Text text = (Text) findComponentById(ResourceTable.Id_text_helloworld);
            text.setText(andy.getName());
        }
    }
    
    • 1.
    • 2.
    • 3.
    • 4.
    • 5.
    • 6.
    • 7.
    • 8.
    • 9.
    • 10.
    • 11.
  4. 结果:

    能够识别.kt文件,并且语法高亮都没问题,说明ide插件配置成功。

    但是,编译失败,提示The 'java' plugin has been applied, but it is not compatible with the Hap plugins.。我的理解是,org.jetbrains.kotlin:kotlin-gradle-plugincom.huawei.ohos:hap冲突。这个问题,已经超出了我的能力范围。

    有没有别的办法呢?

第二次尝试

既然org.jetbrains.kotlin:kotlin-gradle-plugincom.huawei.ohos:hap冲突,我们再建一个java library的module,把.kt文件放在这个module下,是否可行呢?答案是:可行!

直接看源码吧----> https://gitee.com/andych008/HMkt

第三次尝试(kotlin代码中使用鸿蒙sdk中的类)

以自定义一个AlertDialog为例(参考https://gitee.com/chinasoft_ohos/HiPermission/blob/master/library/src/main/java/me/weyye/hipermission/AlertDialog.java)。纯java的代码使用kotlin来实现肯定是没问题的,如果依赖鸿蒙sdk的的类呢?比如ohos.agp.window.dialog.CommonDialog

也很简单,直接把sdk中的ohos.jar(D:\Huawei\Sdk\java\3.0.0.0\api\ohos.jar)放在我的们MyJavaLib中,依赖这个类只是为了完成编译,不会打进最终的hap里。

但是编译过程生成的ResourceTable,我们只能以变通的方式来依赖了。以这个AlertDialog为例,布局文件的资源id就是一个int,其中的组件id也是int,因为id较多,我们封装到一个class ComponentId里。完整的代码如下:(把原java实现通过ide转成kotlin,然后简单修改。我对kotlin用得也不多,所以具体细节肯定有不符合kotlin的使用习惯。这篇文章的目的只是证明可行性,并给出执行方案)

package com.example.myjavalib
import ohos.agp.components.Component
import ohos.agp.components.LayoutScatter
import ohos.agp.components.Text
import ohos.agp.utils.LayoutAlignment
import ohos.agp.window.dialog.CommonDialog
import ohos.agp.window.dialog.IDialog
import ohos.app.Context
import ohos.multimodalinput.event.KeyEvent

/**
 * 自定义提示弹窗
 *
 * @since 2021-04-12
 * https://gitee.com/chinasoft_ohos/HiPermission/blob/master/library/src/main/java/me/weyye/hipermission/AlertDialog.java
 */
class AlertDialog(context: Context?, resId: Int, componentId: ComponentId) : CommonDialog(context) {
    private var tvTitle: Text? = null
    private var tvContent: Text? = null
    private var tvCancel: Text? = null
    private var tvSure: Text? = null

    data class ComponentId(val tvTitle: Int, val tvContent: Int, val tvCancel: Int, val tvSure: Int)


    init {
        setTransparent(true)
        setAlignment(LayoutAlignment.CENTER)
        val contentView = LayoutScatter.getInstance(context)
            .parse(resId, null, true)
        contentCustomComponent = contentView
        initView(componentId)
        siteRemovable(false)
        siteKeyboardCallback { iDialog: IDialog?, keyEvent: KeyEvent? -> true }
    }

    private fun initView(componentId: ComponentId) {
        val component = contentCustomComponent
        tvTitle = component.findComponentById<Component>(componentId.tvTitle) as Text
        tvContent = component.findComponentById<Component>(componentId.tvContent) as Text
        tvCancel = component.findComponentById<Component>(componentId.tvCancel) as Text
        tvSure = component.findComponentById<Component>(componentId.tvSure) as Text

        tvCancel!!.clickedListener = Component.ClickedListener { hide() }
        tvSure!!.clickedListener = Component.ClickedListener { hide() }
    }

    /**
     * 设置标题
     *
     * @param title 标题
     */
    fun setTitle(title: String?) {
        tvTitle!!.text = title
    }

    /**
     * 设置内容
     *
     * @param content 内容
     */
    fun setContent(content: String?) {
        tvContent!!.text = content
    }

    /**
     * 设置取消按钮文字和点击事件监听
     *
     * @param cancel 按钮文字
     * @param listener 点击事件监听
     */
    fun setCancel(cancel: String?, listener: Component.ClickedListener?) {
        tvCancel!!.text = cancel
        tvCancel!!.clickedListener = Component.ClickedListener {
            hide()
            listener?.onClick(tvCancel)
        }
    }

    /**
     * 设置确定按钮文字和点击事件监听
     *
     * @param sure 按钮文字
     * @param listener 点击事件监听
     */
    fun setSure(sure: String?, listener: Component.ClickedListener?) {
        tvSure!!.text = sure
        tvSure!!.clickedListener = Component.ClickedListener {
            hide()
            listener?.onClick(tvSure)
        }
    }
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.
  • 58.
  • 59.
  • 60.
  • 61.
  • 62.
  • 63.
  • 64.
  • 65.
  • 66.
  • 67.
  • 68.
  • 69.
  • 70.
  • 71.
  • 72.
  • 73.
  • 74.
  • 75.
  • 76.
  • 77.
  • 78.
  • 79.
  • 80.
  • 81.
  • 82.
  • 83.
  • 84.
  • 85.
  • 86.
  • 87.
  • 88.
  • 89.
  • 90.
  • 91.
  • 92.
  • 93.

注意:我们依赖的ohos.jar要和entry编译用的api level保持一致。

总结

虽然,我们经过尝试,可以在鸿蒙app开发中使用kotlin,但是,我还是不建议在实际中大量使用。说不准哪一天,DevEco Studio 中完全不支持kotlin了,那样之前写的代码维护会不太方便。但是,如果有一些平台不相关的代码,比如,在android和鸿蒙上都会使用,那么我们完全可以放心使用kotlin来实现。
另外,在华为开发者大会上(2021-10-22),华为表示,将发布自研编程语言。我们就暂时忍受一下难用的java吧。华为的自研编程语言,在特性上肯定会和kotlin、swift之类的现代语言一样。现在学习使用kotlin对以后一定会有用的!

©著作权归作者所有,如需转载,请注明出处,否则将追究法律责任
标签
已于2022-1-8 08:13:50修改
7
收藏 4
回复
举报
7
8
4
8条回复
按时间正序
/
按时间倒序
红叶亦知秋
红叶亦知秋

期待华为的自研语言!

回复
2022-1-4 09:55:10
陈浩南xxx
陈浩南xxx

难用的java??????????     搞笑把

回复
2022-1-4 10:15:06
没用的喵叔
没用的喵叔 回复了 陈浩南xxx
难用的java?????????? 搞笑把

和kt比较一下:要定义一个Data Class得写一堆的get set;要实现代理也得定义一堆的类;实现一个单例都不确定写得对不对,是否线程安全;java里继承满天飞,kt很多地方用扩展更直观;要把函数作为参数,java得定义一个接口类。还有很多。

回复
2022-1-4 12:06:26
没用的喵叔
没用的喵叔 回复了 红叶亦知秋
期待华为的自研语言!

到时候也很尴尬,java js ets 这么多了,表示学不动了。

1
回复
2022-1-4 12:08:19
BinWoo
BinWoo 回复了 没用的喵叔
和kt比较一下:要定义一个Data Class得写一堆的get set;要实现代理也得定义一堆的类;实现一个单例都不确定写得对不对,是否线程安全;java里继承满天飞,kt很多地方用扩展更直观;要把函数作为参数,java得定义一个接口类。还有很多。

你听过lombok没?

回复
2022-1-5 15:23:34
没用的喵叔
没用的喵叔 回复了 BinWoo
你听过lombok没?

lombok 已经被多数人批判了

回复
2022-1-5 16:54:07
qq5848d10879e10
qq5848d10879e10
纯鸿蒙应用是运行在虚拟机上的吗?
回复
2022-1-6 09:50:36
没用的喵叔
没用的喵叔 回复了 qq5848d10879e10
纯鸿蒙应用是运行在虚拟机上的吗?

对。用java写的app都是跑在虚拟机上。

回复
2022-1-6 18:25:49


回复
    相关推荐