Android和Linux的嵌入式开发差异
最近开始投入Android的怀抱。说来惭愧,08年就听说这东西,当时也有同事投入去看,因为恶心Java,始终对这玩意无感,没想到现在不会这个嵌入式都快要没法搞了。为了不中年失业,所以只能回过头又来学。
首先还是说Android是基于Linux内核的,所以说骨子里还是linux,但是针对移动端,进行了深度优化。在这里结合GPT还有网上的信息,这里主要针对相比嵌入式Linux的差异,简单总结一下:
1 首先当然是前后端的分离。
有点类似Openwrt的发展。在之前,linux要开发图形程序是比较复杂的,QT,还有几个库,要么要收费,要么难以学习,最关键是和硬件平台绑定,换到新硬件要做移植。所以谷歌在操盘的时候,直接用Java做了前端GUI。当然,Java以前也有用J2SE做应用的传统。这样搞的好处也显而易见,前端和硬件分离,不管什么硬件,都不用改app代码了,虚拟机做适配就行了。IOS因为平台封闭,所以选了类C语言(具体机制空了再看看)。
所以首先的差异就是前端App要用Java开发,所以这个基本上绕不开。框架还是必须要学一下。
在前端App的层面,谷歌提供了Framework,还有大量的原生应用,比如电话,计算器,日历,email等。简化了很多开发。
=====================================
按照现在主流的分工,不管是Java还是Kotlin,应该都有专门的工程师。这部分对于嵌入式开发工程师我的理解是不用太懂,但是基本框架要懂,app是怎么怎么调用底层的,这个必须要知道。
试手一下CameraX(TODO)-CSDN博客
2 内核的变化(部分内容来自GPT)
Wakelocks: Android引入了"Wakelocks"机制,用于管理设备的唤醒状态。这是为了优化移动设备的电源管理,确保在需要时设备保持唤醒状态。就是俗称的唤醒锁,貌似这玩意主要是app层在用。
Low Memory Killer: Android引入了Low Memory Killer机制,用于监测并终止占用过多内存的应用程序进程,以维护系统的内存稳定性。这有助于提高系统的性能和响应能力。
Binder IPC(Inter-Process Communication): Android使用了自己的进程间通信机制,称为Binder。Binder允许Android系统中的不同组件进行通信,例如Activity、Service和BroadcastReceiver。据说这玩意类似Windows上的COM机制。
Ashmem: Android引入了Anonymous Shared Memory(Ashmem)机制,用于进程间共享匿名内存区域。这在Android中的共享内存和图形子系统中使用。
Logger: Android在内核中添加了用于日志记录的Logger模块,以支持Android的日志系统。
Android File System(FUSE): Android引入了FUSE(Filesystem in Userspace)用于文件系统的实现,允许用户空间程序运行文件系统代码而不需要修改内核。
安全性增强: Android对Linux内核进行了一些安全性的增强,包括SELinux(Security-Enhanced Linux)的集成,以提高系统的安全性。
调度策略: Android可能会使用不同的调度策略,以适应移动设备的性能和电源管理需求。
库的替换:好像glibc这些也都换了。
=====================================
大部分都是应用层。Binder是一个重头,因为涉及到接口。其他了解就行,真用到了再去看也可以。
Tee学习(TODO)-CSDN博客
3 驱动层的变化。(部分内容来自GPT)
Linux以前是标准的ioctl接口,Android改成了HAL接口,重要实现都切到了user space,用JNI封装。这些都要很了解。此外电源管理机制,还有Binder机制要看一下。
内核版本: Linux驱动接口: Linux驱动接口通常是为通用Linux内核设计的,支持各种硬件架构和设备类型。Linux内核在不同的发行版和版本中可能会有一些变化,但整体上是相对一致的。 Android驱动接口: Android使用了经过修改的Linux内核,因此Android驱动接口可能在某些方面与标准的Linux内核驱动接口略有不同。Android还可能引入一些额外的特定于移动设备的驱动需求。
电源管理: Linux驱动接口: Linux提供了通用的电源管理框架,驱动可以利用这些机制进行设备电源的管理。电源管理策略可能因硬件和内核配置而异。 Android驱动接口: Android对电源管理有其专有的需求,引入了Wakelocks等机制,以便更好地适应移动设备的电源管理和唤醒状态。
进程间通信(IPC): Linux驱动接口: 通常情况下,Linux驱动接口的设计并不直接涉及进程间通信,因为它主要关注设备与内核的交互。 Android驱动接口: Android引入了Binder机制,用于进程间通信。这对于Android中各种组件的通信非常重要,例如Activity、Service和BroadcastReceiver。
设备节点和HAL层: Linux驱动接口: 在标准Linux系统中,设备节点通常位于/dev目录下,用户空间可以通过这些设备节点与驱动进行通信。 Android驱动接口: Android引入了硬件抽象层(HAL),这是一个在用户空间和驱动之间的接口层,用于将Android系统与底层硬件驱动隔离开来。HAL层提供了标准接口,使得不同设备的驱动可以以相似的方式与Android系统进行交互。
调度策略: Linux驱动接口: Linux内核使用通用的进程调度策略,适用于各种设备和场景。 Android驱动接口: Android可能对调度策略进行定制,以适应移动设备的特殊需求,例如更好的响应性和电源管理。
驱动总览:
=====================================
这部分是重头,尤其是HAL层的差异,改的挺大的,必须要懂,而且要很懂。然后Android驱动层的一些实现,FrameBuffer(貌似现在换了),V4L2,ALSA,USB的gaeget框架,可能都要看看。当然,后面这部分也是Linux平台的玩法,区别不大。
Android的硬件接口HAL-CSDN博客
4 安卓运行环境增加的命令
am activity管理器,启动activity等。直接管理activity的原因可能还是为了调试方便。(猜测)
pm package管理器。这个没啥好多说的。
svc 服务管理器。和linux标准的service有点接近,但是可以管理的是Android的特定服务,比如蓝牙,wifi等等。
input 模拟输入,主要应该还是用在调试方面。(搞外挂利器?)
getprop/setprop 以前在路由器上,这一套要单独实现,就是TR069,SNMP之类,现在谷歌整体给你实现了。轻松很多。在嵌入式开发中,这个是大头。
settings 这个说的主要是设置android。但是和上面的有什么区别呢?还是看了下GPT,setting主要修改和用户界面相关的东西,主要是app领域。getprop/setprop则更广,包含一些系统底层属性,需要的权限也更高,有些要root权限。貌似在嵌入式开发中,主要还是后者。
安卓平台在本地也集成了一个数据库,有点类似SQLite。当然,这个就是爱用不用了。
getevent 获取本地事件。nandread 读取nand数据。
最后有别于一般用的busybox,安卓用的toolbox,常用命令大同小异。
=================================================
要点有两个,一个是通过getprop/setprop给adb调试接口,另一个是用getevent,inputs去做定位分析,也都不是太难。其他的知道怎么用就行了。
5 用户层差异
主要就是运行的文件系统,一些重要的守护进程,然后一些工具的了解。
守护进程列表:
安卓启动流程:
文件结构
重要服务的位置:
文件系统差异(来自GPT)
应用和用户数据: Android: Android 将应用和用户数据存储在 /data 分区中。每个应用程序都有自己的私有数据目录,其中包含其数据和设置。例如,应用程序的数据库、缓存和共享首选项通常存储在 /data/data/<package_name> 目录下。 嵌入式 Linux: 在一般的嵌入式 Linux 系统中,应用程序的数据和设置通常存储在 /usr、/var 或其他指定目录中。
系统文件和可执行文件: Android: Android 的系统文件和可执行文件通常存储在 /system 分区中。这包括 Android 操作系统的核心文件、系统应用和一些系统级别的设置。 嵌入式 Linux: 一般的嵌入式 Linux 系统的系统文件和可执行文件可能分散在不同的目录中,通常包括 /bin、/sbin、/lib 等。
可变数据和缓存: Android: 可变数据和缓存通常存储在 /cache 分区中。这包括一些临时文件和缓存,可能会在系统启动时被清理。 嵌入式 Linux: 一般的嵌入式 Linux 系统可能将临时文件和缓存存储在 /tmp 目录下。
外部存储: Android: 外部存储通常映射到 /sdcard,用于存储用户的媒体文件、下载内容等。 嵌入式 Linux: 嵌入式 Linux 系统也可以使用外部存储,但其挂载点和目录结构可能不同,具体取决于设备和系统定制。
硬件相关文件: Android: Android 中可能包含一些硬件相关的文件和节点,如 /dev 目录下的设备节点,用于与硬件交互。 嵌入式 Linux: 嵌入式 Linux 系统也有 /dev 目录,但硬件节点的命名和数量可能根据系统的硬件配置而变化。
=======================================
和第4点差不多,没什么难度,用两次熟悉就可以了,细节要用的时候知道怎么查就行了。
Android系统目录介绍_android 目录-CSDN博客
6 源码结构
看起来好像使用envsetup.sh,lunch还有make就够了。但是还是要深入了解一下。
这部分核心要点一个是如何新增设备,另一个是如何新增一个app。最后就是怎么在启动脚本里面加东西。
安卓的编译加速使用了CCache,我在之前公司曾经建议使用这玩意 ,不过后面被否了。。。
还有一个要了解的就是Android.bp。其实本质就是cmake那种东西。。。
============================================
这个也算次重点吧。主要是要懂编译环境,各个部分怎么生成的。要懂怎么增加一个设备,或者怎么增加一个App到编译环境。
CMake小结-CSDN博客
Android SDK学习(TODO)-CSDN博客
AOSP系列—阅读源码并熟悉AOSP目录结构_aospxref-CSDN博客
7 Fastboot
也就是俗称的刷机模式。是一个Android特有的工具,在Linux上没有。Android通过特定手段进入该模型,上位机配合一个exe文件。实现操作固件的一些功能。
在我看来本质上就是对bootloader的一个封装,好像官方名也叫bootloader interface, 提供的功能也是uboot命令行的那些,分区,升级,刷固件,主备分区管理,系统变量的设置。常规嵌入式开发中,规划分区也是很重要的工作,貌似Android不能修改分区。
里面有Recovery模式和Fastboot模式,一个主要专注系统恢复,一个功能更全面。
=====================================================
如果用过uboot的话,看一下用法即可。
8 调试方法的学习
主要是ADB,这个比较简单,之前有写过。可能有一些细节,以后用到再说吧。
=====================================================
虽然用的很多,但是没什么难的。
ADB的使用-CSDN博客
9 平台特性的学习
主要是硬件平台的特性。
======================================================
这个也是重点,毕竟到什么山头唱什么歌,高通的,瑞芯微的,还是谁的,都要仔细学习。
就写这么多吧,还有的看到再写写。虽然说换汤不换药,本质核心还是编程能力。懂不懂这个汤可能就是会不会失业,运气好工资一个月差几十K也都有可能。所以还是学学吧。
参考资料:
Embedded Android
ChatGPT
相关文章:

Android和Linux的嵌入式开发差异
最近开始投入Android的怀抱。说来惭愧,08年就听说这东西,当时也有同事投入去看,因为恶心Java,始终对这玩意无感,没想到现在不会这个嵌入式都快要没法搞了。为了不中年失业,所以只能回过头又来学。 首先还是…...
关于Node.js异常处理的教程
在Node.js开发中,异常处理是非常重要的一部分。良好的异常处理可以帮助我们及时发现和解决问题,提高系统的稳定性和可靠性。本教程将向您介绍Node.js中异常处理的最佳实践和策略。 1. 使用try-catch捕获同步异常 在Node.js中,可以使用try-c…...

13. Springboot集成Protobuf
目录 1、前言 2、Protobuf简介 2.1、核心思想 2.2、Protobuf是如何工作的? 2.3、如何使用 Protoc 生成代码? 3、Springboot集成 3.1、引入依赖 3.2、定义Proto文件 3.3、Protobuf生成Java代码 3.4、配置Protobuf的序列化和反序列化 3.5、定义…...
Spring: Springboot 框架集成不同版本的spring redis
文章目录 一、集成不同版本的spring redis1、Spring Data Redis 1.x:2、Spring Data Redis 2.x:3、Spring Data Redis 3.x(Spring Boot 2.x): 二、springboot集成Spring Data Redis 2.x1、首先,确保在 pom.…...
学习JAVA的第八天(基础)
目录 多态 前提 形式 测试类 调用成员的特点 优势 劣势 包 注意事项: final关键字 常量 命名规范: 注意事项: 权限修饰符 分类 代码块 局部代码块 构造代码块 静态代码块 抽象类 抽象类: 定义格式 抽象…...

【硬件相关】IB网/以太网基础介绍及部署实践
文章目录 一、前言1、Infiniband网络1.1、网络类型1.2、网络拓扑1.3、硬件设备1.3.1、网卡1.3.2、连接线缆a、光模块b、线缆 1.3.4、交换机 2、Ethernet网络 二、部署实践(以太网)1、Intel E810-XXVDA21.1、网卡信息1.2、检查命令1.2、驱动编译 2、Mella…...

【JavaEE】_Spring MVC项目之建立连接
目录 1. Spring MVC程序编写流程 2. 建立连接 2.1 RequestMapping注解介绍 2.2 RequestMapping注解使用 2.2.1 仅修饰方法 2.2.2 修饰类与方法 2.3 关于POST请求与GET请求 2.3.1 GET请求 2.3.2 POST请求 2.3.3 限制请求方法 1. Spring MVC程序编写流程 1. 建立连接&…...
【JavaEE进阶】 Spring AOP源码简单剖析
文章目录 🍃前言🍀Spring AOP源码剖析⭕总结 🍃前言 前面的博客中,博主对代理模式进行了一个简单的讲解,接下来博主将对Spring AOP源码进行简单剖析,使我们对Spring AOP了解的更加深刻。 🍀Sp…...
Redis--内存回收机制详解
什么是内存回收机制? 众所周知Redis之所以性能高是因为数据都存在内存中,内存是很宝贵的,Redis的内存回收机制本质就是处理达到过期时间的key-value,以及当内存到达最大使用值时候触发的内存淘汰策略。 Redis数据删除的策略有哪些…...

win安装卸载python3.13
一、安装 访问python官网:https://www.python.org/ 点击“Downloads” 点击“Windows” 找到自己要下载的版本和位数,比如我这个是3.13版本、64位的安装包 下载好了之后,双击安装包 勾选“Add python.exe to PATH”:把python环…...

APIFox-自动获取登录状态操作
APIFox-自动获取登录状态操作 概述 作为纯后端开发码农,每次接口开发完的调试很重要,因此每次重复的手动获取登陆状态Token或者直接放行就太麻烦了。 APIFox提供了前置操作,可以很方便的自动获取登录状态,节省大量重复劳动时间。…...

【NDK系列】Android tombstone文件分析
文件位置 data/tombstone/tombstone_xx.txt 获取tombstone文件命令: adb shell cp /data/tombstones ./tombstones 触发时机 NDK程序在发生崩溃时,它会在路径/data/tombstones/下产生导致程序crash的文件tombstone_xx,记录了死亡了进程的…...

CentOS7 Hive2.3.8安装
CentOS7 Hive2.3.8 安装 建议从头用我的博客,如果用外教的文件到 一、9)步骤了,就用他的弄完,数据库不一样,在9步骤前还能继续看我的 一、 安装MySQL 0.0)查询mariadb,有就去0.1),没有就不管…...
代码随想录算法训练营第四十四天 完全背包 、零钱兑换 II 、组合总和 Ⅳ
代码随想录算法训练营第四十四天 | 完全背包 、零钱兑换 II 、组合总和 Ⅳ 完全背包 题目链接:题目页面 (kamacoder.com) 解释一、01背包 一维 :为什么要倒序遍历背包? 首先要明白二维数组的递推过程,然后才能看懂二维变一维的…...

【经验】vscode 鼠标拖曳不能选中整行文字,只能选中纵向矩形范围
1、问题描述 不知道昨天操作vscode设置界面时,误选择了啥,导致鼠标拖曳不能选中整行文字,只能选中纵向矩形范围,现象如下: 2、解决方法 1)打开设置界面 点击左下角按键,选择“设置” 2&…...

Redis--事务机制的详解及应用
Redis事务的概念: Redis事务就是将一系列命令包装成一个队列,在执行时候按照添加的顺序依次执行,中间不会被打断或者干扰,在执行事务中,其他客户端提交的命令不可以插入到执行事务的队列中,简单来说Redis事…...

路由器端口映射如何配置?
在网络通信中,路由器是一个重要的设备,它负责将数据包从一个网络传输到另一个网络。路由器的端口映射配置是一种重要的设置,可以使外部网络中的计算机通过访问路由器上的特定端口与内部网络中的计算机进行通信。本文将介绍什么是路由器端口映…...

力扣34. 在排序数组中查找元素的第一个和最后一个位置(二分查找)
Problem: 34. 在排序数组中查找元素的第一个和最后一个位置 文章目录 题目描述思路复杂度Code 题目描述 思路 Problem: 二分查找常用解题模板(带一道leetcode题目) 直接套用上述中的寻找左、右边界的二分查找模板即可 复杂度 时间复杂度: O ( l o g n )…...

【每日一题】3.2 求逆序对
题目描述 给定一个长度为 n的整数数列,请你计算数列中的逆序对的数量。 逆序对的定义如下:对于数列的第 i个和第 j个元素,如果满足 i<j 且 a[i]>a[j],则其为一个逆序对;否则不是。 输入格式 第一行包含整数 n…...
NTP时间源服务器(NTP网络时钟)助力智慧医院数字化
NTP时间源服务器(NTP网络时钟)助力智慧医院数字化 NTP时间源服务器(NTP网络时钟)助力智慧医院数字化 目前计算机网络中各主机和服务器等网络设备的时间基本处于无序的状态。 随着计算机网络应用的不断涌现,计算机的时…...

如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...

莫兰迪高级灰总结计划简约商务通用PPT模版
莫兰迪高级灰总结计划简约商务通用PPT模版,莫兰迪调色板清新简约工作汇报PPT模版,莫兰迪时尚风极简设计PPT模版,大学生毕业论文答辩PPT模版,莫兰迪配色总结计划简约商务通用PPT模版,莫兰迪商务汇报PPT模版,…...

pikachu靶场通关笔记19 SQL注入02-字符型注入(GET)
目录 一、SQL注入 二、字符型SQL注入 三、字符型注入与数字型注入 四、源码分析 五、渗透实战 1、渗透准备 2、SQL注入探测 (1)输入单引号 (2)万能注入语句 3、获取回显列orderby 4、获取数据库名database 5、获取表名…...
嵌入式常见 CPU 架构
架构类型架构厂商芯片厂商典型芯片特点与应用场景PICRISC (8/16 位)MicrochipMicrochipPIC16F877A、PIC18F4550简化指令集,单周期执行;低功耗、CIP 独立外设;用于家电、小电机控制、安防面板等嵌入式场景8051CISC (8 位)Intel(原始…...

QT开发技术【ffmpeg + QAudioOutput】音乐播放器
一、 介绍 使用ffmpeg 4.2.2 在数字化浪潮席卷全球的当下,音视频内容犹如璀璨繁星,点亮了人们的生活与工作。从短视频平台上令人捧腹的搞笑视频,到在线课堂中知识渊博的专家授课,再到影视平台上扣人心弦的高清大片,音…...

快速排序算法改进:随机快排-荷兰国旗划分详解
随机快速排序-荷兰国旗划分算法详解 一、基础知识回顾1.1 快速排序简介1.2 荷兰国旗问题 二、随机快排 - 荷兰国旗划分原理2.1 随机化枢轴选择2.2 荷兰国旗划分过程2.3 结合随机快排与荷兰国旗划分 三、代码实现3.1 Python实现3.2 Java实现3.3 C实现 四、性能分析4.1 时间复杂度…...

【若依】框架项目部署笔记
参考【SpringBoot】【Vue】项目部署_no main manifest attribute, in springboot-0.0.1-sn-CSDN博客 多一个redis安装 准备工作: 压缩包下载:http://download.redis.io/releases 1. 上传压缩包,并进入压缩包所在目录,解压到目标…...
从零手写Java版本的LSM Tree (一):LSM Tree 概述
🔥 推荐一个高质量的Java LSM Tree开源项目! https://github.com/brianxiadong/java-lsm-tree java-lsm-tree 是一个从零实现的Log-Structured Merge Tree,专为高并发写入场景设计。 核心亮点: ⚡ 极致性能:写入速度超…...

ABAP设计模式之---“Tell, Don’t Ask原则”
“Tell, Don’t Ask”是一种重要的面向对象编程设计原则,它强调的是对象之间如何有效地交流和协作。 1. 什么是 Tell, Don’t Ask 原则? 这个原则的核心思想是: “告诉一个对象该做什么,而不是询问一个对象的状态再对它作出决策。…...