LVGL基础部件学习笔记

news2025/1/18 5:41:34

目录

1、基础对象基本属性用法(所有部件都可以用)

1.1、基础对象的大小(Size)

1.2、基础对象的位置(Position)

1.3、基础对象对齐(Alignment)

1. 4、基础对象的盒子模型(border-box)

1.5、基础对象的样式(styles)

1.6、基础对象的事件(events)

2、标签(lv_label)

2.1、显示图标

2.2、显示中文

3、按钮(lv_btn)

4、使用物理按键代替触摸(groups)


        LVGL采用面向对象的编程思想(OOP),她的基本构造块(类)是对象(实例),也就是我们所说的部件(Widgets)就是一个个部件,比如button、label、image等等。

        lvgl编程思想:lv_obj_t (类)定义了部件的抽象特点,其定义包含了数据的形式以及对数据的操作。部件(子类)比原本的类(称为父类或基类)要更加具体化,子类会继承父类的属性和行为。

        在LVGL中,所有的对象都在 lv_obj_t 这个结构体的基础上进行演变,所以我们就看到了各种不一样的部件,就算是一样的部件,继承基础父类(基类)之后演变出来对象(实例)的形态或风格样式都不一样,如下图举例一下。

 屏幕是没有父类的基础对象

lv_obj_create()       这个函数是让我们所选的部件实例化,实例化之后就可以对其进行操作。

其参数填写创建在谁之上,可以在其他部件上也可以在我们屏幕上。屏幕分为三类:

1、lv_scr_act(void);     // 活动屏幕

2、lv_layer_top (void);     // 顶层     

3、lv_layer_sys (void);    // 系统层

1、基础对象基本属性用法(所有部件都可以用)

1.1、基础对象的大小(Size)

设置宽度: lv_obj_set_width(obj, new_width);

设置高度: lv_obj_set_height(obj, new_height);

同时设置宽度、高度: lv_obj_set_size(obj, new_ width, new_ height);

获取宽度: lv_obj_get_width(obj);

获取高度: lv_obj_get_height(obj);

1.2、基础对象的位置(Position)

LVGL的坐标系是我们 可以称为“LCD坐标系”

 如下是屏幕的位置坐标:(x和y根据自己需求进行调整)

设置x轴方向的坐标位置: lv_obj_set_x(obj, new_x);

设置y轴方向的坐标位置: lv_obj_set_y(obj, new_y);

同时设置x、y坐标位置:   lv_obj_set_pos(obj, new_x, new_y);  // position

获取x轴坐标位置: lv_obj_get_x(obj);

获取y轴坐标位置: lv_obj_get_y(obj);

1.3、基础对象对齐(Alignment)

参照父对象对齐: lv_obj_set_align(obj, LV_ALIGN_...);

参照父对象对齐后再设置坐标位置: lv_obj_align(obj, LV_ALIGN_..., x, y);

参照另一个对象(无父子关系)对齐后设置坐标位置: lv_obj_align_to(obj_to_align, obj_referece, LV_ALIGN_..., x, y)

1. 4、基础对象的盒子模型(border-box)

LVGL 遵循 CSS 的 border-box 模型。 对象的“盒子”由以下部分构成:

边界(bounding):元素的宽度/高度围起来的区域(整个盒子)。

边框(border):边框有大小和颜色等属性(相当于盒子的厚度和它的颜色)。

填充(padding):对象两侧与其子对象之间的空间(盒子的填充物)。

内容(content):如果边界框按边框宽度和填充的大小缩小,则显示其大小的内容区域(盒子实际装东西的区域)。

轮廓(outline) :LVGL中没有外边距(margin)的概念(盒子之间的距离),确认代之的是轮廓(outline)。它是绘制于元素(盒子)周围的一条线,它不占据空间,位于边框边缘的外围,可起到突出元素(盒子)的作用。在浏览器里,当鼠标点击或使用Tab键让一个选项或者一个图片获得焦点的时候,这个元素就会多了一个轮廓框围绕。轮廓(outline) 

 下面是对CSS盒子模型的一些扩充(左是框图,右是例子)

网页设计中常听的属性名:内容(content)、内边距(padding)、边框(border)、外边距(margin), CSS盒子模式都具备这些属性。这些属性我们可以把它转移到我们日常生活中的盒子(箱子)上来理解,日常生活中所见的盒子也就是能装东西的一种箱子,也具有这些属性,所以叫它盒子模型。CSS盒子模型就是在网页设计中经常用到的CSS技术所使用的一种思维模型。

content就是盒子里装的东西,它有高度(height)和宽度(width),可以是图片,可以是文字或者小盒子嵌套,在现实中,内容不能大于盒子,内容大于盒子就会撑破盒子,在LVGL中盒子不会变化,会产生滚动条,我们可以滚动查看超出盒子的内容;但在css中,盒子有弹性的,顶多内容太大就会撑大盒子,但是不会损害盒子。

padding即是填充,就好像我们为了保证盒子里的东西不损坏,填充了一些东西,比如泡沫或者塑料薄膜,填充物有大有小,有软有硬,反应在网页中就是padding的大小了。

border就是再外一层的边框,因为边框有大小和颜色的属性,相当于盒子的厚度和它的颜色或者材料。

margin外边距,就是我们的盒子与其他的盒子或者其他东西的距离。假如有很多盒子,margin就是盒子之间直接的距离,可以通风,也美观同时方便取出。

1.5、基础对象的样式(styles)

Styles 用于设置对象的外观

  1. 样式是一个 lv_style_t 变量,它可以保存边框宽度、文本颜色等属性。
  2. 将样式(变量)分配给对象就可以改变其外观。在赋值过程中,可以指定目标部分和目标状态。
  3. 一个样式可以给多个对象使用(正常样式)。
  4. 样式可以级联,也就是可以将多个样式分配给一个对象。所以,我们不用将所有属性都在一个样式中指定,可以通过多个样式组合的形式指定。 LVGL 会优先使用我们定义的样式,如果没有就会使用默认值。
  5. 后来添加的样式具有更高的优先级。也就是说如果在两种样式中指定了同一个属性,则将使用最后添加的样式
  6. 如果对象中未指定某些属性(例如文本颜色),就会从父级继承。
  7. 上面说的是 “正常” 样式,对象还有本地样式,它比 “正常” 样式具有更高的优先级。
  8. 可以定义有过渡效果的样式。
  9. 默认有一个样式主题,我们也可以自己定义样式主题,作为默认的样式主题使用。

样式操作流程:

static lv_style_t style_obj;      样式变量应该是 静态 、全局或动态分配的

lv_style_init(&style_obj);        初始化

lv_style_set_<property_name>(&style, <value>);        设置样式属性(很多)

lv_obj_add_style(obj, &style, <selector>)        添加(应用)样式到对象

lv_obj_add_style(obj, &style_obj, 0);                       // 默认(常用)   

lv_obj_add_style(obj, &style_obj, LV_STATE_PRESSED);  // 在对象被按下时应用样式

获取样式属性

lv_obj_get_style_<property_name>(obj, <part>);

lv_color_t color = lv_obj_get_style_bg_color(obj, LV_PART_MAIN);       返回默认值 

删除样式

lv_obj_remove_style_all(obj);        删除对象的所有样式:

lv_obj_remove_style(obj, &style_obj, selector);        删除对象的特定样式:

如果 selector 是 LV_STATE_ANY 或 LV_PART_ANY 就会删除具有任何状态或部分的样式。

(如删除对象所有样式一样)

背景属性和我们前面学习的盒子模型关系很大,背景属性主要有一下这些:

  • 背景(Background)
  • 边界(Border)
  • 轮廓(Outline)
  • 阴影(Shadow)
  • 填充(Padding)
  • 宽度和高度变换 X和Y变换

状态(States)对象可以处于以下状态的组合:

LV_STATE_DEFAULT (0x0000) 正常,释放状态

LV_STATE_CHECKED (0x0001) 切换或检查状态

LV_STATE_FOCUSED (0x0002) 通过键盘或编码器聚焦或通过触摸板/鼠标点击 LV_STATE_FOCUS_KEY (0x0004) 通过键盘或编码器聚焦,但不通过触摸板/鼠标聚焦 LV_STATE_EDITED (0x0008) 由编码器编辑

LV_STATE_HOVERED (0x0010) 鼠标悬停(现在不支持)

LV_STATE_PRESSED (0x0020) 被按下

LV_STATE_SCROLLED (0x0040) 正在滚动

LV_STATE_DISABLED (0x0080) 禁用状态

LV_STATE_USER_1 (0x1000) 自定义状态

LV_STATE_USER_2 (0x2000) 自定义状态

LV_STATE_USER_3 (0x4000) 自定义状态

LV_STATE_USER_4 (0x8000) 自定义状态

对象可以有 部分(parts) ,它们也可以有自己的样式。LVGL 中存在以下预定义部分:

LV_PART_MAIN  类似矩形的背景

LV_PART_SCROLLBAR  滚动条

LV_PART_INDICATOR  指标,例如用于滑块、条、开关或复选框的勾选框 LV_PART_KNOB  像手柄一样可以抓取调整值

LV_PART_SELECTED  表示当前选择的选项或部分

LV_PART_ITEMS  如果小部件具有多个相似元素(例如表格单元格)

LV_PART_TICKS  刻度上的刻度,例如对于图表或仪表

LV_PART_CURSOR  标记一个特定的地方,例如文本区域或图表的光标 LV_PART_CUSTOM_FIRST 可以从这里添加自定义部件

除了“普通” 样式外,对象还可以存储 本地样式(私有样式) 

本地样式与普通样式类似,但是它不能在其他对象之间共享

使用本地样式,将自动分配局部样式,并在删除对象时释放

本地样式的接口函数是这样的格式:

lv_obj_set_style_<property_name>(obj, <value>, <selector>);

删除本地样式的时候我们删除某一个样式:

 lv_obj_remove_local_style_prop(obj, LV_STYLE_..., selector);

1.6、基础对象的事件(events)

        当发生用户可能感兴趣的事情时,LVGL 中会触发事件。

添加事件:

lv_obj_add_event_cb(obj, event_cb, event_code, user_data);

发送事件:

lv_event_send(obj, event_cb, event_code, user_data);

删除事件:

lv_obj_remove_event_cb(obj, event_cb);

lv_obj_remove_event_dsc(obj, event_dsc);//event_dsc 是 lv_obj_add_event_cb 返回的指针

事件类型(event_code)

  • 输入设备事件(Input device events)
  • 绘图事件(Drawing events)
  • 其他事件(Special events)
  • 特殊事件(Other events)
  • 自定义事件(Custom events)

事件回调函数的 lv_event_t 参数

static void my_event_cb(lv_event_t * event);

获取触发的事件代码:     lv_event_code_t code = lv_event_get_code(e);

获取触发事件的对象:     lv_obj_t * target = lv_event_get_target(e);

获取最初触发事件的对象(事件冒泡):      lv_obj_t * target = lv_event_get_current_target(e); 获取事件传递的用户数据:     

lv_event_get_user_data(e); 获取使用 lv_obj_add_event_cb 传递的用户数据   

lv_event_get_param(e); 获取使用 lv_event_send 传递的用户数据

  • 一个事件回调函数可给多个对象使用
  • 一个对象可以使用多个事件回调函数
  • 如果传入的用户数据不一样,一个对象可以绑定同一个事件回调函数多次,事件将按照添加的顺序调用

事件冒泡

如果对象启用了 lv_obj_add_flag(obj, LV_OBJ_FLAG_EVENT_BUBBLE),该对象的所有事件将会发送到该对象的父级。如果父级也启用了 LV_OBJ_FLAG_EVENT_BUBBLE,那么事件继续发送到他的父级,依此类推。

lv_event_get_target(e); 获取触发事件的当前对象。

lv_event_get_current_target(e); 获取事件冒泡的父对象。

2、标签(lv_label)

标签是用来显示文本的基本对象类型

标签的组成包括:

LV_PART_MAIN 矩形部分(盒子区域)。 填充值可用于在文本和背景之间添加空间。    LV_PART_SCROLLBAR 当要展示的文本大于部件的大小时,显示的滚动条部分。    LV_PART_SELECTED 选中文本时,突出显示的部分。label只能使用 text_color 和 bg_color 样式属性。

创建标签:     lv_obj_t * label = lv_label_create(parent);

直接设置要显示的文本:     lv_label_set_text(label, "New text");

格式化给定要显示的文本:     lv_label_set_text_fmt(label, “%s: %d”, “Value”, 15);

文本不存储在动态内存中,而是直接使用给定的缓冲区:     

lv_label_set_text_static(label, "New text");

要在label换行非常简单,像printf函数那样使用 \n 即可:     

lv_label_set_text(label, " line1\nline2\n\nline4 ");

如果出现文本的宽度或高度大于或小于label的情况,就需要做一些调整,下面是几种模式

  • LV_LABEL_LONG_WRAP 如果有多个换行,并且如果高度为LV_SIZE_CONTENT,那么高度会根据文本换行被自动扩展;否则文本将被剪掉。(默认设置)     
  • LV_LABEL_LONG_DOT 如果文本太长,就保持大小并在末尾写3个点 .
  • LV_LABEL_LONG_SCROLL 如果文本比标签宽(太长),则可以水平来回滚动显示它。如果它很高(多个\n换行),可以垂直滚动。只滚动一个方向,水平滚动的优先级更高。   
  • LV_LABEL_LONG_SCROLL_CIRCULAR 如果文本比标签宽,则水平滚动它。如果它更高,就垂直滚动。只滚动一个方向,水平滚动的优先级更高。     
  • LV_LABEL_LONG_CLIP 剪掉超出标签范围外的文本部分。

 lv_label_set_long_mode(label, LV_LABEL_LONG_...) 指定模式

注意:

LV_LABEL_LONG_DOT 是直接操作文本缓冲区以添加/删除点。如果使用 lv_label_set_text 和 lv_label_set_text_fmt 它们会分配一个单独的缓冲区,不会出问题。

但是如果使用 lv_label_set_text_static 时我们传递给它的缓冲区必须是可写的。

文本重新着色,可以通过样式来上色

lv_style_set_text_color(&style_obj, lv_color_hex(0xf7b37b));     //样式着色 lv_obj_set_style_text_color(label, lv_color_hex(0xf7b37b), 0);        //样式着色

lv_label_set_recolor(label1, true);     //label从新着色

lv_label_set_text(label1, "#0000ff Re-color# #ff00ff words# #ff0000 of a# label);

        如果在 lv_conf.h 中打开了 LV_LABEL_TEXT_SELECTION (默认开启),就可以选择部分文本。这个效果只能在文本框(lv_textarea)中实现。Label只能事先手动选择指定范围的文本

lv_label_set_text_sel_start(label, 1);     lv_label_set_text_sel_end(label, 6);

注意,这里的第一个字符从1开始算,而不是0。

2.1、显示图标

LVGL内置了一些图标,它们是全局变量我们可以直接使用:

用法:

lv_label_set_text(my_label, LV_SYMBOL_OK);    // 直接显示图标    lv_label_set_text(my_label, LV_SYMBOL_OK “Apply”);     // 图标与字符串一起使用    lv_label_set_text(my_label, LV_SYMBOL_OK LV_SYMBOL_WIFI LV_SYMBOL_PLAY);     

Label默认不接收输入事件,如果我们想设置输入类型的样式或者事件会无法生效,就需要打开 LV_OBJ_FLAG_CLICKABLE,示例:     

lv_obj_add_flag(label, LV_OBJ_FLAG_CLICKABLE);  // 使输入设备可点击对象

2.2、显示中文

LVGL有一个中文字库 CJK字库 ,字库在 lv_conf.h中定义为:LV_FONT_SIMSUN_16_CJK。

要在lvgl中使用显示自己的中文字库,我们需要用到两个东西:字体文件和字体转换器。 字体文件我们可以使用开源的字体或者自己制作出来,准备好了字体文件之后使用字体转换器即可转换成可以在lvgl上使用的字体格式。

开源字体获取页面:     http://lvgl.100ask.net/8.1/tools/fonts-zh-source.html

字体转换工具界面:

http://lvgl.100ask.net/8.1/tools/fonts-zh-source.html

https://lvgl.io/tools/fontconverter

2.2.1、如何使用字体转换器?

  1. 为要输出字体命名。例如“font_source_han_sans_bold_20”
  2. 以 px为单位指定高度(字体大小)
  3. 设置bpp (bit-per-piel)。值越高,字体越平滑(抗锯齿)
  4. 选择TTF 或 WOFF 格式字体文件
  5. 设置要包含在字体中的 Unicode 字符范围或在符号字段中列出字符
  6. 可以同时选择多个字体文件转换,并为其指定范围和/或符号。这些字符将被合并转换到同一个文件中。
  7. 单击转换按钮以下载转换出来的 font_source_han_sans_bold_20.c 文件。

2.2.2如何在 LVGL 中使用生成的字体? 

  1. 将结果 C 文件复制到你的 LVGL 项目中,并包含到项目;
  2. 在你的项目应用程序的 C 文件中,将字体声明为:                                                extern lv_font_t my_font_name;        或:     LV_FONT_DECLARE(my_font_name);
  3. 在样式中设置字体:                                                                         lv_style_set_text_font(&style_obj, &my_font_name);    // 普通(共享)样式     lv_obj_set_style_text_font(label, &my_font_name, 0);    // 私有(本地)样式
  4. unicode 是统一所有语言的一套编码。
  5. utf-8是基于unicode编码的一种节约字节的编码。

3、按钮(lv_btn)

按钮是和基础对象最像的部件。与基础对象相比,按钮没有新的功能

按钮和基础对象的不同点:     

不可滚动     

添加到默认组(可修改为给其他输入设备控制用,比如按键)     

高度和宽度默认为 LV_SIZE_CONTENT

创建按键的方法:      lv_obj_t * btn = lv_btn_create(parent);

按钮的部分(Part)只有下面这个:     LV_PART_MAIN 按钮的背景

样式修改示例:     

lv_obj_set_style_<property_name>(obj, <value>, <selector>);    // 本地(私有)样式    lv_obj_add_style(obj, &style, <selector>)            // 普通(共享)样式

所以修改样式的时候我们修改部分(Part)的时候只有 LV_PART_MAIN 可以生效;状态部分和前面说的样式状态(States)都可以用

如果打开了 LV_OBJ_FLAG_CHECKABLE ,当对象被点击时有选中切换(Toggle)状态的效果,并且可以在 LV_EVENT_VALUE_CHANGED 事件类型中处理事件,比如:     lv_obj_add_flag(btn, LV_OBJ_FLAG_CHECKABLE);

        总的来看,按钮是用来按的(单击、按下或不按下),所以我们用物理按键控制的时候只需要LV_KEY_ENTER 就可以控制按钮, 并且 LV_KEY_ENTER 可以触发LV_EVENT_PRESSED/PRESSING/RELEASED 等事件类型。

4、使用物理按键代替触摸(groups)

        在LVGL中我们可以使用键盘或编码器替换触摸板或鼠标控制。它的工作原理类似于 PC 上的 TAB 键,用于选择应用程序或网页中的元素。

  1. 首先我们需要将要用键盘或编码器控制的部件添加到组,有的部件在创建时会加入到了默认组。
  2. 在每个组中,同时只有一个对象聚焦并接收按键或编码器动作。
  3. 将对象添加到组还不够,我们还需要将输入设备与组关联。一个输入设备只能将按键发送给一组,反过来一组可以从多个输入设备接收数据。

5种输入设备类型:     

LV_INDEV_TYPE_NONE         不使用输入设备(未初始化状态)    LV_INDEV_TYPE_POINTER     触摸板、鼠标、外接按钮     

LV_INDEV_TYPE_KEYPAD      键盘     

LV_INDEV_TYPE_BUTTON     分配给屏幕特定点的外部(硬件按钮)    LV_INDEV_TYPE_ENCODER   只有左、右和按下三个按键的编码器(比如鼠标中间的滚轮)

自定义组创建过程:

  1. 首先要创建一个 组(Groups) : lv_group_t * g = lv_group_create();
  2. 然后将一个对象添加到 组(Groups) 中: lv_group_add_obj(g, obj);
  3. 最后要将组(Groups)与输入设备相关联: lv_indev_set_group(indev, g);     其中 indev 是 lv_indev_drv_register(); 的返回值

使用默认组

在lvgl中有些部件,在创建的时候加入到默认组中,但是默认组变量lvgl并没有帮我们初始化好,我们需要在创建部件之前初始化好才能使用默认组,这个自定义组一样不同的是对于创建时添加到默认的部件我们可以跳过lv_group_add_obj:     

lv_group_t * g = lv_group_create();    // 创建一个组     

lv_group_set_default(g);        // 设置为默认组     

lv_indev_set_group(indev, g);    // 将组和输入设备相关联

有一些预定义的键具有特殊含义:

必须的按键

最重要且建议必要的按键是:     

LV_KEY_NEXT/PREV     

LV_KEY_ENTER     

LV_KEY_UP/DOWN/LEFT/RIGHT

在我们回调函数 read_cb 中,应该优先考虑将现有的键对应转换为上面这些键,以便能在组中导航并与所选对象进行交互。

其实一般来说,只使用 LV_KEY_LEFT/RIGHT 就足够了,因为大多数对象都可以通过它们被完全控制。

对于编码器,默认只使用 LV_KEY_LEFT、LV_KEY_RIGHT 和 LV_KEY_ENTER

编辑(edit)和导航(navigate)模式

  • 如果我们使用编码器,因为编码器的按键有限,所以在各个对象之间进行导航就不方便了。这个时候就需要区分模式了,lvgl帮我们区分了两种模式:编辑和导航模式
  • 在导航模式下,编码器的 LV_KEY_LEFT/RIGHT 被转换为 LV_KEY_NEXT/PREV,这样就可以通过转动编码器来选择下一个或上一个对象。
  • 如果需要修改对象的值,比如有个滑杆(lv_slider)代表音量或亮度,我们通过按下 LV_KEY_ENTER 将模式切换为编辑模式,这样就可以通过转动编码器修改滑杆的值。然后根据对象的类型,短按或长按 LV_KEY_ENTER 切换为导航模式(离开编辑模式)。

按键控制的初始化、工作流程和使用

 其实还有许多部件,这里就不一一介绍了,很多部件原理都差不多,这里推荐一个学习LVGL非常不错的网址        量规(lv_gauge) — 百问网LVGL中文教程手册文档 1.0 文档

韦东山老师的视频真不错,底层去解剖程序,看一遍不懂就两遍。

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

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

相关文章

小偷与守望者.math库.动态规划.贪心.C

1.分别讨论0或1;边界 2.写出递推方程;最优子结构、状态转移方程、重叠子问题 int rob(int* arr, int size){ int dp[size];int result; if(size0)return 0; if(size1)return arr[0]; else{ dp[0]arr[0]; dp[1]arr[1] > arr[0] ? arr[1]:arr[0]; for (int i 2; i < siz…

php学习笔记-连接操作mysq数据库(基础)-day08

php学习笔记-连接操作mysq数据库-day08php中常用的mysql操作函数1.连接mysql服务器mysqli_connect()2.设置数据库字符集 和 发送sqlmysqli_query() 函数 可以用来设置数据库字符集。3.关闭mysql服务器连接mysqli_close();4.选择要操作的数据库mysqli_select_db()5.结果集释放my…

C#大型在线学习平台源码(PC端+手机端)

本套源码目录结构清晰、代码层次分明、文档齐全。源码分享&#xff01;需要源码学习可私信我获取。 系统简介 中小学教育平台是一个集学、练、考于一体的在线教育平台&#xff0c;可以帮助中小学教育机构建设网络课堂&#xff0c;用于同步课程、专题课堂、兴趣课堂等网络教育的…

当用户输入一堆这样的字符串到 Elasticsearch ?

1、问题引出如下样例数据已导入 Elasticsearch&#xff0c;如何实现特定字段检索&#xff1f;并计算出特定子字段的长度&#xff1f;"message": "[策略排序]排序后结果:[{\"intentItems\":[\"200001\"],\"level\":1,\"modul…

2021帆软杯一等奖 | 《游乐园综合管理系统》

# 作品主题——游乐园综合管理系统 #作品简介&#xff1a;该获奖作品来自零代码开发赛道。针对一些个人经营、没有统一规范的游乐园&#xff0c;希望通过创建一个简单、高效的系统应用&#xff0c;管理人员能够通过对表单的简单操作就能管理游乐园&#xff0c;如获取工作人员信…

程序分析-动态程序切片

最近在调研符号执行工具优化方式时&#xff0c;发现好几篇工作都用到了动态程序切片&#xff0c;以前大部分接触的都是静态切片&#xff0c;对动态切片几乎不了解。所以开始学习动态切片&#xff0c;我主要参考的是90年的一篇上古paper。 1.静态依赖图和静态切片 说到程序切片…

Obsidian Templater 批量新建笔记

Obsidian Templater 批量新建笔记 背景 工作中的新项目都有这么几个固定的笔记&#xff0c;比如项目背景、原理图设计、PCB设计、调试等&#xff0c;每个笔记文件又有一些固定的内容&#xff0c;想着在开始一个新项目时&#xff0c;一次性把这些笔记都建好&#xff0c;并且统…

一文教会你 如何在Github中创建仓库?如何将多个项目放到一个仓库中管理?如何将本地项目上传到GitHub中?

文章目录前言1、如何在GitHub中创建仓库1.1 点击New repository1.2 填写仓库的基本信息1.3 完成创建2 、克隆仓库地址到本地2.1 克隆地址2.2 克隆到本地2.3 将后端代码项目提交到远程服务器2.4 将前端代码项目提交到远程服务器3、Github上查看自己上传项目代码4、使用这个方式存…

【程序分析】Code Lifting

我理解&#xff0c;Code Lifting和反编译的概念类似 常规的编译过程是这样的&#xff1a; 这是一个从高级形式到低级形式的过程&#xff0c;一般叫做 lowering 而反编译&#xff0c;是这样一个过程&#xff1a; 这个过程是完全相反的&#xff0c;叫做lifting 需要注意 Bi…

FPGA:逻辑函数的卡诺图化简法

文章目录最小项与最小项表达式最小项的定义最小项的性质逻辑函数的最小项表达式卡诺图化简法用卡诺图表示逻辑函数卡诺图的引出两变量卡诺图三变量卡诺图四变量卡诺图已知逻辑函数真值表&#xff0c;画卡诺图已知逻辑函数画卡诺图用卡诺图化简逻辑函数化简的依据化简的步骤用卡…

Hexo + Butterfly 侧边栏公众号

原文链接 &#xff1a;Hexo Butterfly 侧边栏公众号 推荐阅读 基于 Hexo 从零开始搭建个人博客&#xff08;一&#xff09;: 环境准备基于 Hexo 从零开始搭建个人博客&#xff08;二&#xff09;: 项目初识基于 Hexo 从零开始搭建个人博客&#xff08;三&#xff09;: 主题安…

Lichee_RV学习系列--stream移植

Lichee_RV学习系列文章目录 Lichee_RV学习系列—认识Lichee Rv Dock、环境搭建和编译第一个程序 Lichee_RV学习系列—移植dhrystone 文章目录Lichee_RV学习系列文章目录一、stream简介二、源码下载三、文件移植1、makefile文件编译makefile文件移植四、运行结果五、移植过程中…

【尚硅谷】Java数据结构与算法笔记08 - 查找算法

文章目录一、查找算法介绍二、线性查找算法三、二分查找算法3.1 思路分析3.2 代码实现四、插值查找算法4.1 思路分析4.2 代码实现4.3 注意事项五、斐波那契&#xff08;黄金分割法&#xff09;查找算法5.1 思路分析5.2 原理讲解5.3 代码实现一、查找算法介绍 在 java 中, 我们…

C++多线程入门及基础知识

什么是C多线程 线程即操作系统进行CPU任务调度的最小单位。C的多线程并发&#xff0c;简单理解的话就是&#xff0c;将任务的不同功能交由多个函数实现&#xff0c;创建多个线程&#xff0c;每个线程执行一个函数&#xff0c;一个任务就同时由不同线程执行。 什么时候使用多…

微信小程序:骨架屏的实现方法

骨架屏是为了展示一个页面骨架而不含有实际的页面内容&#xff0c;从渲染效率上来讲&#xff0c;骨架屏它并不能使首屏渲染加快。由于骨架屏的一些使用又向用户渲染了额外的一些内容&#xff0c;这些内容是额外添加的、本来是不需要渲染的&#xff0c;它反而从整体上加长了首屏…

Windows 虚拟磁盘驱动开发(采用原始办法实现类似Storport框架的相同功能)

其实以前讲述windows平台下的磁盘驱动的开发挺多&#xff0c;而且时间也是非常早。以下连接&#xff1a;https://blog.csdn.net/fanxiushu/article/details/9903123?spm1001.2014.3001.5501https://blog.csdn.net/fanxiushu/article/details/11713357?spm1001.2014.3001.5501…

游戏开发 状态同步

【状态同步】1、将所有的操作发送给Server&#xff08;T1&#xff09;&#xff0c;由Server计算&#xff08;T2&#xff09;&#xff0c;并返回结果&#xff08;T3&#xff09;。权威服务器架构能够防止很多的作弊&#xff0c;但是直接用这种方法会让游戏的响应变得迟缓。如果 …

three games 之 桌球

接下来介绍一些 Vue4 中的一些进阶使用&#xff0c;希望对大家有所帮助&#xff0c;谢谢。 如果文中有不对、疑惑的地方&#xff0c;欢迎在评论区留言指正&#x1f33b; 一、项目结构 Vuex 并不限制你的代码结构。但是&#xff0c;它规定了一些需要遵守的规则&#xff1a; …

移动硬盘怎么分区?

硬盘分区指的是硬盘上被划分出来的区块&#xff0c;可用于分类存储各种数据。而我们日常购买的移动硬盘通常来说分为两种&#xff0c;一种是买回来已分好区的&#xff0c;还有一种是未经过分区的。如果移动硬盘没有经过分区&#xff0c;那么在将它连接到电脑的USB接口时&#x…

android四大组件之四-BroadCast实现原理分析

前言&#xff1a; 一开始的目标是解决各种各样的ANR问题的&#xff0c;但是我们知道&#xff0c;ANR总体上分有四种类型&#xff0c;这四种ANR类型有三种是和四大组件相对应的&#xff0c;所以&#xff0c;如果想了解ANR发生的根因&#xff0c;对安卓四大组件的实现原理必须要…