详谈Android进程间的大数据通信机制:LocalSocket

news2025/2/28 19:53:09

在这里插入图片描述

前言

说起Android进行间通信,大家第一时间会想到AIDL,但是由于Binder机制的限制,AIDL无法传输超大数据。

比如我们在之前文章《WebRtc中是如何处理视频数据的?》提到的我们可以得到WebRtc的视频数据,这时候我们如果有一个需求,希望将这些视频数据传送给另外一个APP呢?这个数据量就会很大。

那么我们如何在进程间传输大数据呢?

Android中给我们提供了另外一个机制:LocalSocket

它会在本地创建一个socket通道来进行数据传输。

那么它怎么使用?

首先我们需要两个应用:客户端和服务端

初始化LocalSocket

服务端初始化

override fun run() {
    server = LocalServerSocket("xxxx")
    remoteSocket = server?.accept()
    ...
}

先创建一个LocalServerSocket服务,参数是服务名,注意这个服务名需要唯一,这是两端连接的依据。

然后调用accept函数进行等待客户端连接,这个函数是block线程的,所以例子中另起线程。

当客户端发起连接后,accept就会返回LocalSocket对象,然后就可以进行传输数据了。

客户端初始化

var localSocket = LocalSocket()
localSocket.connect(LocalSocketAddress("xxxx"))

首先创建一个LocalSocket对象

然后创建一个LocalSocketAddress对象,参数是服务名

然后调用connect函数连接到该服务即可。就可以使用这个socket传输数据了。

数据传输

两端的socket对象是一个类,所以两端的发送和接受代码逻辑一致。

通过localSocket.inputStreamlocalSocket.outputStream可以获取到输入输出流,通过对流的读写进行数据传输。

注意,读写流的时候一定要新开线程处理。

因为socket是双向的,所以两端都可以进行收发,即读写

发送数据

代码如下:

var pool = Executors.newSingleThreadExecutor()
var runnable = Runnable {
	try {
		var out = xxxxSocket.outputStream
		out.write(data)
		out.flush()
    } catch (e: Throwable) {
        Log.e("xxx", "xxx", e)
    }
}
pool.execute(runnable)

发送数据是主动动作,每次发送都需要另开线程,所以如果是多次,我们需要使用一个线程池来进行管理

如果需要多次发送数据,可以将其进行封装成一个函数

接收数据

接收数据实际上是进行while循环,循环进行读取数据,这个最好在连接成功后就开始,比如客户端代码如下:

localSocket.connect(LocalSocketAddress("xxx"))
var runnable = Runnable {
    while (localSocket.isConnected){
		var input = localSocket.inputStream
		input.read(data)
        ...
    }
}
Thread(runnable).start()

接收数据实际上是一个while循环不停的进行读取,未读到数据就继续循环,读到数据就进行处理再循环,所以这里只另开一个线程即可,不需要线程池。

传输复杂数据

上面只是简单事例,无法传输复杂数据,如果要传输复杂数据,就需要使用DataInputStreamDataOutputStream

首先需要定义一套协议。

比如定义一个简单的协议:传输的数据分两部分,第一部分是一个int值,表示后面byte数据的长度;第二部分就是byte数据。这样就知道如何进行读写

复杂数据写入

复杂数据写入代码如下:

var pool = Executors.newSingleThreadExecutor()
var out = DataOutputStream(xxxSocket.outputStream)
var runnable = Runnable {
	try {
		out.writeInt(data.size)
		out.write(data)
		out.flush()
    } catch (e: Throwable) {
        Log.e("xxx", "xxx", e)
    }
}
pool.execute(runnable)

先写入数据块的大小,再写入数据。

复杂数据读取

上面数据的读取代码如下:

var runnable = Runnable {
var input = DataInputStream(xxxSocket.inputStream)
var outArray = ByteArrayOutputStream()
    while (true) {
        outArray.reset()
		var length = input.readInt()
		if(length > 0) {
			var buffer = ByteArray(length)
			input.read(buffer)
            ...
        }
    }

}
Thread(runnable).start()

先读取一个int,这是接下来数据块的大小,然后直接读取一个这个长度的数据,这样就会把这部分数据完整读取出来。这样当读取下一部份数据的时候正好是开头。

这样就可以传输复杂数据,不会导致数据错乱。

传输超大数据

上面虽然可以传输复杂数据,但是当我们的数据过大的时候,也会出现问题。

比如传输图片或视频,假设byte数据长度达到1228800,这时我们通过

var buffer = ByteArray(1228800)
input.read(buffer)

无法读取到所有数据,只能读到一部分。而且会造成后面数据的混乱,因为读取位置错位了。

读取的长度大约是65535个字节,这是因为TCP被IP包包着,也会有包大小限制65535。

但是注意!写数据的时候如果数据过大就会自动进行分包,但是读数据的时候如果一次读取貌似无法跨包,这样就导致了上面的结果,只能读一个包,后面的就错乱了。

那么这种超大数据该如何传输呢,我们用循环将其一点点写入,也一点点读出,并根据结果不断的修正偏移。代码:

超大数据写入

写入超大数据代码如下:

var pool = Executors.newSingleThreadExecutor()
var out = DataOutputStream(xxxSocket.outputStream)
var runnable = Runnable {
	try {
		out.writeInt(data.size)
		var offset = 0
		while ((offset + 1024) <= data.size) {
		 	out.write(data, offset, 1024)
	        offset += 1024
	    }
		out.write(data, offset, data.size - offset)
		out.flush()
    } catch (e: Throwable) {
        Log.e("xxxx", "xxxx", e)
    }

}

pool.execute(runnable)

同样先写入数据大小,然后循环写入1024大小的数据,记录偏移,最后不足的部分一次性写入即可。

超大数据读取

上面的数据读取代码如下:

var input = DataInputStream(xxxSocket.inputStream)
var runnable = Runnable {
var outArray = ByteArrayOutputStream()
    while (true) {
        outArray.reset()
		var length = input.readInt()
		if(length > 0) {
			var buffer = ByteArray(1024)
			var total = 0
            while (total + 1024 <= length) {
				var count = input.read(buffer)
                outArray.write(buffer, 0, count)
                total += count
            }
			var buffer2 = ByteArray(length - total)
			input.read(buffer2)
            outArray.write(buffer2)
			var result = outArray.toByteArray()
            ...
        }
    }
}
Thread(runnable).start()

同样先读取一个int,即数据大小。然后循环读取1024大小的数据,直到不足的时候将剩余部分一起读取出来即可。这样这部分数据就完整读取出来了,然后读取下一部分数据正好在开头。

这样可以避免因为分包而导致读取的长度不匹配的问题

总结

LocalSocket是基于Linux底层IPC通信机制(UNIX-domain socket)的,因此相对于网络通信使用的socket不需要经过网络协议栈,不需要打包拆包、计算校验。所以LocalSocket的效率是很高的。

网上有人进行过测试,结果如下:
在这里插入图片描述
可以看到LocalSocket的传输接近AIDL,所以还是非常高效的。

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

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

相关文章

C++类和对象(4)

C类和对象 1.拷贝构造函数1.1 概念1.2. 特征1.2.1. 拷贝构造函数构造函数的一种重载形式&#xff1b;1.2.2. 拷贝构造函数的参数只能有一个&#xff0c;是对本类类型对象的引用&#xff0c;不能是传值调用&#xff0c;编译会直接报错&#xff0c;或者是直接进入死循环。1.2.3. …

wvp开发环境搭建

代码下载地址 代码下载地址 https://gitee.com/pan648540858/wvp-GB28181-pro.git 开发工具 采用jetbrain idea 利用开发工具下载代码 文件-新建-来自版本控制的项目 url是上面的代码下载链接&#xff0c;点击克隆即可 下图是已经克隆并打开的代码 安装依赖环境 安装redi…

基于html+css的图展示44

准备项目 项目开发工具 Visual Studio Code 1.44.2 版本: 1.44.2 提交: ff915844119ce9485abfe8aa9076ec76b5300ddd 日期: 2020-04-16T16:36:23.138Z Electron: 7.1.11 Chrome: 78.0.3904.130 Node.js: 12.8.1 V8: 7.8.279.23-electron.0 OS: Windows_NT x64 10.0.19044 项目…

Linux套接字编程

在上一篇博客中我们对网络中一些基本概念进行了简单阐述&#xff0c;这一篇博客我们来对套接字编程的内容进行初步了解。 目录 1.引入 2.UDP协议 2.1通信两端流程 2.1.1服务端流程 2.1.2客户端流程 2.2套接字相关操作接口 2.2.1创建套接字 2.2.2为套接字绑定地址信息 …

SSL证书周期变为90天? 锐成让您轻松应对行业新规

3月3日&#xff0c;谷歌在其“Move Forward, Together”栏目中&#xff0c;称已向CA/B论坛发起了投票提案&#xff0c;建议将公共TLS&#xff08;也称为SSL&#xff09;证书的最长有效期从398天减少到90天。值得注意的是&#xff0c;即便CA/B论坛没有通过这一提议&#xff0c;谷…

【C语言】函数讲解(下)

【C语言】函数讲解&#xff08;下&#xff09; 1.函数的声明和定义1.1函数声明1.2函数定义 2.函数的嵌套调用和链式访问2.1嵌套调用2.2链式访问 3.函数递归3.1什么是递归3.2递归的两个必要条件3.2.1练习13.2.2练习2 3.3递归与迭代3.3.1练习13.3.2练习2 所属专栏&#xff1a;C语…

Android Jetpack—LiveData

1.LiveData LiveData是Android Jetpack包提供的一种可观察的数据存储器类&#xff0c;它可以通过添加观察者被其他组件观察其变更。不同于普通的观察者&#xff0c;LiveData最重要的特征是它具有生命周期感知能力&#xff0c;它遵循其他应用组件&#xff08;如 Activity、Frag…

软件测试—进阶篇

软件测试—进阶篇 &#x1f50e;根据测试对象划分界面测试可靠性测试容错性测试文档测试兼容性测试易用性测试安装卸载测试安全性测试性能测试内存泄漏测试 &#x1f50e;根据是否查看代码划分黑盒测试白盒测试灰盒测试 &#x1f50e;根据开发阶段划分单元测试集成测试系统测试…

mulesoft MCIA 破釜沉舟备考 2023.04.27.25 (易错题)

@[TOC](mulesoft MCIA 破釜沉舟备考 2023.04.27.25 (易错题)) 1. According to MuleSoft, which deployment characteristic applies to a microservices application architecture? A. Services exist as independent deployment artifacts and can be scaled independently…

ABeam Insight | 智能制造系列(6):虚拟/增强现实(VR/AR)×智能制造

虚拟现实&#xff08;VR&#xff09;和增强现实&#xff08;AR&#xff09;的概念早在20世纪60年代就被提出&#xff0c;但由于当时的技术水平无法满足相关应用的需求&#xff0c;这些概念并没有引起广泛关注。直到近年来随着计算机技术的飞速发展&#xff0c;虚拟现实和增强现…

python+nodejs+php+springboot+vue高校教室自习室预约管理系统

建立的自习室预约管理系统用户使用浏览器就可以对其进行访问&#xff0c;管理员在操作上面能够方便管理&#xff0c;因此用户和管理员能够方便对这个系统进行操作。论文全面介绍系统数据库&#xff0c;功能设计和业务流程设计。数据库能够存储自习室预约管理系统需要的数据。 …

Leanback(1)-播放控制栏下添加新的行

我们要在播放控制栏下面加入下面一行。 这个就是标准的row。 leanback的原理 Android Leanback结构源码简析 - 简书 我们知道Row用来提供数据&#xff0c;row可以通过一个ObjectAdapter来管理和提供数据 我们知道presenter是一个负责将数据绑定到视图上的对象&#xff0c;它可以…

基于STM32的智能语音垃圾桶设计

一. 系统设计及框图&#xff1a; 本设计整体功能如下&#xff1a; 1. 超声波感应到有人靠近时语音提示“垃圾放置请分类”。 2. 检测垃圾筒时是否满&#xff0c;当满时语音提示“垃圾桶已满”。 3. 光传感器检测&#xff0c;指示灯指示。 4. 语音识别不同的垃圾类型。 二.…

前端程序员的职业发展规划与路线——ChatGPT的回答

文章目录 一、前端程序员的职业规划是&#xff1f;回答1&#xff1a; 作为一个前端开发程序员&#xff0c;您的职业发展路线可能如下&#xff1a;回答2&#xff1a;作为前端开发程序员&#xff0c;您的职业发展路线可能如下&#xff1a;回答3&#xff1a; 你的职业发展路线可能…

ASEMI代理ADI亚德诺ADM3051CRZ-REEL7车规级芯片

编辑-Z ADM3051CRZ-REEL7芯片参数&#xff1a; 型号&#xff1a;ADM3051CRZ-REEL7 显性状态&#xff1a;78 mA 隐性状态&#xff1a;10 mA 待命状态&#xff1a;275μA CANH输出电压&#xff1a;4.5V CANL输出电压&#xff1a;2V 差动输出电压&#xff1a;3V 输入电压…

【AI生产力工具】Upscale.media:用AI技术提升照片质量,让你的作品更出色

文章目录 简介一、Upscale.media是什么&#xff1f;二、如何使用Upscale.media&#xff1f;三、总结 简介 在如今的数字时代&#xff0c;图片已经成为我们日常生活中不可或缺的一部分&#xff0c;从社交媒体到电子商务网站&#xff0c;从广告宣传到个人生活&#xff0c;都需要…

璀璨盛启·焕美升级 上颜集团杭州医学旗舰中心盛大启幕

2023年4月26日&#xff0c;「璀璨盛启焕美升级」上颜杭州医学旗舰中心启幕盛典在杭州滨江钱龙大厦耀新启幕。上颜用审美、匠心构建城市的活力与色彩&#xff0c;致力于为客户带来全新的美丽方式和一流的品质服务。 &#xff08;上台剪彩嘉宾从左至右依次为&#xff09;上颜集团…

CSS3小可爱亲吻表白特效,给你的五一假期增添点小乐趣

马上五一假期了&#xff0c;小伙伴们是不是都准备出去旅游呢&#xff0c;或者回老家陪陪父母。今天我用CSS3制作一个小可爱亲吻表白的特效&#xff0c;来给你即将到来的五一假期增添点小小的乐趣。 目录 实现思路 左边小可爱的实现 右边小可爱的实现 左右摇摆动效的实现 右…

LeetCode43. 字符串相乘(Java解法)

0 题目描述 给定两个以字符串形式表示的非负整数 num1 和 num2&#xff0c;返回 num1 和 num2 的乘积&#xff0c;它们的乘积也表示为字符串形式。 注意&#xff1a;不能使用任何内置的 BigInteger 库或直接将输入转换为整数。 示例 1: 输入: num1 “2”, num2 “3” 输出…

【三十天精通 Vue 3】 专栏内容介绍

在这个专栏中&#xff0c;我们将带你深入了解 Vue 3 的各个方面。首先&#xff0c;我们将带你了解 Vue 3 的新特性和改进&#xff0c;包括 Composition API、Provide/Use Case、Vuex 3 等。然后&#xff0c;我们将详细介绍 Vue 3 的组件化开发、路由、状态管理等方面的内容。 …