技术标签: GICI源码阅读 导航定位 GNSS Camera INS
- 原始 Markdown文档、Visio流程图、XMind思维导图见:https://github.com/LiZhengXiao99/Navigation-Learning
- 欢迎关注我的公众号:NaviCode小工匠
作者的介绍:为了阐明 GNSS 的算法模型,加快在多源融合应用中针对 GNSS 的开发效率,我们开源了 GICI-LIB,并辅以详尽的文档和全面的数据集。GICI-LIB 以可扩展的设计理念,实现了GIC传感器之间多种形式的松紧组合。评估结果表明,GIC 系统能够在多种复杂环境下,提供分米到米级的高精度导航。
GICI-LIB 全称 GNSS/INS/Camera Integrated Navigation Library,是上海交大最新开源的一套基于图优化的 GNSS+INS+Camera 集成导航定位库。基于 RTKLIB 处理 I/O 流、编解码;基于 OKVIS 因子图优化类型封装;基于 SVO 做特征提取。以 GNSS 为主,再加入 INS、Camera 做组合,支持相当多的数据格式、定位模式,包含很多 GNSS 因子、惯导因子、视觉因子及运动约束。以处理实时数据为主,后处理也采用模拟实时数据处理的方式进行。典型的应用方式如下图:
支持非 ROS 模式和 ROS 模式,ROS 模式下可以使用 ROS 话题、RVIZ 显示轨迹。
支持多种数据形式:serial 串口、TCP/IP server、TCP/IP client、Ntrip server、Ntrip client、V4L2、file 文件。
支持多种数据编码:RTCM2、RTCM3、 Ublox raw、Septentrio raw、Tersus raw、NMEA、DCB file、ATX file、V4L2 image pack、GICI image pack、GICI IMU pack
支持多种数据传输方式:I/O 端口、串口、ROS topics、TCP/IP、NTRIP、V4L2、文件
支持多种定位模式:SPP、SDGNSS、DGNSS、RTK、PPP、SPP-based LC GINS、TC GINS、RTK-based LC GINS、SRR GVINS、RRR GVINS。
支持多种因子:
支持三个层级的 SSR 服务:
GICI 不支持时间系统偏差的估计,需要进行硬件层面的时间对准,且最好精度要达到 5ms。
GICI-LIB 主要使用 C++ 编写,且大量使用 C++ 高级语法,CMake 文件里规定以 C++11 的标准编译,采用多线程、面向对象程序设计方法,写法"很C++",想看懂源码需要有一定的 C++ 基础。C++11 语法的内容可以看下图:
具体来说,还有以下特点:
std::function
、std::bind
。用 cloc 统计 include、src、tools 文件夹代码量,结果如下:
语言 | 文件数 | 空行 | 注释行 | 代码行 |
---|---|---|---|---|
C++ | 110 | 3704 | 4547 | 26189 |
C/C++ Header | 104 | 2746 | 4841 | 8452 |
MATLAB | 16 | 111 | 232 | 666 |
C | 10 | 98 | 93 | 394 |
CMake | 9 | 71 | 36 | 262 |
YAML | 4 | 8 | 17 | 185 |
Gencat NLS | 13 | 0 | 0 | 94 |
XML | 2 | 8 | 0 | 56 |
总计 | 268 | 6746 | 9766 | 36298 |
src 和 tools 文件夹内各子文件夹功能信息如下表所示:
文件名 | 功能 | 代码行数 |
---|---|---|
estimate | 因子图优化相关类型、函数 | 2451 |
fusion | 多源融合导航相关:估计器、初始化器类型定义 | 2872 |
gnss | GNSS相关:误差改正、模糊度固定、SPP、PPP、RTK | 9347 |
imu | IMU相关:数值更新、误差改正 | 1725 |
stream | 数据流相关、Node处理相关类型定义 | 2829 |
utility | 全局变量、配置选项、信号处理函数、 Spin线程定义 | 1159 |
vision | 视觉相关:初始化、特征提取、处理、跟踪、相对位置计算 | 1900 |
conversions | 时间转换、坐标转换 | 175 |
edit_binary | 二进制数据处理、处理采样间隔 | 1137 |
evaluation | NMEA相关 | 1672 |
ros | ros 相关、话题消息发布、msg定义 | 2149 |
matlab_plot | MATLAB 画图脚本 | 666 |
GICI-LIB 的 manual 足足有 117 面,挺详细的,内容如下:
force initial global position
选项设置为 true,我们通过 initial global position
选项从配置文件中加载点坐标。摄像机状态之间没有相互联系,因为相应的参数是时变的。我们在图中保留了连接,以表示由于跟踪地标的切换,各次的估计参数会发生变化。
GICI-LIB 提供的多种传感器在不同定位模式下的很多因子,并封装了一个基类,为所有传感器提供基础的函数去操作变量因子和量测因子,具体可以看各种 xxx_estimator_base.h
,看算法的时候肯定重点要看这些文件。
git clone
下载不了,可以去git clone
后面的网站上手动下载。本文只介绍非 ROS 版。
我使用的环境是 VSCode + WSl,很多导航定位的开源软件都基于 Linux,比起虚拟机,VSCode + WSl 要流畅一些,不熟悉的推荐看这篇文章。
确保之前已经配置好 C++ 环境(g++、Cmake、VScode 插件)。
看别人的博客,都提到了曾经安装过 glog/gflag 会出问题:GICI-OPEN多源融合导航框架编译及问题说明
构建、编译的时候找不到库,可能是因为库装到
/usr/local/lib
里了,试试创建软链接到/usr/lib
ln -s /usr/local/lib/库名.a /usr/lib/库名.a
sudo apt-get install libeigen3-dev
文件下载
环境配置
sudo apt-get install cmake git libgtk2.0-dev pkg-config libavcodec-dev libavformat-dev libswscale-dev
sudo apt-get install python-dev python-numpy libtbb2 libtbb-dev libjpeg-dev libpng-dev libtiff-dev libjasper-dev libdc1394-22-dev
进入 opencv 目录编译安装
cd opencv
mkdir build
cd build
sudo cmake -D CMAKE_BUILD_TYPE=Release -D CMAKE_INSTALL_PREFIX=/usr/local ..
sudo make -j8
sudo make install
将 OpenCV 的库添加到路径,从而可以让系统找到
sudo vim /etc/ld.so.conf.d/opencv.conf
在文件中加上并保存退出 /usr/local/lib
sudo ldconfig
配置 bash
sudo vim /etc/bash.bashrc
# 在打开的文件最末尾添加以下代码并保存退出
PKG_CONFIG_PATH=$PKG_CONFIG_PATH:/usr/local/lib/pkgconfig
export PKG_CONFIG_PATH
source /etc/bash.bashrc
执行 pkg-config --cflags opencv
如果报错,可以看博客
下载 glfg
git clone https://github.com/gflags/gflags
进入 glfg 目录编译安装
cd glfg
mkdir build
cd build/
cmake -DBUILD_SHARED_LIBS=ON -DBUILD_STATIC_LIBS=ON -DINSTALL_HEADERS=ON -DINSTALL_SHARED_LIBS=ON -DINSTALL_STATIC_LIBS=ON -DCMAKE_INSTALL_PREFIX=/usr/ ..
make
sudo make install
下载 glog
git clone https://github.com/google/glog
进入 glog 目录编译安装
cd glog
mkdir build
cd build
cmake -DGFLAGS_NAMESPACE=google -DCMAKE_CXX_FLAGS=-fPIC -DBUILD_SHARED_LIBS=ON -DCMAKE_INSTALL_PREFIX=/opt/glog ..
make
sudo make install
为使 glog 库生效,需要在 /etc/ld.so.conf.d
下新建配置文件并使其生效
cd /etc/ld.so.conf.d
sudo vim glog.conf
输入 /opt/glog/lib
sudo ldconfig
下载
git clone https://github.com/jbeder/yaml-cpp.git
进入 Yaml-cpp 目录编译安装
cd yaml-cpp
mkdir build
cd build
cmake -D BUILD_SHARED_LIBS=ON ..
make -j16
sudo make install
下载
git clone https://ceres-solver.googlesource.com/ceres-solver
进入 ceres-solver 目录编译安装
mkdir ceres-bin
cd ceres-bin
cmake ../ceres-solver-2.1.0
make -j3
make test
make install
下载
git clone https://github.com/chichengcn/gici-open
编译
在工程目录下打开终端,输入以下命令:
mkdir build
cd build
cmake ..
make -j4 # 编译需要一段时间
glog 即 Google Log ,是一个 Google 开源的日志库,它提供了一个轻量级的、可扩展的、跨平台的日志系统。 glog 的用法包括:
引入头文件:需要包含 glog 的头文件:
include <glog/logging.h>
初始化库:在开始使用 glog 之前, 初始化库,例如:
google::InitGoogleLogging(argv[0])
配置日志:可以通过配置文件或代码来配置 glog 的参数,例如:
google::SetLogDestination(LOG_TO_FILE, "/path/to/logfile.log")
输出日志:使用 LOG(level)
宏函数来输出日志。level
表示日志的严重程度,可以是以下几个级别之一:INFO
:一般信息、WARNING
:警告信息、ERROR
:错误信息、FATAL
:致命错误信息,输出后会终止程序。例如输出一般信息:
LOG(INFO) << "This is an informational message."
条件输出日志:使用 LOG_IF()
、LOG_EVERY_N()
和 LOG_FIRST_N()
宏函数来条件输出日志。例如:
LOG_IF(INFO, num_cookies > 10) << "Got lots of cookies"
关闭日志:在程序结束之前,关闭 glog:
google::ShutdownGoogleLogging()
在手册的 9~39 面,详细的介绍了配置文件的具体内容。GICI-LIB 采用 YAML 配置文件格式,下面先对 YMAL 做个简单介绍。
链接时找不到 yaml-cpp,可以参考博客:error while loading shared libraries的解决方案,在
/etc/ld.so.conf
文件中加上/usr/local/lib
YAML(YAML Ain’t Markup Language)是一种轻量级的数据序列化格式,可以用于配置文件、数据交换、API请求等多种场景。它是一种简单易用的数据序列化格式,使得数据可以以人类可读的方式进行存储和传输。YAML的语法非常简单,它使用缩进和符号来表示数据结构。以下是一些YAML的基本语法:
字符串:用引号括起来的文本,例如:“hello world”。
数字:没有引号的数字,例如:42。
布尔值:用 true 或 false 表示的真或假。
缩进:YAML使用缩进来表示嵌套关系,每个缩进级别用空格数表示。例如,下面的代码段表示一个包含两个列表的字典:
字典/对象/键值对:用短横线 -
或中括号[]
表示的键值对的集合。例如:{name: John, age: 30}
或- name: John age: 30
。多层对象可表示为:
key: {
key1: value1, key2: value2}
或者
key:
key1: value1
key2: value2
数组/列表:用短横线 -
或中括号 []
表示的值的列表。例如:[apple, banana, orange]
或 - apple - banana - orange
。复杂一点的如:
streamers:
- streamer:
tag: str_gnss_rov
output_tags: [fmt_gnss_rov]
type: file
path: <data-directory>/gnss_rover.bin
- streamer:
tag: str_gnss_ref
output_tags: [fmt_gnss_ref]
type: file
path: <data-directory>/gnss_reference.bin
引用:&
用来建立锚点,<<
表示合并到当前数据,*
用来引用锚点。
注释:在YAML中,使用 #
表示注释。
YAML 需要特别注意的几个点:
- 大小写敏感。
- 缩进不允许使用 tab,只允许空格。
- 缩进的空格数不重要,只要相同层级的元素左对齐即可。
YMAL 在 C++ 中以 Node 类表示。
LoadFile():从文件中加载 YAMl 到 C++ 中 Node 对象:
yaml_node = YAML::LoadFile(文件名);
[]:Node 对象可以理解为是树形的,用中括号可以取出里面的子数,创建一个新的 Node 对象:
YAML::Node logging_node = yaml_node["logging"];
safeGet():第一个参数为 Node,第二个参数为关键字,判断配置文件的 Node 里有没有你要的那个关键字,有的话再把对应的值作为第三个参数返回。
option_tools::safeGet(logging_node, "log_to_stderr", &FLAGS_logtostderr);
checkSubOption():第一个参数为 Node、第二个参数为子配置选项,检查参一中是否存在参二子配置选项,如果不存在有两种处理:
下面三个函数都是写了函数模板,然后重载写了很多的 :
配置文件以 数组 + 键值对 的方式组织,每一个键值对都是一个配置项,用多级数组来分模块组织配置项键值对。有三大模块:stream、estimate、logging,其中 stream 模块内还分为三个子模块 steamers、formators、replay。如下图:
option 文件夹里有一些配置文件,以伪实时定位解算为主,对应于下面的应用方式,图上每个模块都对应着咱们要配置的内容。
使用方式:
<data-directory>
、<gici-root-directory>
和 <output-directory>
分别换成你的数据文件夹路径
、gici-open 文件夹的路径
、和 输出文件夹路径
。start_time
,起始时间。streamer
写了两套,非 ROS 模式和 ROS 模式,想用哪套就把另一套注释掉。streamers
里一项项 streamer
的路径项 path
,确保数据文件夹中都有对应的数据。streamer
中路径设置在 option 文件夹中,程序会从 gici-open 文件夹的路径
找 option 文件夹,确保 option 文件夹和里面数据在对应位置,最好不要动 option 文件夹。xxx_solution.txt
文件可以直接用 rtkplot
打开查看结果。用matlab_plot
里的脚本应该也行。一定注意,配置文件中有好几处
<data-directory>
、<gici-root-directory>
、<output-directory>
、start_time
要改,不能漏,我在这卡了很久。没运行成功,仔细看看报错信息,INFO不用看,关注ERROR,看对应配置是否正确。
对 GitHub 上的介绍简单做个翻译
作者专为开发 GICI-LIB 而搭建的数据采集的平台如下图所示:
开发了一块 GICI 板,用于收集 IMU 和摄像头数据,并在整个平台中应用了与其他传感器同步的硬件。板载 IMU 和摄像头分别为博世 BMI088 和 Onsemi MT9V034。GNSS 接收器为 Tersus David30 多频接收器。我们还从千寻 SI 数据流中收集了参考站数据,用于 RTD 和 RTK ,并从国际 GNSS 服务(IGS)数据流中收集了状态空间表示(SSR)数据,用于 PPP。光纤 IMU 通过对其数据和 GNSS 原始数据进行后处理来提供参考值。
收集了两种数据集:不同场景的短期(几分钟)实验(1.1 ~ 4.3)和涵盖多个场景的长期(几十分钟)实验(5.1 ~ 5.2)。在短期实验中,我们将场景分为 4 类: 开阔天空、绿树成荫、典型城市和密集城市。对于每个场景,我们提供 2 ~ 3 条轨迹。在长期实验中,我们提供了在上海市中心收集到的涵盖这些场景的两个轨迹。
ID | Scene | Size | Date | Scene View |
---|---|---|---|---|
1.1 | Open-sky | 0.7 GB | 2023.03.20 | Images |
1.2 | Open-sky | 0.5 GB | 2023.03.27 | Images |
2.1 | Tree-lined | 1.4 GB | 2023.03.20 | Images |
2.2 | Tree-lined | 0.6 GB | 2023.03.27 | Images |
3.1 | Typical urban | 1.7 GB | 2023.03.27 | Images |
3.2 | Typical urban | 1.4 GB | 2023.03.27 | Images |
3.3 | Typical urban | 1.9 GB | 2023.03.27 | Images |
4.1 | Dense urban | 1.4 GB | 2023.05.21 | Images |
4.2 | Dense urban | 0.8 GB | 2023.03.27 | Images |
4.3 | Dense urban | 1.6 GB | 2023.03.27 | Images |
5.1 | Long-term | 8.2 GB | 2023.05.21 | Images |
5.2 | Long-term | 5.8 GB | 2023.05.21 | Images |
用对应数据的时候记得改时间
提供了各种 YAML 配置文件示例,在 <gici-root-directory>/option
。请记住替换所有 <path>
和 "start_time"
。通过命令来运行软件处理数据集:
./gici_main <gici-config-file>
要将实时输出流连接到 RTKLIB,应执行以下步骤:
我们提供了一个将 bin 文件转换为 rosbags 的工具,请参见 <gichi-root-directory>/tools/ros/gici_tools/src/gici_files_to_rosbag.cpp
。其配置文件位于 <gici-root-directory>/tools/ros/gici_tools/option/convert_rosbags.yaml
中。请记住替换所有 <path>
和 "start_time"
。
可以通过以下方式编译转换器:
cd \<gici-root-directory\>/tools/ros
catkin_make -DCMAKE_BUILD_TYPE=Release
然后可以通过以下方式运行转换器:
./devel/lib/gici_tools/gici_files_to_rosbag <config-file>
YAML 配置文件示例,请参见 <gichi-root-directory>/ros_wrapper/src/gici/option
。使用前要替换所有<path>
和 "start_time"
。在运行 ROS 可执行文件之前,请记得运行一个 roscore。然后,可以通过以下方式运行可执行文件:
rosrun gici_ros gici_ros_main <gici-config-file>
或者:
cd \<gici-root-directory\>/ros_wrapper
./devel/lib/gici_ros/gici_ros_main <gici-config-file>
之后,您可以通过以下方式播放从我们的 bin 文件转换而来的 rosbags
rosbag play <data1.bag> <data2.bag> <data3.bag> ...
为了实现可视化,您可以通过以下方式运行我们的 RVIZ 配置:
rviz -d \<gici-root-directory\>/ros_wrapper/src/gici/rviz/gici_gic.rviz
我们为每个数据集提供 ground_truth.txt。参考数据采用光纤 IMU 的框架。在比较结果之前,您应该进行坐标转换。对于包含 IMU 的估计器,GICI 以 IMU 框架输出解决方案。我们提供将参考值转换为 IMU 框架的工具。首先要编译这个工具:
cd \<gici-root-directory\>tools/evaluation/alignment
mkdir build
cd build
cmake .. -DCMAKE_BUILD_TYPE=Release
make -j8
cd \<gici-root-directory\>tools/evaluation/format_converters
mkdir build
cd build
cmake .. -DCMAKE_BUILD_TYPE=Release
make -j8
然后,您可以通过以下方法转换参考值:
\<gici-root-directory\>tools/evaluation/format_converters/build/ie_to_nmea ground_truth.txt
\<gici-root-directory\>tools/evaluation/alignment/build/nmea_pose_to_pose ground_truth.txt.nmea
nmea_pose_to_pose.cpp 中的默认设置是将姿势从光纤 IMU 帧转换为数据集的 IMU 帧。如果您有其他要求,应修改 nmea_pose_to_pose.cpp 中的参数。现在,您将获得以 NMEA 格式转换的参考值文件 ground_truth.txt.nmea.transformed。为了便于可视化,您可以通过以下方法将该文件转换为 TUM 格式
\<gici-root-directory\>tools/evaluation/format_converters/build/nmea_to_tum ground_truth.txt.nmea.transformed
还可以将 GICI NMEA 输出转换为 TUM 格式,然后用任何软件进行比较。
对于纯 GNSS 估计器,GICI 以 GNSS 天线框架输出解决方案。您应进一步将参考值转换为 GNSS 天线,方法是
\<gici-root-directory\>tools/evaluation/alignment/build/nmea_pose_to_position ground_truth.txt.nmea.transformed
现在您会得到一个参考值文件 ground_truth.txt.nmea.transformed.translated。然后就可以继续上面的操作了。
nd_truth.txt.nmea
nmea_pose_to_pose.cpp 中的默认设置是将姿势从光纤 IMU 帧转换为数据集的 IMU 帧。如果您有其他要求,应修改 nmea_pose_to_pose.cpp 中的参数。现在,您将获得以 NMEA 格式转换的参考值文件 ground_truth.txt.nmea.transformed。为了便于可视化,您可以通过以下方法将该文件转换为 TUM 格式
```bash
\<gici-root-directory\>tools/evaluation/format_converters/build/nmea_to_tum ground_truth.txt.nmea.transformed
还可以将 GICI NMEA 输出转换为 TUM 格式,然后用任何软件进行比较。
对于纯 GNSS 估计器,GICI 以 GNSS 天线框架输出解决方案。您应进一步将参考值转换为 GNSS 天线,方法是
\<gici-root-directory\>tools/evaluation/alignment/build/nmea_pose_to_position ground_truth.txt.nmea.transformed
现在您会得到一个参考值文件 ground_truth.txt.nmea.transformed.translated。然后就可以继续上面的操作了。
文章浏览阅读1.6k次。安装配置gi、安装数据库软件、dbca建库见下:http://blog.csdn.net/kadwf123/article/details/784299611、检查集群节点及状态:[root@rac2 ~]# olsnodes -srac1 Activerac2 Activerac3 Activerac4 Active[root@rac2 ~]_12c查看crs状态
文章浏览阅读1.3w次,点赞45次,收藏99次。我个人用的是anaconda3的一个python集成环境,自带jupyter notebook,但在我打开jupyter notebook界面后,却找不到对应的虚拟环境,原来是jupyter notebook只是通用于下载anaconda时自带的环境,其他环境要想使用必须手动下载一些库:1.首先进入到自己创建的虚拟环境(pytorch是虚拟环境的名字)activate pytorch2.在该环境下下载这个库conda install ipykernelconda install nb__jupyter没有pytorch环境
文章浏览阅读5.2k次,点赞19次,收藏28次。选择scoop纯属意外,也是无奈,因为电脑用户被锁了管理员权限,所有exe安装程序都无法安装,只可以用绿色软件,最后被我发现scoop,省去了到处下载XXX绿色版的烦恼,当然scoop里需要管理员权限的软件也跟我无缘了(譬如everything)。推荐添加dorado这个bucket镜像,里面很多中文软件,但是部分国外的软件下载地址在github,可能无法下载。以上两个是官方bucket的国内镜像,所有软件建议优先从这里下载。上面可以看到很多bucket以及软件数。如果官网登陆不了可以试一下以下方式。_scoop-cn
文章浏览阅读4.5k次,点赞2次,收藏3次。首先要有一个color-picker组件 <el-color-picker v-model="headcolor"></el-color-picker>在data里面data() { return {headcolor: ’ #278add ’ //这里可以选择一个默认的颜色} }然后在你想要改变颜色的地方用v-bind绑定就好了,例如:这里的:sty..._vue el-color-picker
文章浏览阅读640次。基于芯片日益增长的问题,所以内核开发者们引入了新的方法,就是在内核中只保留函数,而数据则不包含,由用户(应用程序员)自己把数据按照规定的格式编写,并放在约定的地方,为了不占用过多的内存,还要求数据以根精简的方式编写。boot启动时,传参给内核,告诉内核设备树文件和kernel的位置,内核启动时根据地址去找到设备树文件,再利用专用的编译器去反编译dtb文件,将dtb还原成数据结构,以供驱动的函数去调用。firmware是三星的一个固件的设备信息,因为找不到固件,所以内核启动不成功。_exynos 4412 刷机
文章浏览阅读2w次,点赞24次,收藏42次。Linux系统配置jdkLinux学习教程,Linux入门教程(超详细)_linux配置jdk
文章浏览阅读3.3k次,点赞5次,收藏19次。xlabel('\delta');ylabel('AUC');具体符号的对照表参照下图:_matlab微米怎么输入
文章浏览阅读119次。顺序读写指的是按照文件中数据的顺序进行读取或写入。对于文本文件,可以使用fgets、fputs、fscanf、fprintf等函数进行顺序读写。在C语言中,对文件的操作通常涉及文件的打开、读写以及关闭。文件的打开使用fopen函数,而关闭则使用fclose函数。在C语言中,可以使用fread和fwrite函数进行二进制读写。 Biaoge 于2024-03-09 23:51发布 阅读量:7 ️文章类型:【 C语言程序设计 】在C语言中,用于打开文件的函数是____,用于关闭文件的函数是____。
文章浏览阅读3.4k次,点赞2次,收藏13次。跟随鼠标移动的粒子以grid(SOP)为partical(SOP)的资源模板,调整后连接【Geo组合+point spirit(MAT)】,在连接【feedback组合】适当调整。影响粒子动态的节点【metaball(SOP)+force(SOP)】添加mouse in(CHOP)鼠标位置到metaball的坐标,实现鼠标影响。..._touchdesigner怎么让一个模型跟着鼠标移动
文章浏览阅读178次。项目运行环境配置:Jdk1.8 + Tomcat7.0 + Mysql + HBuilderX(Webstorm也行)+ Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。项目技术:Springboot + mybatis + Maven +mysql5.7或8.0+html+css+js等等组成,B/S模式 + Maven管理等等。环境需要1.运行环境:最好是java jdk 1.8,我们在这个平台上运行的。其他版本理论上也可以。_基于java技术的停车场管理系统实现与设计
文章浏览阅读3.5k次。前言对于MediaPlayer播放器的源码分析内容相对来说比较多,会从Java-&amp;gt;Jni-&amp;gt;C/C++慢慢分析,后面会慢慢更新。另外,博客只作为自己学习记录的一种方式,对于其他的不过多的评论。MediaPlayerDemopublic class MainActivity extends AppCompatActivity implements SurfaceHolder.Cal..._android多媒体播放源码分析 时序图
文章浏览阅读2.4k次,点赞41次,收藏13次。java 数据结构与算法 ——快速排序法_快速排序法