【面试笔记】嵌入式软件工程师,汽车电子软件相关

在这里插入图片描述

文章目录

  • 1. C语言基础
    • 1.1 const
    • 1.2 static
    • 1.3 回调函数的用法
    • 1.4 宏定义
    • 1.5 编译、链接过程
    • 1.6 堆与栈的区别?
    • 1.7 简单的字符串算法题,C语言实现
      • 1.7.1 给定一个字符串,按顺序筛选出不重复的字符组成字符串,输出该字符串
      • 1.7.2 给定4*4矩阵,回文打印输出
    • 1.8 字节对其
  • 2. MCU相关
    • 2.1 MCU的启动过程描述
    • 2.2 MCU的内存布局
    • 2.3 使用volatile关键字的作用?
  • 3. 汽车电子软件
    • 3.1 CAN/CANFD相关
      • 3.1 概述一个CAN消息如何被发送和接收的
      • 3.2 CAN和FIFO CAN
      • 3.3 CANFD的知识点
    • 3.2 概述bootloader实现要点
      • 3.2.1 跳转前要做什么?
    • 3.3 简述ASPICE在项目研发中的应用
    • 3.4 举例说明某个功能安全需求的实现过程
    • 3.5 概述14229协议
    • 3.6 概述15765协议


1. C语言基础

1.1 const

修饰变量
只可访问,不可重新赋值。

const int MAX_VALUE = 100;void printValue(const int value);

修饰指针

  • 限制指向位置
const int *ptr;
  • 限制指向数据
const int *const ptr;

1.2 static

静态变量
使用 static 关键字声明静态变量时,变量的生命周期会延长到整个程序运行期间,而不仅仅局限在其定义的作用域内。静态变量在第一次被赋值时初始化,并且保留其值直到程序结束。

静态全局变量
使用 static 关键字在全局作用域中声明的变量具有静态存储持续时间,但是其作用域被限制在声明该变量的源文件内。这使得该变量对其他源文件不可见,可以防止命名冲突。

静态函数
使用 static 关键字声明静态函数时,该函数仅在声明所在的源文件中可见,即它具有内部链接性。静态函数的作用域仅限于声明所在的源文件。这种方式可以避免与其他源文件中的同名函数产生冲突。


1.3 回调函数的用法

用于在函数执行过程中调用另一个函数。回调函数允许我们向一个函数传递另一个函数的地址,从而在需要时执行特定的操作。回调函数通常用于事件处理、异步编程、库函数的扩展等场景。

  1. 定义回调函数
    首先定义一个函数作为回调函数,其函数原型应与回调的要求相匹配。例如:
void callbackFunction(int result) {printf("Callback result: %d\n", result);
}
  1. 在函数中注册回调函数
    在需要的地方将回调函数注册进目标函数中,通常通过函数指针实现。例如:
void performOperation(void (*callback)(int)) {int result = 100; // 模拟操作结果// 执行操作...// 调用回调函数callback(result);
}
  1. 调用包含回调函数的函数
    最后调用包含回调函数的函数,将回调函数的地址传递给要调用的函数。例如:
int main() {performOperation(callbackFunction); // 注册回调函数return 0;
}

在这个示例中,performOperation 函数执行某个操作后调用了注册的回调函数 callbackFunction,并将结果传递给回调函数进行处理。

通过回调函数,我们可以实现灵活的程序设计,允许函数根据不同情况来调用不同的操作,增加了程序的可扩展性和可重用性。当需要在函数执行过程中动态切换功能时,回调函数是一个非常有用的工具。

使用回调函数有以下一些好处:

  1. 灵活性和可扩展性
    回调函数提供了一种灵活的机制,使得代码可以在不同的场景中进行定制和扩展。通过将特定的功能封装在回调函数中,可以根据需要动态地更改或添加行为,而无需修改主函数的逻辑。
  2. 解耦和模块化
    回调函数有助于将不同的功能模块分离,使代码更具有模块化和可维护性。主函数可以专注于其核心逻辑,而将特定的任务委托给回调函数来处理。这样可以提高代码的可读性和可重用性。
  3. 异步处理和事件驱动
    回调函数常用于异步操作或事件驱动的场景中。例如,在异步 I/O 操作完成或特定事件发生时,可以通过回调函数来处理相应的逻辑。这有助于提高程序的并发性和响应性。
  4. 定制性和扩展性
    回调函数允许用户提供自己的自定义逻辑,以满足特定的需求。这使得程序可以更好地适应各种不同的用例和业务逻辑。
  5. 代码复用
    回调函数可以作为可复用的模块,在多个地方被调用,从而减少代码冗余。

需要注意的是,在使用回调函数时,要确保正确处理回调函数的参数和返回值,并注意线程安全等问题。合理使用回调函数可以提高代码的灵活性和扩展性,但也需要谨慎设计和管理,以避免引入复杂度过高或难以调试的问题。


1.4 宏定义

在 C 语言中,宏定义是一种预处理器指令,用于在编译阶段进行文本替换。它允许你定义一个标识符(通常是一个宏名),并将其与一个特定的文本表达式或代码块关联起来。当在代码中使用该宏名时,编译器会将其替换为相应的文本。
宏定义的常见用法和好处包括:

  1. 常量定义
    使用宏定义可以创建常量,例如定义一些具有特定值的常量,以增强代码的可读性和可维护性。
  2. 代码简化和抽象
    宏定义可以用于简化复杂的表达式或代码块,使其更易于阅读和理解。例如,将常用的计算或操作封装在宏中,以便在多个地方重复使用。
  3. 条件编译
    通过宏定义可以实现条件编译,根据不同的条件编译不同的代码块。这对于处理不同平台、版本或配置的情况非常有用。
  4. 代码移植性
    宏定义可以帮助提高代码的可移植性。例如,可以使用宏来定义平台特定的代码或处理不同编译器的差异。
  5. 提高性能
    在一些情况下,宏定义可以提供一定的性能优势,特别是对于一些简单的计算或操作。
    例如,以下是一个简单的宏定义示例:
#define MAX_SIZE 100

在上面的示例中,MAX_SIZE 是一个宏名,100 是它关联的文本。在代码中使用 MAX_SIZE 时,它将被替换为 100。
需要注意的是,宏定义也有一些潜在的问题和限制:

  1. 宏展开问题
    宏在编译时会进行文本替换,可能会导致一些意外的副作用,例如嵌套宏展开、参数求值顺序等问题。
  2. 缺乏类型检查
    宏不进行类型检查,可能会导致在使用时出现类型不匹配或其他错误。
  3. 可读性问题
    过度使用宏可能会使代码变得难以理解,特别是当宏的定义和使用变得复杂时。

因此,在使用宏定义时,应该谨慎考虑,并确保其使用不会导致代码的可读性和可维护性下降。在一些情况下,使用函数或其他语言特性可能是更好的选择。


1.5 编译、链接过程

在这里插入图片描述
预处理
根据以字符#开头的命令修饰的main.c的C源文件,生成预处理后的C源文件 main.i。
该过程主要进行文本替换、宏展开、删除注释等工作。
对应的gcc命令:

gcc -E main.c main.i

编译
编译器将文本文件main.i翻译(编译)成汇编文件main.s
对应的gcc命令:

gcc -S main.i mian.s

汇编
编译器将main.s翻译成机器语言指令,并把这些指令打包成一种可重定位目标程序的格式,并将结果保存在目标文件main.o中

把一个源文件翻译成目标程序的工作过程分为五个阶段:词法分析、语法分析、语义检查和中间代码生成。主要是进行词法分析和语法分析,又称为源程序分析,分析过程中发现语法错误并给出提示信息。
对应的gcc命令:

gcc -c main.s mian.o

链接
该过程编译器将静态库和动态库的库函数链接到可执行程序中。
静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时就不在需要库文件了,其后缀一般为.a。
动态库则是在程序运行时被链接加载,这样可以节省系统的开销,其后缀一般为.so,gcc在编译时默认使用动态库。


1.6 堆与栈的区别?

请添加图片描述

  1. 栈空间是系统自动分配和回收,堆的空间是用户手动分配回收的;
  2. 栈空间较小,堆空间较大;
  3. 栈的地址空间向下生长,堆则向上生长;
  4. 栈的存储效率更高。
    参考:栈和堆,以STM32为例说明

1.7 简单的字符串算法题,C语言实现

1.7.1 给定一个字符串,按顺序筛选出不重复的字符组成字符串,输出该字符串

参考示例:

#include <stdio.h>
#include <string.h>void removeDuplicates(char *str) {int len = strlen(str);if (len < 2) return;int tail = 1;for (int i = 1; i < len; ++i) {int j;for (j = 0; j < tail; ++j) {if (str[i] == str[j]) break;}if (j == tail) {str[tail] = str[i];++tail;}}str[tail] = '\0'; //此处是关键
}int main() {char str[100];printf("Enter a string: ");scanf("%s", str);removeDuplicates(str);printf("String with duplicates removed: %s\n", str);return 0;
}

测试结果:

Enter a string: asbdssjikSNjs78137!@#ssa00smk
String with duplicates removed: asbdjikSN7813!@#0m

1.7.2 给定4*4矩阵,回文打印输出

参考示例:

#include <stdio.h>#define ROWS 4
#define COLS 4void printClockwise(int matrix[ROWS][COLS]) {int top = 0, bottom = ROWS - 1, left = 0, right = COLS - 1;while (top <= bottom && left <= right) {// Print top rowfor (int i = left; i <= right; ++i)printf("%d ", matrix[top][i]);top++;// Print right columnfor (int i = top; i <= bottom; ++i)printf("%d ", matrix[i][right]);right--;// Print bottom rowif (top <= bottom) {for (int i = right; i >= left; --i)printf("%d ", matrix[bottom][i]);bottom--;}// Print left columnif (left <= right) {for (int i = bottom; i >= top; --i)printf("%d ", matrix[i][left]);left++;}}
}int main() {int matrix[ROWS][COLS] = {{1, 2, 3, 4},{5, 6, 7, 8},{9, 10, 11, 12},{13, 14, 15, 16}};printf("Clockwise printing of the matrix:\n");printClockwise(matrix);return 0;
}

测试结果:

Clockwise printing of the matrix:
1 2 3 4 8 12 16 15 14 13 9 5 6 7 11 10

参考:算法11:顺时针转圈打印矩阵


1.8 字节对其

问题】32位系统,一个结构体中,成员依次是char、short、int、char类型,问这个结构体总共占多少字节?
回答这个问题需要深刻理解结构体所占空间的分布:

|char |-----|short|short|4字节
|int  |int  |int  |int  |4字节
|char |-----|-----|-----|4字节

所以,该结构体共占12字节

测试代码:

#include <stdio.h>struct TMP{char a;short b;int c;char d;
};int main(void) {printf("size = %d", sizeof(struct TMP));return 0;
}

总结:

  • 结构体成员占位是其自身类型长度的整数倍
  • 结构体整体需要对齐,目标对齐长度的整数倍

2. MCU相关

2.1 MCU的启动过程描述

参考STM32的启动过程 — startup_xxxx.s文件解析(MDK和GCC双环境)


2.2 MCU的内存布局

在这里插入图片描述
在这里插入图片描述

参考:

  1. 内存布局:深度剖析应用程序中的内存布局
  2. stm32的内存分布

2.3 使用volatile关键字的作用?

  1. 硬件寄存器操作
    单片机通常与硬件设备交互,硬件寄存器的值可能会在硬件事件的触发下发生改变。通过将访问硬件寄存器的变量声明为 volatile,可以告诉编译器不要对该变量进行优化,以确保每次访问都能获取到最新的寄存器值。
  2. 共享变量
    在多线程或中断处理程序中,多个执行路径可能同时访问和修改同一个变量。将这样的共享变量声明为 volatile,可以确保编译器生成的代码正确地处理变量的读和写,避免出现竞态条件等问题。
  3. 中断服务程序
    中断服务程序可能会修改一些全局变量,而这些变量在主程序中也会被访问。将这些变量声明为 volatile,可以保证中断服务程序对变量的修改能及时反映到主程序中。
  4. 实时性要求高的代码
    在一些对实时性要求较高的场景中,使用 volatile 可以确保关键变量的访问不会被编译器优化掉,从而保证代码的实时性。

通过使用 volatile,可以帮助编译器生成更准确的代码,避免一些由于变量的不确定性导致的问题。然而,具体的应用场景和使用方法可能会因单片机的类型、编译器的特性以及项目的需求而有所不同。在实际编程中,还需要根据具体情况进行适当的测试和调试。


3. 汽车电子软件

3.1 CAN/CANFD相关

3.1 概述一个CAN消息如何被发送和接收的

TBD.


3.2 CAN和FIFO CAN

TBD.


3.3 CANFD的知识点

TBD.


3.2 概述bootloader实现要点

3.2.1 跳转前要做什么?

  • 禁止所有外设时钟;
  • 禁止使用的 PLL;
  • 禁止所有中断;
  • 清除所有中断挂起标志。

3.3 简述ASPICE在项目研发中的应用

TBD.


3.4 举例说明某个功能安全需求的实现过程

TBD.


3.5 概述14229协议

TBD.


3.6 概述15765协议

TBD.

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

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

相关文章

Python3 迭代器和生成器

前言 本文主要介绍Python中的迭代器和生成器&#xff0c;主要内容包括 迭代器概述、生成器简介。 文章目录 前言一、迭代器简介二、生成器简介 一、迭代器简介 在 Python 中&#xff0c;迭代器(iterator)是一个实现了迭代器协议&#xff08;Iterator Protocol&#xff09;的…

opencv进阶 ——(十一)基于RMBG实现生活照生成寸照

实现步骤 1、检测人脸&#xff0c;可以使用opencv自带的级联分类器或者dlib实现人脸检测 2、放大人脸范围&#xff0c;调整到正常寸照尺寸 3、基于RMGB算法得到人像掩码 4、生成尺寸相同的纯色背景与当前人像进行ALPHA融合即可 alpha融合实现 void alphaBlend(cv::Mat&…

1 机器人软件开发学习所需通用技术栈(一)

机器人软件工程师技术路线&#xff08;如有缺失&#xff0c;欢迎补充&#xff09; 1. 机器人软件开发工程师技术路线 1.1 基础知识 C/C编程&#xff1a;掌握C/C语言基础&#xff0c;包括数据结构、算法、内存管理等。操作系统&#xff1a;了解Linux或Windows等操作系统的基本…

2.1 初识Windows程序

Windows程序设计是一种面向对象的编程。Windows操作系统以数据结构的形式定义了大量预定义的对象作为操作系统的数据类型。Windows动态链接库提供了各种各样的API接口函数供Windows应用程序调用。一个Windows应用程序是运行在Windows操作系统之上的。这些API接口函数的调用所实…

【Vue】路由的基本使用

文章目录 一、固定5个固定的步骤二、代码示例三、两个核心步骤四、完整代码 vue-router插件作用 修改地址栏路径时&#xff0c;切换显示匹配的组件 说明 Vue 官方的一个路由插件&#xff0c;是一个第三方包 官网 https://v3.router.vuejs.org/zh/ VueRouter的使用&#xff0…

TCP/IP协议介绍——三次握手四次挥手

TCP/IP&#xff08;Transmission Control Protocol/Internet Protocol&#xff0c;传输控制协议/网际协议&#xff09;是指能够在多个不同网络间实现信息传输的协议簇。TCP/IP协议不仅仅指的是TCP 和IP两个协议&#xff0c;而是指一个由FTP、SMTP、TCP、UDP、IP等协议构成的协议…

CSS学习|css三种导入方式、基本选择器、层次选择器、结构伪类选择器、属性选择器、字体样式、文本样式

第一个css程序 css程序都是在style标签中书写 打开该网页&#xff0c;可以看到h1标签中的我是标题被渲染成了红色 可以在同级目录下创建一个css目录&#xff0c;专门存放css文件&#xff0c;可以和html分开编写 然后在html页面中&#xff0c;利用link标签以及css文件地址&…

大模型基架:Transformer如何做优化?

大模型的基础模式是transformer&#xff0c;所以很多芯片都实现先专门的transformer引擎来加速模型训练或者推理。本文将拆解Transformer的算子组成&#xff0c;展开具体的数据流分析&#xff0c;结合不同的芯片架构实现&#xff0c;分析如何做性能优化。 Transformer结构 tr…

go的反射和断言

在go中对于一个变量&#xff0c;主要包含两个信息变量类型&#xff08;type&#xff09;和变量值&#xff08;value&#xff09; 可以通过reflect包在运行的时候动态获取变量信息&#xff0c;并能够进行操作 对于Type可以通过reflect.TypeOf()获取到变量的类型信息 reflect.Ty…

13_前端工程化_ES6

1.前端工程化概念 前端工程化是使用软件工程的方法来单独解决前端的开发流程中模块化、组件化、规范化、自动化的问题,其主要目的为了提高效率和降低成本。 前后端分离&#xff08;前端代码工程化独立出来形成一个单独的app&#xff09; 1.开发分离 2.部署分离 3.服务器分离…

信号(上)

本节目标&#xff1a; 1. 掌握Linux信号的基本概念 2. 掌握信号产生的一般方式 3. 理解信号递达和阻塞的概念&#xff0c;原理。 4. 掌握信号捕捉的一般方式。 5. 重新了解可重入函数的概念。 6. 了解竞态条件的情景和处理方式 7. 了解SIGCHLD信号&#xff0c; 重新编写信号处理…

ChatGPT基本原理详细解说

ChatGPT基本原理详细解说 引言 在人工智能领域&#xff0c;自然语言处理&#xff08;NLP&#xff09;一直是研究的热点之一。随着技术的发展&#xff0c;我们见证了从简单的聊天机器人到复杂的语言模型的演变。其中&#xff0c;ChatGPT作为一项突破性技术&#xff0c;以其强大…

2004NOIP普及组真题 2. 花生采摘

线上OJ&#xff1a; 【04NOIP普及组】花生采摘 核心思想&#xff1a; 1、本题为贪心即可。 2、因为本题严格限制了顺序&#xff0c;所以先把每个节点的花生数量按降序排序。然后逐一判断下一个花生是否需要去采摘即可 3、每一次采摘完&#xff0c;记录耗时 t 以及采集的花…

基于web的垃圾分类回收系统的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;管理员管理&#xff0c;用户管理&#xff0c;公告管理&#xff0c;运输管理&#xff0c;基础数据管理 用户账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;运输管理&#xff0c;公告…

pyqt QlineEdit内部增加按钮方法

pyqt QlineEdit内部增加按钮方法 def addButton(self,lineEdit):btn QtWidgets.QPushButton("")icon1 QtGui.QIcon()icon1.addPixmap(QtGui.QPixmap(":/image/images/th.png"), QtGui.QIcon.Normal, QtGui.QIcon.Off)btn.setIcon(icon1)btn.setStyleShe…

全光谱led灯的危害有哪些?曝光低质量全光谱led灯产生的四大风险

眼睛是人类获取信息最重要的感官器官之一&#xff0c;而近视则会导致视力模糊&#xff0c;进而影响学习效果和生活品质。因此&#xff0c;如何保护眼睛&#xff0c;尤其是在学习和使用电子设备时&#xff0c;成为了一个迫切需要解决的问题。然而在护眼领域上&#xff0c;护眼台…

SCAU 数据结构 实验六 排序算法

![[Pasted image 20240 8638 直接插入排序 Description 用函数实现直接插入排序&#xff0c;并输出每趟排序的结果. 输入格式 第一行&#xff1a;键盘输入待排序关键的个数n 第二行&#xff1a;输入n个待排序关键字&#xff0c;用空格分隔数据 输出格式 每行输出一趟排序…

十三、resultMap解析

分为两部分&#xff1a;解析和使用 解析 1.解析XML的时候单独解析所有的resultMap标签&#xff0c;封装成ResultMap对象存入configuration中 2.解析XML中的SQL语句&#xff0c;封装MappedStatement对象&#xff0c;这里会根据SQL的返回类型是resultMap还是resultType做处理。如…

C语言 | Leetcode C语言题解之第133题克隆图

题目&#xff1a; 题解&#xff1a; struct Node** visited; int* state; //数组存放结点状态 0&#xff1a;结点未创建 1&#xff1a;仅创建结点 2&#xff1a;结点已创建并已填入所有内容void bfs(struct Node* s) {if (visited[s->val] && state[s->val] 2…

Python Lambda函数的应用实例教程

在Python编程中&#xff0c;lambda函数是一种简洁且强大的工具&#xff0c;用于创建小型匿名函数。它们在需要快速定义简单函数时特别有用。本文将详细介绍lambda函数的语法及其多种应用实例&#xff0c;帮助读者更好地理解和使用lambda函数。 一、lambda函数的基本概念 1.1 什…