基于Kotlin Multiplatform实现静态文件服务器(三)

news2024/9/22 5:37:34

Expect 和 Actual

expect 关键字用于定义一个多平台通用的声明,即该声明在所有平台上都可用,并且需要在特定平台上实现。actual 关键字通常与 expect 关键字配合使用,用于定义多平台通用的接口和函数,从而允许在不同的平台上使用相同的 API。官方建议只对平台API使用expect/actual,否则使用普通接口。

比如获取IPv4地址:

 获取本机ip地址

这里使用的可能不是KMP的最佳实践,而是借鉴Android程序使用的MVVM模式。关于KMP项目的最佳实践,大家可以参考其他文档。

如前小节图例,在Android设备和PC设备上,获取IP地址的方式不同,因此业务侧的实现也会存在差异。

interface ServerViewModel { // 定义接口,用来统一各平台函数实现。
    fun getLocalIpAddressV4(address: MutableStateFlow<String>)
    ...
}

class ComServerViewModel {
    private val _ipAddress: MutableStateFlow<String> = MutableStateFlow("")
    val ipAddress = _ipAddress.asStateFlow() // 由状态流记录结果
    ...
    fun getLocalIpAddressV4() { // 界面通过viewmodel调用该函数
        // 调用expect函数获取各平台viewmodel实现对象
        getMsgViewModel().getLocalIpAddressV4(_ipAddress) 
    }
    ...
}

expect fun getMsgViewModel(): ServerViewModel  // expect函数,不同模块分别实现。

Android实现

Android获取ip地址需通过Connectivity获取网络状态,获取到ip地址。(考虑到文件传输对流量消耗的影响,仅支持在WiFi下,当然也可以使用流量,甚至通过花生壳等将文件映射到外网访问,但请注意隐私)。

class AndroidServerViewModel : ViewModel(), ServerViewModel {
    override fun getLocalIpAddressV4(address: MutableStateFlow<String>) {
        val request =
            NetworkRequest.Builder().addTransportType(NetworkCapabilities.TRANSPORT_WIFI)
                .build()
        val connectivityManager =
            FileServerApp.app.applicationContext.getSystemService(ConnectivityManager::class.java)
        connectivityManager.registerNetworkCallback( // 注册网络回调
            request,
            object : ConnectivityManager.NetworkCallback() {
                override fun onUnavailable() {
                    super.onUnavailable()
                    LogTool.i("Network is onUnavailable")
                }

                override fun onAvailable(network: Network) {
                    super.onAvailable(network)
                    LogTool.i("Network is onAvailable")
                }

                @RequiresApi(VERSION_CODES.Q)
                override fun onCapabilitiesChanged(
                    network: Network,
                    networkCapabilities: NetworkCapabilities
                ) {
                    super.onCapabilitiesChanged(network, networkCapabilities)
                    var ipAddress: String? = ""
                    if (Build.VERSION.SDK_INT > VERSION_CODES.Q) { // Q版本以上需通过LinkPropertiese获取。
                        ipAddress =
                            connectivityManager?.getLinkProperties(network)?.linkAddresses?.get(1)?.address?.address
                                ?.let { numericToTextFormat(it) }
                    } else { // Q版本以下通过WifiInfo获取,在Q版本以上被废弃,获取不到结果。
                        val wifiInfo = networkCapabilities.transportInfo as WifiInfo
                        ipAddress = intToIp(wifiInfo.ipAddress)
                    }
                    runBlocking { // 通过协程将结果发送出去。
                        address.emit(ipAddress ?: "")
                    }
                }
            })
        }
    private fun numericToTextFormat(src: ByteArray): String {
        return (src[0].toInt() and 0xff).toString() + "." + (src[1].toInt() and 0xff) + "." + (src[2].toInt() and 0xff) + "." + (src[3].toInt() and 0xff)
    }

    private fun intToIp(i: Int): String =
        ((i and 0xFF).toString() + " . " + ((i shr 8) and 0xFF) + " . "
            + ((i shr 16) and 0xFF) + " . " + ((i shr 24) and 0xFF))
    }
    ...
    
    // expect函数的实现,由actual关键字修饰。
    actual fun getMsgViewModel(): ServerViewModel = AndroidServerViewModel()

JVM实现

在JVM平台,直接通过java.net包中的Inet4Address获取ip地址即可。

class DesktopServerViewModel : ServerViewModel, ViewModel() {
    override fun getLocalIpAddressV4(address: MutableStateFlow<String>) {
        viewModelScope.launch { // 由于获取ip地址为耗时动作,因此启动协程执行。
            withContext(Dispatchers.IO) {
                Inet4Address.getLocalHost().hostAddress?.let {
                    address.emit(it) // 发送结果
                }
            }
        }
    }
    ...
}

actual fun getMsgViewModel(): ServerViewModel = DesktopServerViewModel()

配置信息读写

interface ServerViewModel {
    ...
    fun loadConfigs(config: MutableStateFlow<HttpFileServerConfig?>)

    fun updateConfigs(newConfig: HttpFileServerConfig)
    ...
}

class ComServerViewModel {
    ...
    private val _httpConfig: MutableStateFlow<HttpFileServerConfig?> = MutableStateFlow(null)
    val httpServerConfig = _httpConfig.asStateFlow()
    ...
    fun loadConfigs() {
        getMsgViewModel().loadConfigs(_httpConfig)
    }

    fun updateConfig(newConfig: HttpFileServerConfig) {
        _httpConfig.value?.serverPort = newConfig.serverPort
        getMsgViewModel().updateConfigs(newConfig)
    }
    ...
}
...
expect fun getMsgViewModel(): ServerViewModel

和获取ip地址一样,读写配置文件也用了不同平台不同的方式进行。在JVM平台,使用文件.conf存储;在Android平台,使用SharedPreference存储。

Android实现
class AndroidServerViewModel : ViewModel(), ServerViewModel {
    ...
    override fun loadConfigs(config: MutableStateFlow<HttpFileServerConfig?>) {
        fileServerConfig.serverPort = sp.getInt(PROP_KEY_SERVER_PORT, 8080)
        fileServerConfig.fileDirectory = sp.getString(
            PROP_KEY_SERVER_ROOT, getPlatform().getPlatformDefaultRoot()
        )!!
        LogTool.i(fileServerConfig.toString())
        runBlocking {
            config.emit(fileServerConfig)
        }
    }

    override fun updateConfigs(newConfig: HttpFileServerConfig) {
        fileServerConfig = newConfig.copy()
        sp.edit().apply {
            putInt(PROP_KEY_SERVER_PORT, fileServerConfig.serverPort)
            putString(PROP_KEY_SERVER_ROOT, getPlatform().getPlatformDefaultRoot())
        }.apply()
        LogTool.i(fileServerConfig)
    }
    ...

    private companion object {
        private val sp: SharedPreferences =
            FileServerApp.app.getSharedPreferences(PROP_FILE_NAME, Context.MODE_PRIVATE)
        private var fileServerConfig = HttpFileServerConfig()
    }
}

actual fun getMsgViewModel(): ServerViewModel = AndroidServerViewModel()

JVM实现

class DesktopServerViewModel : ServerViewModel, ViewModel() {
    ...
    override fun loadConfigs(config: MutableStateFlow<HttpFileServerConfig?>) {
        viewModelScope.launch {
            withContext(Dispatchers.IO) {
                try {
                    val configFile = File(PROP_FILE_NAME)
                    if (!configFile.exists()) {
                        configFile.createNewFile()
                    } else if (configFile.isDirectory) {
                        configFile.delete()
                        configFile.createNewFile()
                    } else {
                        LogTool.i("Config file is exist.")
                    }
                    prop.load(FileInputStream(PROP_FILE_NAME))
                    fileServerConfig.serverPort =
                        prop.getProperty(PROP_KEY_SERVER_PORT)?.toInt() ?: DEFAULT_SERVER_PORT
                    fileServerConfig.fileDirectory =
                        prop.getProperty(PROP_KEY_SERVER_ROOT) ?: DEFAULT_SERVER_ROOT
                    config.emit(fileServerConfig)
                    LogTool.i(config.toString())
                } catch (e: IOException) {
                    e.printStackTrace()
                }
            }
        }
    }

    override fun updateConfigs(newConfig: HttpFileServerConfig) {
        fileServerConfig = newConfig.copy()
        prop.setProperty(PROP_KEY_SERVER_PORT, fileServerConfig.serverPort.toString())
        prop.setProperty(PROP_KEY_SERVER_ROOT, fileServerConfig.fileDirectory)
    }
    ...

    private companion object {
        private const val PROP_FILE_NAME = ".conf"
        private val prop = Properties()
        private var fileServerConfig = HttpFileServerConfig()
    }
}

actual fun getMsgViewModel(): ServerViewModel = DesktopServerViewModel()

生成了一个apk,想用的可以体验一下,欢迎吐槽。 基于KMP的Android静态文件服务程序。

Exe和Deb的暂没环境,改天再打包。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2051188.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PyTorch--深度学习

onux部署功能 cpu运行时间 3. 自动求导 求导结果为&#xff1a;2 1 1

在java中前后端进行交互使用的内容

前言 本文将讲解在java前后端进行交互时会使用的内容, 过滤器 , 前后端交互时: 同步请求(了解)与异步请求, 后端响应json格式数据, 后端标准响应数据格式 过滤器 首先需要了解什么是过滤器: 过滤器是javaEE中在前向后端发送请求时进行拦截的技术,作用…

Linux系统编程(10)线程资源回收和互斥锁

一、pthread_cancel函数 pthread_cancel 函数用于请求取消一个线程。当调用 pthread_cancel 时&#xff0c;它会向指定的线程发送一个取消请求。 #include <pthread.h>int pthread_cancel(pthread_t thread);thread&#xff1a;要发送取消请求的线程标识符。 成功时&a…

函数递归,匿名、内置行数,模块和包,开发规范

一、递归与二分法 一&#xff09;递归 1、递归调用的定义 递归调用&#xff1a;在调用一个函数的过程中&#xff0c;直接或间接地调用了函数本身 2、递归分为两类&#xff1a;直接与间接 #直接 def func():print(from func)func()func() # 间接 def foo():print(from foo)bar…

mac安装ipd包【金铲铲为例】

mac安装ipd包 安装PlayCover 链接&#xff1a;https://github.com/PlayCover/PlayCover 1、点最新Releases 2、cmd ↓&#xff0c;拉到最下面下载dmg 3、安装 图标拖拽到Applications里 IPA下载 以金铲铲为例&#xff0c;良心砸壳包站点&#xff0c;有能力可以支持一下…

系列:水果甜度个人手持设备检测-行业法律法规

系列:水果甜度个人手持设备检测 --行业法律法规 背景 由于我们目标是制作针对水果的便携或手持式检测仪器&#xff0c;既然是民用产品&#xff0c;必然受一系列法律法规的约束&#xff0c;产品在上市之前将会受各种国家标准和地方标准的检验。本篇章中我们采用启发性搜索的方…

Python自动化:解锁高效工作与生产力的密钥

在当今快节奏的数字时代&#xff0c;自动化已成为提升工作效率、优化流程、减少人为错误的不可或缺的工具。Python&#xff0c;作为一种功能强大、易于学习且应用广泛的编程语言&#xff0c;在自动化领域扮演着举足轻重的角色。无论是数据处理、Web自动化、软件测试&#xff0c…

ETL数据集成丨将SQL Server数据同步至Oracle的具体实现

一、背景 在构建企业级数据架构时&#xff0c;将SQL Server数据库的数据同步至数仓数据库&#xff08;如Oracle&#xff09;是一项至关重要的任务。这一过程不仅促进了跨系统数据的一致性与可用性&#xff0c;还为数据分析、商业智能以及决策支持系统提供了坚实的数据基础。 …

黑神话悟空什么配置可以玩?什么样的游戏本配置可以畅玩《黑神话:悟空》?黑神话悟空电脑配置推荐

相信不少游戏爱好者&#xff0c;近期被《黑神话&#xff1a;悟空》这款游戏刷屏了&#xff0c;预售开启不到5分钟&#xff0c;所有的产品即宣告售罄&#xff0c;预购3天销售额就破亿&#xff0c;并迅速登顶Steam全球榜。作为一款备受期待的国产3A游戏&#xff0c;以其精美的画面…

IOS 10 统一颜色管理和适配深色模式

实现分析 像系统那样&#xff0c;给项目中常用的颜色取名字&#xff0c;这里使用扩展语法实现&#xff0c;好处是可以像访问系统颜色那样访问自定义的颜色。 添加依赖 为了能使用16进制的颜色值&#xff0c;这里通过依赖DynamicColor框架来实现 #颜色工具类 #https://githu…

美国洛杉矶大带宽服务器安全与权限

美国洛杉矶的大带宽服务器因其优越的地理位置、高速稳定的网络连接、丰富的资源以及强大的计算能力而受到众多企业和个人用户的青睐。尤其是在网络安全和权限管理方面&#xff0c;洛杉矶的大带宽服务器更是表现突出。下面我们就来详细了解一下这些服务器在安全与权限方面的特点…

Linux配置Maven环境

目录 一、Linux配置Maven环境1.1 挑选Maven版本1.2 下载到linux服务器1.3 解压1.4 配置环境变量1.5 测试 一、Linux配置Maven环境 1.1 挑选Maven版本 官网地址&#xff1a;https://maven.apache.org/download.cgi 这里以当前最新版本3.9.8为例 获取其对应的下载链接&#xf…

人工智能的新兴能力:我们是在追逐神话吗

图片由作者使用 DALL-E 拍摄 模型的涌现属性 突现属性不仅是人工智能的一个概念&#xff0c;也是所有学科&#xff08;从物理学到生物学&#xff09;的一个概念。这一概念一直让科学家着迷&#xff0c;他们既在描述这一概念&#xff0c;也在试图理解其起源。诺贝尔物理学奖得…

bootchart抓Android系统启动各阶段性能数据

最近在做Android系统启动优化&#xff0c;首要任务是找到启动过程中各阶段耗时点&#xff0c;进而有针对性地进行优化。主要用bootchart抓开机数据&#xff0c;本文主要记录下工具的使用方法。 1.抓开机数据 adb root adb shell ‘touch /data/bootchart/enabled’ adb rebo…

HDFS的透明加密

一、HDFS透明加密原理 Hadoop的透明加密(HDFS Transparent Data Encryption) - TDE 1.HDFS中的数据明文存储 HDFS中的数据会以block的形式保存在各台数据节点的本地磁盘中,但这些block都是明文的。 通过Web UI页面找到Block的ID和副本位于的机器信息 如果在操作系统中直接访…

2024年运营技术与网络安全态势研究报告:遭遇多次网络威胁的比例暴增

随着 OT 组织不断在其业务环境中集成各种数字工具和技术&#xff0c;它们面临的安全挑战也日益变得愈加复杂和多样化。正如 NIST 指出&#xff0c; “虽然安全解决方案旨在解决典型 IT系统中的一些问题&#xff0c;但将这些相同的解决方案引入不同的 OT 环境时&#xff0c;必须…

excel实现图片转文字功能/excel 实现导出图片功能/excel导出图片不失真(解决excel导出图片模糊的问题)

excel实现图片转文字功能 excel实现图片转文字功能&#xff1a;方法1&#xff1a;使用QQ的在线文档进行图片转文字方法2&#xff1a;使用WPS的excel文档进行图片转文字pdf图片转表格 使用excel 导出图片的方法&#xff08;使用Excel内置的“复制为图片”功能&#xff09;1. 复制…

Java序列化流和反序列化流

序列化流&#xff1a; 序列化&#xff1a;将一个对象转换成网络中传输的流 对象输出流&#xff1a;ObjectOutputStream 反序列化&#xff1a;将网络中传输的流还原成一个对象 对象输入流&#xff1a;ObjectInputStream 一个类对象将来…

轻松上手MYSQL:MYSQL权限配置全攻略,打造安全的数据库环境

​ &#x1f308; 个人主页&#xff1a;danci_ &#x1f525; 系列专栏&#xff1a;《设计模式》《MYSQL》 &#x1f4aa;&#x1f3fb; 制定明确可量化的目标&#xff0c;坚持默默的做事。 ✨欢迎加入探索MYSQL权限配置之旅✨ &#x1f44b; 大家好&#xff01;文本学习和…

【秋招笔试】8.14联想(算法岗)-三语言题解

🍭 大家好这里是 春秋招笔试突围,一起备战大厂笔试 💻 ACM金牌团队🏅️ | 多次AK大厂笔试 | 编程一对一辅导 ✨ 本系列打算持续跟新 春秋招笔试题 👏 感谢大家的订阅➕ 和 喜欢💗 和 手里的小花花🌸 ✨ 笔试合集传送们 -> 🧷春秋招笔试合集 🍒 本专栏已收…