kotlin和java混合开发总结

声明:本文是作者AWeiLoveAndroid原创,版权归作者AWeiLoveAndroid所有,侵权必究。如若转发,请注明作者和来源地址!未经授权,严禁私自转载!

我本来是不想学习kotlin的,但是现在的形势,很多公司都在使用kotlin开发,可以说学会kotlin也是无奈之举,既然是潮流,谷歌也在大力推广,所以还是只能硬着头皮逼迫自己学一下,也能更快适应公司的需要。正所谓:“识时务者为俊杰”,改变不了公司的发展,那就改变自己。本文首发在 慕课网,转载请注明原文链接和作者信息。

废话少说,直接上正题。本文主要讲解kotlin和java混合开发的一个初探。这是一个入门篇的总结博客。我这里使用的Kotlin版本是最新的版本,版本号是 1.3.0


目录

本文配套源码请看github:https://github.com/AweiLoveAndroid/Kotlin-learn/

一、Kotlin集成步骤和注意事项(基于Kotlin1.3.0):

1.在工程根目录的 build.gradle 里面的 buildscript 节点的 dependencies 子节点 添加一行依赖:

buildscript {
    dependencies {
        // 添加这一行依赖
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
    }
}

2.在工程根目录 build.gradlebuildscript节点 配置kotlin版本号:

buildscript {
    ext.kotlin_version = '1.3.0'
    ...
}

3.在项目module的 build.gradle 的顶部添加以下代码:

apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'

4.在项目module的 build.gradledependencies 节点里添加以下代码:

dependencies {
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8:$kotlin_version"
}

5.注意事项:

1.本地kotlin版本路径:

Android Studio安装根路径/gradle/m2repository/org/jetbrains/kotlin/kotlin-stdlib/

2.不同了版本的依赖库

    1.2.0以前使用这个依赖库
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jre8:$kotlin_version"
    1.2.0+版本使用这个依赖库
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8:$kotlin_version"

3.Tools --> kotlin --> Decompiler Kotlin to Java,如果该选项是灰色,请下载插件 Java Bytecode Decompiler

4.这几个依赖库的区别:

依赖库名称 含义
implementation "org.jetbrains.kotlin:kotlin-stdlib-jre7:$kotlin_version" 1.2.0版本以前使用的配置,兼容java7
implementation "org.jetbrains.kotlin:kotlin-stdlib-jre8:$kotlin_version" 1.2.0以前使用的配置,兼容java8
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version" 1.2.0+版本以前使用的配置,兼容java7
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8:$kotlin_version" 1.2.0+版本使用的配置,兼容java8

5.Kotlin下载到本地的路径在哪里?

工程目录的末尾有个External Libraries,选择某一个kotlin的jar包,打开,
右键选择 Show in Explorer,就可以看到kotlin的jar包和资源下载在本地的路径了。

如下图所示:

kotlin的所有jar下载的路径为:
C:/Users/Administrator/.gradle/caches/modules-2/files-2.1/org.jetbrains.kotlin/
Users是我的电脑名。

如下图所示:


二、Kotlin和Java文件互相转换

(1)Java转Kotlin

  • 1.选中Java文件,点击菜单栏Code-->Convert Java File to Kotlin File ,即可把java文件转换成kotlin文件。
  • 2.点击右上角提示的 configue ,然后会出现一个弹窗。
  • 3.设置工程里面哪些需要配置kotlin文件支持。比如我这里选中的是single module,module名为kotlinlibrary。还可以选择kotlin的编译和运行版本号,比如我选择的1.2.30,然后点击OK即可,如下图所示。
  • 4.然后我们就可以看到项目根路径的build.gradlekotlinlibrary这个module的build.gradle都做了kotlin的配置。如图所示:

根路径的 build.gradle 配置:

kotlinlibrarybuild.gradle 配置:

(2)Kotlin转Java

点击菜单栏Tools-->Kotlin-->Decompile Kotlin To Java 就可以了,我用的AS3.1.3,不知道为啥我的是灰色的?有人可以告诉我原因吗?


三、Kotlin项目里面集成Java的module

这一部分源码请看github:https://github.com/AweiLoveAndroid/Kotlin-learn/tree/master/KotlinAndJavaDemo

(1)创建一个名为 KotlinAndJavaDemo 的项目创建一个module,取名为JavaModule,跟普通创建一样,只是最后一步要选择使用的语言是Java,如图所示:


(2)项目代码和平常使用Java开发一样。

代码、资源文件、预览图都和平时开发是一样的。


(3)如果我们是一个Java的lib库,kotlin也是可以直接调用的。

我的kotlin的module名为app, 依赖了一个java的名为javalibrary的lib。我需要在app里面使用javalibraryResourcesUtils,然后获取状态栏高度,如果成功,就弹吐司显示状态栏高度。

这个是javalibraryResourcesUtils

/**
 * 描述:
 *  资源工具类
 */
public class ResourcesUtils {
    /**
     * 获取状态栏高度
     * @return 状态栏高度
     */
    public static int getStatusBarHeight(Context context) {
        //获取status_bar_height资源的ID
        int resourceId = context.getResources().getIdentifier("status_bar_height", "dimen", "android");
        if (resourceId > 0) {
            //根据资源ID获取响应的尺寸值
            return context.getResources().getDimensionPixelSize(resourceId);
        } else {
            return dip2px(context,24f);
        }
    }

    // dp 转成 px
    private static int dip2px(Context context, Float dpValue) {
        float scale = context.getResources().getDisplayMetrics().density;
        return (int) (dpValue * scale + 0.5f);
    }
}

再来看看appMainActivity里面的使用:

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        init(this)
    }

    fun init(context: Context) {
        val statusBarHeight = ResourcesUtils.getStatusBarHeight(context)
        Toast.makeText(context, "状态栏高度为:$statusBarHeight",Toast.LENGTH_SHORT).show()
    }
}

(4)我们在看看看完整的工程结构图


四、Java项目里面集成Kotlin的module

这一部分源码请看github:https://github.com/AweiLoveAndroid/Kotlin-learn/tree/master/JavaAndKotlinDemo

(1)创建一个名为 KotlinAndJavaDemo 的项目创建一个module,取名为KotlinModule,跟普通创建一样,只是最后一步要选择使用的语言是Kotlin,如图所示:


(2)项目代码和平常使用Kotlin开发一样。


(3)如果我们是一个Kotlin的lib库,Java也是可以直接调用的。

我的Java的module名为app, 依赖了一个Kotlin的名为kotlinlibrary的lib。我需要在app里面使用kotlinlibraryResourcesUtils,然后获取状态栏高度,如果成功,就弹吐司显示状态栏高度。

这个是kotlinlibraryResourcesUtils

package com.lzw.kotlinlibrary

import android.content.Context

/**
 * 描述:
 * 资源工具类
 */
class ResourcesUtils {

    // 伴生对象   这里面包裹的方法  实际上类似于java的静态方法
    companion object {
        /**
         * 获取状态栏高度
         * @return 状态栏高度
         */
        fun getStatusBarHeight(context: Context): Int {
            //获取status_bar_height资源的ID
            val resourceId = context.resources.getIdentifier("status_bar_height", "dimen", "android")
            return if (resourceId > 0) {
                //根据资源ID获取响应的尺寸值
                context.resources.getDimensionPixelSize(resourceId)
            } else {
                dip2px(context, 40f)
            }
        }

        /**
         * dp 转 px(像素)
         */
        fun dip2px(context: Context, dpValue: Float?): Int {
            val scale = context.resources.displayMetrics.density
            return (dpValue!! * scale + 0.5f).toInt()
        }

        /**
         * px(像素) 的单位 转 dp
         */
        fun px2dip(context: Context, pxValue: Float?): Int {
            val scale = context.resources.displayMetrics.density
            return (pxValue!! / scale + 0.5f).toInt()
        }

    }

}

再来看看appMainActivity里面的使用:

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        init(this);
    }

    private void init(Context context) {
        int statusBarHeight = ResourcesUtils.Companion.getStatusBarHeight(context);
        Toast.makeText(context, "状态栏高度为:" + statusBarHeight + "像素",Toast.LENGTH_SHORT).show();
    }
}

(4)我们在看看看完整的工程结构图

image

五、同一个module同时使用Java和Kotlin

(一)给kotlin文件单独配置文件路径

我们还是拿第一个前面的工程KotlinAndJavaDemo为例,我在moduleapp里面同时使用kotlin和java两种语言做开发。 为了便于区分,需要在该module的build.gradle文件做配置。

默认情况,Kotlin文件存放在src/main/java/文件夹下。若是需要隔开java文件和Kotlin文件,我们可以指定一个专门存放Kotlin文件的文件夹, 首先在src/main目录下创建文件夹kotlin,然后做以下配置:

android {
    // 配置一个路径专门存放kotlin文件
    sourceSets {
        main.java.srcDirs += 'src/main/kotlin'
    }

图示如下:

(二)代码里面的使用

(1)如果主工程是kotlin写的,部分是Java写的。

比如Activity是kotlin写的,工具类是java写的,实例代码如下:

package com.lzw.kotlinandjavademo

import android.content.Context
import android.os.Bundle
import android.support.v7.app.AppCompatActivity
import android.util.Log
import android.widget.Toast

// 使用本包名里面的对应文件
import com.lzw.kotlinandjavademo.util.ResourcesUtils

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        init(this)
    }

    fun init(context: Context) {
        val statusBarHeight = ResourcesUtils.getStatusBarHeight(context)
        Toast.makeText(context, "状态栏高度为:$statusBarHeight 像素", Toast.LENGTH_SHORT).show()
        Log.e("tag", "状态栏高度为:$statusBarHeight 像素")
    }
}

工程目录结构如图所示:

(2)如果主工程是Java写的,部分是Kotlin写的。

比如Activity是Java写的,工具类是kotlin写的,实例代码如下:

package com.lzw.javaandkotlindemo;

import android.content.Context;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.Toast;

import com.lzw.javaandkotlindemo.util.ResourcesUtils;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        init(this);
    }

    private void init(Context context) {
        int statusBarHeight = ResourcesUtils.Companion.getStatusBarHeight(context);
        Toast.makeText(context, "状态栏高度为:" + statusBarHeight + "像素",Toast.LENGTH_SHORT).show();
    }
}

工程目录结构如图所示:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容