C – 튜토리얼에서 문자열 매핑
이 튜토리얼은 Kotlin과 C 매핑 시리즈의 마지막 부분입니다. 계속하기 전에 이전 단계를 완료했는지 확인하세요.
C로부터 기본 데이터 타입 매핑하기
C로부터 구조체와 공용체 타입 매핑하기
함수 포인터 매핑하기
C로부터 문자열 매핑하기
C 라이브러리 임포트는 Experimental입니다. cinterop 툴에 의해 C 라이브러리로부터 생성된 모든 Kotlin 선언은 @ExperimentalForeignApi
어노테이션을 가져야 합니다.
Kotlin/Native와 함께 제공되는 네이티브 플랫폼 라이브러리(예: Foundation, UIKit, POSIX)는 일부 API에 대해서만 옵트인을 요구합니다.
이 시리즈의 마지막 부분에서는 Kotlin/Native에서 C 문자열을 다루는 방법을 살펴보겠습니다.
이 튜토리얼에서는 다음 방법을 배웁니다.
C 문자열 작업
C에는 전용 문자열 타입이 없습니다. 메서드 시그니처 또는 문서를 통해 주어진 char *
가 특정 컨텍스트에서 C 문자열을 나타내는지 확인할 수 있습니다.
C 언어의 문자열은 null-terminated이므로, 문자열의 끝을 표시하기 위해 바이트 시퀀스의 끝에 후행 제로 문자 \0
가 추가됩니다. 일반적으로 UTF-8 인코딩 문자열이 사용됩니다. UTF-8 인코딩은 가변 너비 문자를 사용하며 ASCII와 하위 호환됩니다. Kotlin/Native는 기본적으로 UTF-8 문자 인코딩을 사용합니다.
Kotlin과 C 간에 문자열이 어떻게 매핑되는지 이해하려면 먼저 라이브러리 헤더를 만드십시오. 시리즈의 첫 번째 부분에서 필요한 파일로 C 라이브러리를 이미 만들었습니다. 이 단계를 위해:
-
C 문자열로 작동하는 다음 함수 선언으로
lib.h
파일을 업데이트하십시오.#ifndef LIB2_H_INCLUDED
#define LIB2_H_INCLUDED
void pass_string(char* str);
char* return_string();
int copy_string(char* str, int size);
#endif이 예제는 C 언어에서 문자열을 전달하거나 받는 일반적인 방법을 보여줍니다.
return_string()
함수의 반환 값을 주의해서 처리하십시오. 반환된char*
를 해제하려면 올바른free()
함수를 사용해야 합니다. -
---
구분자 뒤에interop.def
파일의 선언을 업데이트합니다.---
void pass_string(char* str) {
}
char* return_string() {
return "C string";
}
int copy_string(char* str, int size) {
*str++ = 'C';
*str++ = ' ';
*str++ = 'K';
*str++ = '/';
*str++ = 'N';
*str++ = 0;
return 0;
}
interop.def
파일은 애플리케이션을 컴파일, 실행 또는 IDE에서 열기 위해 필요한 모든 것을 제공합니다.
C 라이브러리에 대해 생성된 Kotlin API 검사
C 문자열 선언이 Kotlin/Native로 어떻게 매핑되는지 살펴봅시다.
-
src/nativeMain/kotlin
에서 이전 튜토리얼의hello.kt
파일을 다음 내용으로 업데이트합니다.import interop.*
import kotlinx.cinterop.ExperimentalForeignApi
@OptIn(ExperimentalForeignApi::class)
fun main() {
println("Hello Kotlin/Native!")
pass_string(/*fix me*/)
val useMe = return_string()
val useMe2 = copy_string(/*fix me*/)
} -
IntelliJ IDEA의 Go to declaration 명령 (
Cmd + B /Ctrl + B )을 사용하여 C 함수에 대해 생성된 다음 API로 이동합니다.fun pass_string(str: kotlinx.cinterop.CValuesRef<kotlinx.cinterop.ByteVarOf<kotlin.Byte> /* from: kotlinx.cinterop.ByteVar */>?)
fun return_string(): kotlinx.cinterop.CPointer<kotlinx.cinterop.ByteVarOf<kotlin.Byte> /* from: kotlinx.cinterop.ByteVar */>?
fun copy_string(str: kotlinx.cinterop.CValuesRef<kotlinx.cinterop.ByteVarOf<kotlin.Byte> /* from: kotlinx.cinterop.ByteVar */>?, size: kotlin.Int): kotlin.Int
이러한 선언은 간단합니다. Kotlin에서 C char *
포인터는 매개변수의 경우 str: CValuesRef<ByteVarOf>?
로, 반환 타입의 경우 CPointer<ByteVarOf>?
로 매핑됩니다. Kotlin은 char
타입을 일반적으로 8비트 부호 있는 값인 kotlin.Byte
로 나타냅니다.
생성된 Kotlin 선언에서 str
은 CValuesRef<ByteVarOf<Byte>>?
로 정의됩니다.
이 타입은 nullable이므로 null
을 인자 값으로 전달할 수 있습니다.
Kotlin 문자열을 C로 전달하기
Kotlin에서 API를 사용해 봅시다. 먼저 pass_string()
함수를 호출합니다.
import interop.*
import kotlinx.cinterop.ExperimentalForeignApi
import kotlinx.cinterop.cstr
@OptIn(ExperimentalForeignApi::class)
fun passStringToC() {
val str = "This is a Kotlin string"
pass_string(str.cstr)
}
String.cstr
extension property 덕분에 Kotlin 문자열을 C로 전달하는 것은 간단합니다.
UTF-16 문자가 포함된 경우를 위한 String.wcstr
속성도 있습니다.
Kotlin에서 C 문자열 읽기
이제 return_string()
함수에서 반환된 char *
를 가져와서 Kotlin 문자열로 변환합니다.
import interop.*
import kotlinx.cinterop.ExperimentalForeignApi
import kotlinx.cinterop.toKString
@OptIn(ExperimentalForeignApi::class)
fun passStringToC() {
val stringFromC = return_string()?.toKString()
println("Returned from C: $stringFromC")
}
여기서 .toKString()
확장 함수는 return_string()
함수에서 반환된 C 문자열을 Kotlin 문자열로 변환합니다.
Kotlin은 C char *
문자열을 Kotlin 문자열로 변환하기 위한 여러 확장 함수를 제공하며,
인코딩에 따라 다릅니다.
fun CPointer<ByteVarOf<Byte>>.toKString(): String // UTF-8 문자열을 위한 표준 함수
fun CPointer<ByteVarOf<Byte>>.toKStringFromUtf8(): String // 명시적으로 UTF-8 문자열을 변환합니다.
fun CPointer<ShortVarOf<Short>>.toKStringFromUtf16(): String // UTF-16 인코딩된 문자열을 변환합니다.
fun CPointer<IntVarOf<Int>>.toKStringFromUtf32(): String // UTF-32 인코딩된 문자열을 변환합니다.
Kotlin에서 C 문자열 바이트 받기
이번에는 copy_string()
C 함수를 사용하여 지정된 버퍼에 C 문자열을 씁니다. 이 함수는 문자열을 써야 하는 메모리 위치에 대한 포인터와 허용된 버퍼 크기의 두 가지 인수를 사용합니다.
또한 함수는 성공 또는 실패 여부를 나타내는 값을 반환해야 합니다. 0
은 성공을 의미하고 제공된 버퍼가 충분히 크다고 가정합니다.
import interop.*
import kotlinx.cinterop.ExperimentalForeignApi
import kotlinx.cinterop.addressOf
import kotlinx.cinterop.usePinned
@OptIn(ExperimentalForeignApi::class)
fun sendString() {
val buf = ByteArray(255)
buf.usePinned { pinned `->`
if (copy_string(pinned.addressOf(0), buf.size - 1) != 0) {
throw Error("Failed to read string from C")
}
}
val copiedStringFromC = buf.decodeToString()
println("Message from C: $copiedStringFromC")
}
여기서 네이티브 포인터가 먼저 C 함수에 전달됩니다. .usePinned
확장 함수는 바이트 배열의 네이티브 메모리 주소를 일시적으로 고정합니다. C 함수는 바이트 배열을 데이터로 채웁니다. 다른 확장 함수인 ByteArray.decodeToString()
은 UTF-8 인코딩을 가정하여 바이트 배열을 Kotlin 문자열로 변환합니다.
Kotlin 코드 업데이트
이제 Kotlin 코드에서 C 선언을 사용하는 방법을 배웠으므로 프로젝트에서 사용해 보십시오.
최종 hello.kt
파일의 코드는 다음과 같을 수 있습니다.
import interop.*
import kotlinx.cinterop.*
@OptIn(ExperimentalForeignApi::class)
fun main() {
println("Hello Kotlin/Native!")
val str = "This is a Kotlin string"
pass_string(str.cstr)
val useMe = return_string()?.toKString() ?: error("null pointer returned")
println(useMe)
val copyFromC = ByteArray(255).usePinned { pinned `->`
val useMe2 = copy_string(pinned.addressOf(0), pinned.get().size - 1)
if (useMe2 != 0) throw Error("Failed to read a string from C")
pinned.get().decodeToString()
}
println(copyFromC)
}
모든 것이 예상대로 작동하는지 확인하려면 IDE에서 runDebugExecutableNative
Gradle 작업을 실행하거나
다음 명령을 사용하여 코드를 실행합니다.
./gradlew runDebugExecutableNative
다음 단계
더 고급 시나리오를 다루는 C와의 상호 운용성 문서에서 자세히 알아보십시오.