当前位置: 首页 > news >正文

Kotlin Multiplatform 创建多平台分发库

目标:通过本教程学习如何使用 Kotlin Multiplatform Library 创建多平台分发库(iOS,安卓)。

创建一个项目

1、本教程使用的是Android Studio创建
2、选择 新建工程,选择 Kotlin Multiplatform Library
3、点击next 输入需要创建的项目名称以及存放的位置
4、点击next,输入要创建的库的名称,默认shared
建议都选上 测试单元。
在这里插入图片描述

在这里插入图片描述

到这项目基本创建完毕,开始进入编写分发库代码阶段。

编写跨平台代码

定义要在公共代码中实现的类和接口
1、在公共模块中创建一个目录 org.jetbrains.base64 用来存放编写代码
2、在新建目录下面创建 Base64.kt 文件
3、在新创建的的文件中定义 Base64Encoder 将字节转换为 Base64格式的接口

package org.jetbrains.base64interface Base64Encoder {fun encode(src: ByteArray): ByteArray
}

4、定义对象 Base64Factory 作为公共代码 以提供接口实现实例 Base64Encoder的方式

expect object Base64Factory {fun createEncoder(): Base64Encoder
}

expext 是在跨平台代码中用到的关键字,被expext 关键字修饰的Base64Factory对象 。对于需要分发的平台,需要使用关键字 actual 关键字予以实现。

此时项目应该会爆红,因为在公共模块中使用关键字 expect 创建了对象,对应的分发平台还没实现,这时需要一一进行实现

提供对应平台代码实现

安卓平台
1、找到安卓模块在模块下面创建一个 org.jetbrains.base64 的新包
2、在新包里面创建 Base64.kt 文件
3、使用关键字 actual 实现公共模块声明的 Base64Factory对象方法

package com.example.myapplication.org.jetbrains.base64import android.annotation.TargetApi
import android.os.Build
import java.util.*actual object Base64Factory {actual fun createEncoder(): Base64Encoder = JvmBase64Encoder
}object JvmBase64Encoder : Base64Encoder {@TargetApi(Build.VERSION_CODES.O)override fun encode(src: ByteArray): ByteArray = Base64.getEncoder().encode(src)}

写到这安卓的代码以好,应为安卓可以直接使用api进行转换。

iOS平台

iOS在这方面比较吃亏没有现成的东西可以用,需要自己实现。
1、找到iOS模块,在模块下面创建一个org.jetbrains.base64包
2、在新包下面创建一个 Base64.kt文件
3、实现公共模块的代码

package com.example.myapplication.org.jetbrains.base64private val BASE64_ALPHABET: String = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
private val BASE64_MASK: Byte = 0x3f
private val BASE64_PAD: Char = '='
private val BASE64_INVERSE_ALPHABET = IntArray(256) {BASE64_ALPHABET.indexOf(it.toChar())
}private fun Int.toBase64(): Char = BASE64_ALPHABET[this]actual object Base64Factory {actual fun createEncoder(): Base64Encoder = NativeBase64Encoder
}object NativeBase64Encoder : Base64Encoder {// encode(src: ByteArray): ByteArrayoverride fun encode(src: ByteArray): ByteArray {fun ByteArray.getOrZero(index: Int): Int = if (index >= size) 0 else get(index).toInt()// 4n / 3 is expected Base64 payloadval result = ArrayList<Byte>(4 * src.size / 3)var index = 0while (index < src.size) {val symbolsLeft = src.size - indexval padSize = if (symbolsLeft >= 3) 0 else (3 - symbolsLeft) * 8 / 6val chunk = (src.getOrZero(index) shl 16) or (src.getOrZero(index + 1) shl 8) or src.getOrZero(index + 2)index += 3for (i in 3 downTo padSize) {val char = (chunk shr (6 * i)) and BASE64_MASK.toInt()result.add(char.toBase64().code.toByte())}// Fill the pad with '='repeat(padSize) { result.add(BASE64_PAD.code.toByte()) }}return result.toByteArray()}}

到此,公共模块,安卓模块以及iOS模块代码已经编写完毕。可以在测试类中测试刚才编写的代码。

在先写之前,将 encodeToString 方法添加到 默认实现Base64Encoder 的接口中,该方法 将字节数组转换成字符串,方便测试。

interface Base64Encoder {fun encode(src: ByteArray): ByteArrayfun encodeToString(src: ByteArray): String {val encoded = encode(src)return buildString(encoded.size) {encoded.forEach { append(it.toInt().toChar()) }}}}

公共模块

object JvmBase64Encoder : Base64Encoder {@TargetApi(Build.VERSION_CODES.O)override fun encode(src: ByteArray): ByteArray = Base64.getEncoder().encode(src)@TargetApi(Build.VERSION_CODES.O)override fun encodeToString(src: ByteArray): String = Base64.getEncoder().encodeToString(src)
}

安卓模块

object NativeBase64Encoder : Base64Encoder {// encode(src: ByteArray): ByteArrayoverride fun encode(src: ByteArray): ByteArray {fun ByteArray.getOrZero(index: Int): Int = if (index >= size) 0 else get(index).toInt()// 4n / 3 is expected Base64 payloadval result = ArrayList<Byte>(4 * src.size / 3)var index = 0while (index < src.size) {val symbolsLeft = src.size - indexval padSize = if (symbolsLeft >= 3) 0 else (3 - symbolsLeft) * 8 / 6val chunk = (src.getOrZero(index) shl 16) or (src.getOrZero(index + 1) shl 8) or src.getOrZero(index + 2)index += 3for (i in 3 downTo padSize) {val char = (chunk shr (6 * i)) and BASE64_MASK.toInt()result.add(char.toBase64().code.toByte())}// Fill the pad with '='repeat(padSize) { result.add(BASE64_PAD.code.toByte()) }}return result.toByteArray()}// encodeToString(src: ByteArray): Stringoverride fun encodeToString(src: ByteArray): String {return super.encodeToString(src)}}

iOS 模块

至此,用于测试转换以及各模块代码已经编写完毕。剩下的就是便携测试模块代码。

http://www.lryc.cn/news/98202.html

相关文章:

  • [SQL挖掘机] - union/union all 使用注意事项
  • php 单例模式
  • 【数据结构】实验二:顺序表
  • 【高级数据结构】线段树
  • qt简易闹钟
  • python和c加加有什么区别,c和c++和python先学哪个
  • Visual Studio 2022 cmake配置opencv开发环境
  • C++ GDAL找出多时相遥感影像缺失的日期并自动生成新的全零图像作为替补
  • 【AI底层逻辑】——篇章5(下):机器学习算法之聚类降维时间序列
  • P1980 [NOIP2013 普及组] 计数问题
  • 需求管理全过程流程图及各阶段核心关注点详解
  • Android开源 自定义emoji键盘,EmojiPack v2.1版本
  • SOLIDWORKS软件的优势分析 硕迪科技
  • Android性能优化之游戏的Theme背景图
  • 网络安全(黑客)系统自学,成为一名白帽黑客
  • lua学习-2 常见运算符
  • 【图像处理】使用 OpenCV 将您的照片变成卡通
  • 暖手宝UL认证 亚马逊UL测试报告 UL499测试项目
  • ES6模块化与异步编程高级用法
  • spring-cloud-starter-gateway 4.0.6负载均衡失败
  • Tomcat注册为Windows服务
  • 【Maven】Maven 中 pom.xml 文件
  • 2、Linux驱动开发:模块_引用符号
  • Python web实战 | Docker+Nginx部署python Django Web项目详细步骤【干货】
  • 【uniapp】实现买定离手小游戏
  • 【vim 学习系列文章 3 - vim 选中、删除、复制、修改引号或括号内的内容】
  • webpack联邦模块介绍及在dumi中使用问题整理
  • 记录一下Kotlin: Module was compiled with an incompatible version of Kotlin.的问题
  • html中使用Vue+element UI动态创建表单数据不显示问题
  • CentOS下 Docker、Docker Compose 的安装教程