Я пытаюсь вызвать функцию в Kotlin из Java.
Документация говорит, что это просто. Я до сих пор безуспешно следил за этим постом.

https://kotlinlang.org/docs/java-to-kotlin-interop.html#package-level-functions

При вводе на Java IDE распознает класс, но когда я компилирую, он говорит ошибка: не удается найти символ

Любая помощь более чем приветствуется.

App.kt

package org.cocos2dx.cpp

fun getTime() { /*...*/ }

AppActivity.java

    public void test1() {
        org.cocos2dx.cpp.AppKt.getTime();
    }

Структура проекта

/proj.android
   /app
       /src
          /org.cocos2dx.cpp
              app.kt
              AppActivity

Уровень проекта build.gradle

buildscript {
    ext.kotlin_version = "1.5.0"
    repositories {
        google()
        mavenCentral()
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:4.2.1'
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
        classpath 'com.google.gms:google-services:4.3.8'
        classpath 'com.google.firebase:firebase-crashlytics-gradle:2.7.0'
    }
}

allprojects {
    repositories {
        google()
        mavenCentral() 
    }
}

task clean(type: Delete) {
    delete rootProject.buildDir
}

Уровень модуля build.gradle

import org.gradle.internal.os.OperatingSystem

apply plugin: 'com.android.application'
apply plugin: 'com.google.gms.google-services'
apply plugin: 'com.google.firebase.crashlytics'

android {
    compileSdkVersion PROP_COMPILE_SDK_VERSION.toInteger()

    defaultConfig {
        applicationId "demo.1"
        minSdkVersion PROP_MIN_SDK_VERSION
        targetSdkVersion PROP_TARGET_SDK_VERSION
        versionCode 43
        versionName "0.43"

        externalNativeBuild {
            cmake {
                targets 'MyGame'
                arguments "-DCMAKE_FIND_ROOT_PATH=", "-DANDROID_STL=c++_static", "-DANDROID_TOOLCHAIN=clang", "-DANDROID_ARM_NEON=TRUE"
                cppFlags "-frtti -fexceptions -fsigned-char"
            }
        }

        ndk {
            abiFilters = []
            abiFilters.addAll(PROP_APP_ABI.split(':').collect{it as String})
        }
    }

    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }

    sourceSets.main {
        java.srcDir "src"
        res.srcDir "res"
        manifest.srcFile "AndroidManifest.xml"
        assets.srcDir "../../Resources"
    }

    externalNativeBuild {
        cmake {
            path "../../CMakeLists.txt"
        }
    }

    signingConfigs {
       release {
            if (project.hasProperty("RELEASE_STORE_FILE")) {
                storeFile file(RELEASE_STORE_FILE)
                storePassword RELEASE_STORE_PASSWORD
                keyAlias RELEASE_KEY_ALIAS
                keyPassword RELEASE_KEY_PASSWORD
            }
        }
    }

    buildTypes {
        release {
            debuggable false
            jniDebuggable false
            renderscriptDebuggable false
            minifyEnabled true
            shrinkResources true
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
            if (project.hasProperty("RELEASE_STORE_FILE")) {
                signingConfig signingConfigs.release
            }
        }

        debug {
            debuggable true
            jniDebuggable true
            renderscriptDebuggable true
        }
    }
}

android.applicationVariants.all { variant ->
    def project_root_folder = "${projectDir}/../.."
    def dest_assets_folder = "${projectDir}/assets"

    delete dest_assets_folder
    def targetName = variant.name.capitalize()
    def copyTaskName = "copy${targetName}ResourcesToAssets"
    
    tasks.register(copyTaskName) {
        copy {
            from "${buildDir}/../../../Resources"
            into "${buildDir}/intermediates/assets/${variant.dirName}"
            exclude "**/*.gz"
        }
    }
    tasks.getByName("pre${targetName}Build").dependsOn copyTaskName
}

dependencies {
    implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation project(':libcocos2dx')
    implementation 'androidx.core:core:1.0.0'
    implementation platform('com.google.firebase:firebase-bom:28.1.0')
    implementation 'com.google.firebase:firebase-crashlytics-ktx'
    implementation 'com.google.firebase:firebase-analytics-ktx'
}

Ошибка

error: cannot find symbol
        org.cocos2dx.cpp.AppKt.getTime();
                        ^
  symbol:   class AppKt
  location: package org.cocos2dx.cpp
0
Marcela Nievas 2 Июл 2021 в 22:04

3 ответа

Лучший ответ

Ваш код выглядит нормально; но вам нужен путь класса уровня проекта build.gradle :

classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"

buildscript {
    ext.kotlin_version = "1.5.0"
    repositories {
        google()
        mavenCentral()
    }
    dependencies {
        classpath "com.android.tools.build:gradle:4.2.0"
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"

    }
}

И зависимости стандартной библиотеки Kotlin на уровне модуля build.gradle :

dependencies {
    implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version" 
}
1
Zain 3 Июл 2021 в 18:31

Вы можете решить эту проблему, переместив файл java в каталог с исходными кодами java, а затем сохранив файл kotlin kt в источниках kotlin.

После этого ваша файловая структура может выглядеть примерно так:

proj.android/
├─ app/
│  ├─ src/
│  │  ├─ kotlin/
│  │  │  ├─ org.cocos2dx.cpp/
│  │  │  │  ├─ app.kt
│  │  ├─ java/
│  │  │  ├─ org.cocos2dx.cpp/
│  │  │  │  ├─ AppActivity

В мультипроектах (с использованием JPMS) мне дополнительно пришлось добавить

compileKotlin.destinationDir = compileJava.destinationDir

В мой build.gradle.

0
Dharman 2 Июл 2021 в 20:46

Попробуйте использовать @JVMStatic в функции kotlin.

@JVMStatic
fun getTime() { /*...*/ }

Из JAVA

public void test1() {
   AppKt.getTime();
}

Это должно сработать.

0
Eishon 7 Июл 2021 в 01:27