PHP7垃圾回收算法

前提

本文为了梳理PHP GC工作流程,所以从引用计数、部分标记清除算法做引子,然后介绍PHP GC工作流程,最后介绍性能更高的GC算法

引用计数

概述

引用计数算法中引入了一个概念计数器。计数器代表对象被引用的次数

基本原理

为了记录一个对象有没有被其他对象引用,我们可以在每个对象的头上引用一个叫“计数器”的东西,用来记录有多少其他对象引用了它

这个计数器的值的变化都是由mutator引起的。例如:

public class MyObject {public Object ref = null;public static void main(String[] args) {MyObject objA = new MyObject();MyObject objB = new MyObject();objA.ref = objB;}
}

在这里插入图片描述
上图为例,而objA 做为一个局部变量引用了它,所以它的引用计数就是1,objB这个局部变量引用了它,然后objA又引用了它一次,所以它的引用计数就是2

mutator在运行中还会不断地修改对象之间的引用关系,我们知道,这种引用关系的变化都是发生在赋值的时候。例如,接上文的例子,我们再执行这样一行代码

objA = null;

那么从objA到objB的引用就消失了,也就是上图中,那个从A的ref指向B的箭头就消失了

运行原理

update_ptr(ptr, obj){inc_ref_cnt(obj)  // 计数器+dec_ref_cnt(*ptr) // 计数器-*ptr = obj       // 重新指向 obj
}inc_ref_cnt(obj){obj.ref_cnt++
}dec_ref_cnt(*ptr){obj.ref_cnt--if (obj.ref_cnt == 0)for(child : children(obj)) // 当自己被清除时,自己所引用的子节点的计数器必须减一。进行递归操作。dec_ref_cnt(*child)// 然后通过reclaim()函数,将obj连接到空闲链表上面reclaim(obj)
}

把 obj 赋值给 ptr 这个指针之前,我们可以先改变一下这两个对象的引用计数

在一次赋值中,要先把老的对象的引用计数减一,把新的对象的引用计数加一

如果某个对象的引用计数为0,就把这个对象回收掉,然后把这个对象所引用的所有对象的引用计数减1。

为什么要先inc_ref_cnt(obj)然后再dec_ref_cnt(*ptr)呢?

  • 如果按照先dec_ref_cnt()后inc_ref_cnt()函数的顺序调用ptr和 obj又是同一对象的话执行dec_ref_cnt(ptr)时ptr的计数器的值就有可能变为0而被回收

  • 再想执行inc_ref_cnt(obj)时obj早就被回收了,可能会引发重大的BUG

优点

可即回收的垃圾: 每个对象在被引用次数为0的时候,可以立即知道

没有暂停时间: 对象的回收根本不需要另外的GC线程专门去做,业务线程自己就搞定了,不需要STW

缺点

计数器的增减处理频繁

循环引用无法回收: objA引用了objB,objB也引用了objA, 两个对象的引用计数就都是1。这种情况下,这两个对象是不能被回收的

在这里插入图片描述

部分标记清除算法

概述

为了解决循环依赖的问题

部分标记清除算法通过把对象涂成4种不同的颜色进行管理

四色标记流程

前提

黑(BLACK): 不是垃圾对象(对象产生的初始颜色)

白(WHITE): 垃圾对象

灰(GRAY): 搜索完毕的对象

阴影(HATCH): 可能是循环垃圾

在这里插入图片描述
循环引用的对象群是ABC和DE,其中A和D由根引用,此外C和E引用F

所有对象的颜色现在还是初始的黑色

dec_ref_cnt() 函数

dec_ref_cnt(obj){obj.ref_cnt--if(obj.ref_cnt == 0 )delete(obj)else if(obj.color != HATCH)obj.color = HATCHenqueue(obj, $hatch_queue)
}

算法在对obj的计数器进行减量操作后,检查obj的颜色。当obj的颜色不是阴影的时候,算法会将其涂上阴影并追加到队列中

在这里插入图片描述
由根到A的引用被删除了,指向A的指针被追加到队列($hatch_queue)之中。A被涂上了阴影

new_obj()函数

new_obj(size){obj = pickup_chunk(size)                // 创建对象if(obj != NULL)obj.color = BLACKobj.ref_cnt = 1return objelse if(is_empty($hatch_queue) == FALSE) // 如果$hatch_queue不为空scan_hatch_queue()                  // 标记清除回收垃圾return new_obj(size)                // 重新分配elseallocation_fail()
}

当分配无法顺利进行的时候,程序会调查队列是否为空

当队列不为空时,程序会通过scan_hatch_ queue() 函数搜索队列,分配分块

scan_hatch_queue() 函数执行完毕后,程序会递归地 调用 new_obj() 函数再次尝试分配。 如果队列为空,则分配将会失败

scan_hatch_queue(){// 对象出队obj = dequeue($hatch_queue)// 判断对象是不是阴影if(obj.color == HATCH)paint_gray(obj)     // 查找对象进行计数器的减量操作scan_gray(obj)      // 查找灰色对象,按条件变换白色对象collect_white(obj)   // 回收白色对象else if(is_empty($hatch_queue) == FALSE)scan_hatch_queue()
}

paint_gray()函数

// 查找对象进行计数器的减量操作
paint_gray(obj){if(obj.color == (BLACK|HATCH))obj.color = GRAY // 搜索完毕的颜色for(child :children(obj))(*child).ref_cnt--paint_gray(*child)
}

在这里插入图片描述

scan_gray()函数

// 从第一个灰色的对象开始找,找到后如果计数器为0就将颜色改为白色,计数器大于0就会执行paint_black(). 然后递归子节点
scan_gray(obj){if(obj.color == GRAY)if(obj.ref_cnt > 0 )paint_black(obj)elseobj.color = WHITEfor(child :children(obj))scan_gray(child)
}// 从那些可能被涂成了灰色的有循环引用的对象群中,找出不是垃圾的对象,并将其归回原处
paint_black(obj){obj.color = BLACKfor(child :children(obj))(*child).ref_cnt++if((*child).color != BLACK)paint_black(child)
}

在这里插入图片描述
形成了循环垃圾的对象 A、B、C 被涂成了白色,而有循环引用的非垃圾对象 D、 E、F 被涂成了黑色

collect_white()函数

collect_white(obj){if(obj.color == WHITE)obj.color = BLACKfor(child :children(obj))collect_white(*child)reclaim(obj)
}

在这里插入图片描述

部分标记清除算法的局限性

这个算法不仅付出很大成本搜索对象,还需要查找三次对象,分别是mark_gray()、sacn_gray()、collect_white()

这很大程度的增加了内存管理所花费的时间。还因此对引用计数法最大暂停时间短的优势造成的破坏性的影响

PHP7 GC

对象颜色流转

在这里插入图片描述
目前垃圾回收只针对array、object两种类型

GC算法简述

遍历roots链表, 把当前元素标为灰色(zend_refcounted_h.gc_info置为GC_GREY),然后对当前元素的成员进行深度优先遍历,把成员的refcount减1,并且也标为灰色。(gc_mark_roots())

遍历roots链表中所有灰色元素及其子元素,如果发现其引用计数仍旧大于0,说明这个元素还在其他地方使用,那么将其颜色重新标记会黑色,并将其引用计数加1(在第一步有减1操作)。如果发现其引用计数为0,则将其标记为白色。(gc_scan_roots())

遍历roots链表,将黑色的元素从roots移除。然后对roots中颜色为白色的元素进行深度优先遍历,将其引用计数加1(在第一步有减1操作),同时将颜色为白色的子元素也加入roots链表。最后然后将roots链表移动到待释放的列表to_free中。(gc_collect_roots())

释放to_free列表的元素

zend_refcounted_h 结构体

typedef struct _zend_refcounted_h {uint32_t         refcount;          /* reference counter 32-bit */union {struct {ZEND_ENDIAN_LOHI_3 (zend_uchar    type,     // 当前元素的类型,同zval的u1.v.typezend_uchar    flags,    // 标记数据类型,可以是字符串类型或数组类型等// 后面的两个字节标记当前元素的颜色和垃圾回收池中的位置// 其中高地址的两位用来标记颜色,低地址的14位用于记录位置uint16_t      gc_info)  // keeps GC root number (or 0) and color} v;uint32_t type_info;} u;
} zend_refcounted_h;
  • type: 当前元素的类型,同zval的u1.v.type
  • flags: 标记数据类型,可以是字符串类型或数组类型等
  • gc_info: 后面的两个字节标记当前元素的颜色和垃圾回收池中的位置,其中高地址的两位用来标记颜色,低地址的14位用于记录位置
	 define GC_COLOR  0xc000define GC_BLACK  0x0000(黑色: 不是垃圾对象)define GC_WHITE  0x8000(白色: 垃圾对象)define GC_GREY   0x4000(灰色: 将被标记为白色)define GC_PURPLE 0xc000(紫色: 加入的垃圾收集器)

垃圾收集器结构体 - zend_gc_globals

typedef struct _zend_gc_globals {zend_bool         gc_enabled;   //是否启用gczend_bool         gc_active;    //是否在垃圾检查过程中zend_bool         gc_full;      //缓存区是否已满gc_root_buffer   *buf;              //启动时分配的用于保存可能垃圾的缓存区gc_root_buffer    roots;            //指向buf中最新加入的一个可能垃圾gc_root_buffer   *unused;           //指向buf中没有使用的buffergc_root_buffer   *first_unused;     //指向buf中第一个没有使用的buffergc_root_buffer   *last_unused;      //指向buf尾部gc_root_buffer    to_free;          //待释放的垃圾列表gc_root_buffer   *next_to_free;     //下一待释放的垃圾列表  uint32_t gc_runs;       //统计gc运行次数uint32_t collected;     //统计已回收的垃圾数
} zend_gc_globals;
  • buf: 垃圾缓冲区,PHP7默认10000个节点位置。第0个位置保留
  • roots: 指向缓冲区中最新加入的可能是垃圾的元素
  • unused: 指向缓冲区中没有使用的位置,GC算法没有开始,指向空
  • first_unused: 指向缓冲区中第一个未使用的位置,新的元素插入缓冲区后,指针会向后移动一位
  • last_unused: 指向缓冲区中最后一个位置
  • to_free: 待释放的列表
  • next_to_free: 下一个代释放的列表

gc_possible_root 函数 - 把对象加入缓冲区

当进行unset的时候,会调用对应函数类似于:ZEND_UNSET_VAR_SPEC_CV_UNUSED_HANDLER

同时判断对象为collectable类型,且未加入垃圾回收缓存区。就会调用 gc_possible_root 尝试加入缓冲区

/*** @brief 缓冲区处理*  1. 变量检查,必须是array或object且必须是黑色,说明没有加入过缓冲区*  2. 首先尝试在unused队列中取一个buffer*      1. 如果unused队列不为空,从unused队列中取到一个buffer, unused后移*      2. 如果GC_G(first_unused) != GC_G(last_unused), buffer队列未满,则从first_unused取一个buffer, 同时将first_unused后移*      3. 缓冲区已满的情况。启动垃圾回收 跳转到 zend_gc_collect_cycles 函数。垃圾回收之后,就有空的buffer可以从unused队列取出*  3. 得到了新的buffer,把传入的变量先设置为字符串,然后写入buffer之中,并挂载到全局roots链中* * @param ref 是zend_value相应的gc地址* @return ZEND_API */
ZEND_API void ZEND_FASTCALL gc_possible_root(zend_refcounted *ref)
{
}

zend_gc_collect_cycles函数 - GC回收流程

/*** @brief GC回收流程*  1. GC_G(roots).next != &GC_G(roots), 判断roots链不为空*  2. gc_mark_root函数: 遍历roots链表,对当前节点value的所有成员(如数组元素、成员属性)进行深度优先遍历把成员refcount减1*  3. gc_scan_roots函数: 遍历roots链表中所有灰色元素及其子元素,如果发现其引用计数仍旧大于0,说明这个元素还在其他地方使用*      那么将其颜色重新标记会黑色,并将其引用计数加1。如果发现其引用计数为0,则将其标记为白色*  4. gc_collect_roots 函数: 遍历roots链表,将黑色的元素从roots移除*      对roots中颜色为白色的元素进行深度优先遍历,将其引用计数加1,同时将颜色为白色的子元素也加入roots链表*      最后然后将roots链表移动到待释放的列表to_free中*  5. 把全局to_free列表复制到本地to_free,然后遍历释放,最后把回收使用过的垃圾池buffer,将其放入unused队列* * @return ZEND_API */ZEND_API int zend_gc_collect_cycles(void)
{
}

gc_mark_roots 函数 - 对roots链的紫色对象进行标记

/*** @brief 对roots链的紫色对象进行标记* */
static void gc_mark_roots(void)
{gc_root_buffer *current = GC_G(roots).next;while (current != &GC_G(roots)) {// 对紫色对象进行标记if (GC_REF_GET_COLOR(current->ref) == GC_PURPLE) {gc_mark_grey(current->ref);}current = current->next;}
}

gc_mark_grey 函数 - 标记为灰色

/*** @brief 标记为灰色*  1. 对不是灰色对象,标记为灰色*  2. 对象类型。通过get_gc获取子节点,并递归标记子节点。最后引用计数减1*  3. 数组类型。如果是全局符号表(EG(symbol_table)),则将引用标记为黑色,并返回*              如果不是全局符号表,代码将把引用转换为 zend_array 类型*              然后遍历hashtable,引用计数减1并递归子节点*  4. 引用类型。检查引用对象的 val 是否为引用计数类型(Z_REFCOUNTED)*              如果是,那么它会继续判断对象存储(EG(objects_store).object_buckets)是否为空并且 val 的类型是否为对象类型(IS_OBJECT)*              如果对象存储不为空或者 val 的类型不是对象类型,那么它会将 val 的引用计数值减1* * @param ref */
static void gc_mark_grey(zend_refcounted *ref)
{
}

gc_scan_roots函数 - 对roots链进行扫描

static void gc_scan_roots(void)
{gc_root_buffer *current = GC_G(roots).next;while (current != &GC_G(roots)) {gc_scan(current->ref);current = current->next;}
}

gc_scan 函数 - GC扫描

/*** @brief GC扫描*  1. 对象为灰色*  2. 如果引用计数大于0,就把对象标记为黑色,并跳转到gc_scan_black*  3. 如果引用计数小于0,则标记为白色(垃圾对象)*      1. 对象类型。通过get_gc获取子节点,并递归标记子节点*      2. 数组类型。如果是全局符号表就标记为黑色*                 如果不是全局符号表,代码将把引用转换为 zend_array 类型,然后遍历hashtable并递归子节点*      3. 引用类型。如果对象存储不为空或者 val 的类型不是对象类型,那么就goto tail_call* * @param ref */
static void gc_scan(zend_refcounted *ref)
{
}

gc_scan_black 函数 - 黑色对象扫描

/*** @brief 黑色对象扫描*  1. 标记为黑色(不是垃圾对象)*  2. 对象类型。通过get_gc获取子节点,并递归标记子节点。最后引用计数加1*  3. 数组类型。如果不是全局符号表,代码将把引用转换为 zend_array 类型*              然后遍历hashtable,引用计数加1并递归子节点*  4. 引用类型。如果对象存储不为空或者 val 的类型不是对象类型,那么它会将 val 的引用计数值加1* * @param ref */
static void gc_scan_black(zend_refcounted *ref)
{
}

gc_collect_roots 函数 - 对roots链进行回收

/*** @brief 对roots链进行回收*  1. 把黑色对象从roots链脱链*  2. 对roots链的白色对象(垃圾)进行回收,通过 gc_collect_white 函数*  3. 把roots链的对象交换到to_free列表中* * @param flags * @param additional_buffer * @return int */
static int gc_collect_roots(uint32_t *flags, gc_additional_buffer **additional_buffer)
{
}

gc_collect_white 函数 - 对白色对象进行回收

/*** @brief 对白色对象进行回收*  1. 把白色对象设置为黑色对象*  1. 对象类型。通过get_gc获取子节点,并递归标记子节点。如果为黑色对象,且子节点过多触发 gc_add_garbage 功能*  2. 数组类型。如果不是全局符号表,代码将把引用转换为 zend_array 类型,然后遍历hashtable并递归子节点*              同时如果为黑色对象,且子节点过多触发 gc_add_garbage 功能*  3. 引用类型。如果对象存储不为空或者 val 的类型不是对象类型,那么就goto tail_call* * @param ref * @param flags * @param additional_buffer * @return int */
static int gc_collect_white(zend_refcounted *ref, uint32_t *flags, gc_additional_buffer **additional_buffer)
{
}

gc_add_garbage 函数 - 新增垃圾回收空间

/*** @brief 将不在roots链上的白色元素挂接到roots链上*  1. 将所有白色元素放到roots链上,这当然也包括白色的子元素*  2. 子元素可能有很多,但受限于垃圾缓冲池的大小roots最长只有10000个,不够用怎么办呢?*  3. 这时就需要临时申请额外的存储空间gc_additional_buffer* * @param ref * @param additional_buffer */
static void gc_add_garbage(zend_refcounted *ref, gc_additional_buffer **additional_buffer)
{
}

总结

PHP7 GC 算法 与 部分标记清除算法类似,重点是对象状态流转和缓冲区队列

接下来介绍的是性能更高的引用计数算法

RC Immix

概述

RC Immix 是 合并型引用计数法 + Immix结合.与以往的引用计数法相比,其吞吐量平均提升12%

吞吐量得到改善的原因有两个

  • 合并型引用计数法。因为没有通过写入屏障来执行计数器的增减操作,所以即使对象间的引用关系频繁发生变化,吞吐量也不会下降太多

  • 撤除了空闲链表。通过以线为单位来管理分块,只要在线内移动指针就可以进行分配了

合并型引用计数法

在合并型引用计数法中要将指针发生改动的对象和其所有子对象注册到更改缓冲区中,等到缓冲区满了,就要运行GC(类似于PHP 的unused队列)

等到GC回收时,才能从缓冲区取出对应的变量进行增量/减量

Immix

ImmixGC 构成

ImmixGC 把堆分为一定大小的块(block), 再把每一个块分成一定大小的线(line). 这个算法不是以对象为单位,而是以线为单位回收垃圾

块最合适的大小是32k字节,线最合适的大小是128字节。每个块就有32 * 1024 / 128 = 256个线

各个块由以下4个域

  • line: 线

  • mark_table: 线对应的标记位串

    • FREE(没有对象)
    • MARKED(标记完成)
    • ALLOCATED(有对象)
    • CONSERVATIVE(保守标记)
  • status: 用于表示每个块使用情况的域

    • FREE(所有线为空)
    • RECYCLABLE(一部分线为空)
    • UNAVAILABLE(没有空的线)
  • hole_ctn: 用于表示碎片化严重程度的指标

ImmixGC工作原理

分配时程序首先寻找空的线,然后安排对象。没找到空的线时候就执行GC

GC分为3个步骤执行

  • 选定备用的From 块
    • 通过hole_ctn数来判断,优先选择碎片化最严重的线作为备用From块
    • 判断标准为, “From 块中 ALLOCATED 线和 CONSERVATIVE 线的总数” <= “除From 以外的块中 FREE 线的总数”
  • 搜索阶段
    • 从根开始搜索对象,根据对象分别进行标记处理或复制处理
    • 复制处理指的是将备用 From 块里的对象复制到别的块(To 块/FREE块),并进行压缩
  • 清除阶段
    • 清除阶段中要搜索各个块的 mark_table
    • 如果 mark_table[i] 的值是 ALLOCATED,则设定 mark_table[i] = FREE

合并型引用计数法和Immix融合

RC Immix 中不仅对象有计数器,线也有计数器,这样就可以获悉线内是否存在活动对象

对象的计数器表示的是指向这个对象的引用的数量,而线的计数器表示的是这个线里存在的活动对象的数量

当对象的计数器为 0 时,对线的计数器进行减量操作。当线的计数器为 0 时,我们就可以将线整个回收再利用了

RC Immix 和合并型引用计数法一样,在更改缓冲区满了的时候都会查找更改缓冲区,这时如果发现了新对象,就会把它复制到别的空间(Immix 中的新块)去

同时通过被动碎片整理,对新的对象进行压缩,但是也会导致旧对象碎片化

而积极碎片整理。正好完善无法对旧对象进行压缩、无法回收有循环引用的垃圾的问题。原理就是决定要复制到哪个块,然后把能够通过指针从根查找到的对象全部复制过去

参考资料

  • 《垃圾回收的算法与实现》
  • 《PHP7 底层设计与源码实现》
  • 垃圾回收机制中,引用计数法是如何维护所有对象引用的?
  • php7垃圾回收机制及相关源码解读
  • [Go三关-典藏版]Golang垃圾回收+混合写屏障GC全分析
  • Taking off the gloves with reference counting Immix

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

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

相关文章

探索艺术的新领域——3D线上艺术馆如何改变艺术作品的传播方式

在数字化时代的浪潮下&#xff0c;3D线上艺术馆成为艺术家们展示和传播自己作品的新平台。不仅突破了地域和物理空间的限制&#xff0c;还提供了全新的互动体验。 一、无界限的展示空间&#xff1a;艺术家的新展示平台 3D线上艺术馆通过数字化技术&#xff0c;为艺术家提供了一…

紧急 CCF-C ICPR 2024摘要投稿日期延期至4月10日 速投速成就科研梦

会议之眼 快讯 第27届ICPR&#xff08;The International Conference on Pattern Recognition&#xff09;即国际模式识别会议将于 2024年 12月1日-5日在印度加尔各答的比斯瓦孟加拉会议中心举行&#xff01;ICPR是国际模式识别协会的旗舰会议&#xff0c;也是模式识别、计算机…

面试算法-171-翻转二叉树

题目 给你一棵二叉树的根节点 root &#xff0c;翻转这棵二叉树&#xff0c;并返回其根节点。 示例 1&#xff1a; 输入&#xff1a;root [4,2,7,1,3,6,9] 输出&#xff1a;[4,7,2,9,6,3,1] 解 class Solution {public TreeNode invertTree(TreeNode root) {if (root n…

照片怎么添加时间水印?这篇文章教你水印技巧

照片如何添加时间水印&#xff1f;在数字时代&#xff0c;照片已经成为我们记录生活点滴、分享美好瞬间的重要方式。而给照片添加时间水印&#xff0c;不仅可以保留拍摄时的精确时刻&#xff0c;还能为照片增添一份独特的纪念意义。本文将详细介绍如何为照片添加时间水印&#…

【机器学习】《机器学习算法竞赛实战》第7章用户画像

文章目录 第7章 用户画像7.1 什么是用户画像7.2 标签系统7.2.1 标签分类方式7.2.2 多渠道获取标签7.2.3 标签体系框架 7.3 用户画像数据特征7.3.1 常见的数据形式7.3.2 文本挖掘算法7.3.3 神奇的嵌入表示7.3.4 相似度计算方法 7.4 用户画像的应用7.4.1 用户分析7.4.2 精准营销7…

如何从0到1出海掘金俄罗斯?一文讲透俄罗斯市场、买量、发行、变现最新实用洞察 | TopOn变现干货

中国企业加速出海已经成为一个常态化趋势&#xff0c;出海掘金&#xff0c;从东南亚到北美&#xff0c;欧洲&#xff0c;再到近些年潜力巨大的拉美和中东&#xff0c;中国企业的身影遍布海外市场&#xff0c;出海竞争也随之由蓝海进入红海&#xff0c;那么&#xff0c;全球市场…

【贪玩巴斯】Mac的M芯片(M1/2...)下载homebrew方法(24年最新且已验证可行)

1. 按照目前广为流传的方法&#xff08;M1会出现一些问题&#xff09;&#xff1a; 终端输入&#xff1a; /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)" 使用国内镜像下载。 2. 输入后按照要求步骤执行即可&#xff…

【JavaEE初阶系列】——文件操作 IO 之 文件系统操作

目录 &#x1f4dd;认识文件 &#x1f6a9;树型结构组织 和 目录 &#x1f388;绝对路径和相对路径 &#x1f6a9;文件类型 &#x1f4dd;文件系统操作 &#x1f388;File 概述 &#x1f388;File类的使用 1. 绝对路径 vs 相对路径 2. 路径分隔符 3. 静态成员变量 4…

ELFK (Filebeat+ELK)日志分析系统

一. 相关介绍 Filebeat&#xff1a;轻量级的开源日志文件数据搜集器。通常在需要采集数据的客户端安装 Filebeat&#xff0c;并指定目录与日志格式&#xff0c;Filebeat 就能快速收集数据&#xff0c;并发送给 logstash 进或是直接发给 Elasticsearch 存储&#xff0c;性能上相…

Vue前端框架

1.vue基本使用1 1.vue环境搭建 一般创建vue项目是在cmd命令中用&#xff1a;vue ui 命令&#xff0c;采用ui图形界面的方式直观创建项目。 2.vue基本使用方式&#xff1a;vue组件 3.文本插值 4.属性绑定 5.事件绑定 6.双向绑定 7.条件渲染 2.vue基本使用2 1.axios 安装axios命令…

Linux查看系统配置信息的命令【lscpu】【free】【df】【uname】【lsblk】【top】

目录 1.查看CPU信息【lscpu】 2.查看内存信息【free】 3.查看文件系统信息【df】 4.查看系统信息【uname】 知识扩展&#xff1a;Red Hat Enterprise Linux 和 Debian GNU/Linux 两者的发展介绍 知识扩展&#xff1a;Centos 和 ubuntu的区别 知识扩展&#xff1a;更多 …

vue实现从本地上传头像功能

上传头像&#xff1a; <template><div><el-card class"box-card"><div slot"header" class"clearfix"><span>更换头像</span></div><div><!-- 图片、用来展示用户选择的头像 --><img…

async+await——用法——基础积累

对于asyncawait&#xff0c;我一直都不太会用。。。。 今天记录一下asyncawait的实际用法&#xff1a; 下面是一个实际的使用场景&#xff1a; 上面的代码如下&#xff1a; async fnConfirmCR(){let type this.crType;let crId this.crId;if(typeof crId object){let ne…

《从零开始学架构》读书笔记(一)

目录 软件架构设计产生的历史背景 软件架构设计的目的 系统复杂度来源 追求高性能 一、单机高性能 二、集群的高性能 追求高可用 一、计算高可用 二、存储高可用 追求可扩展性 一、预测变化 二、应对变化 追求安全、低成本、规模 一、安全 二、低成本 三、规模…

1. VirtualBox安装CentOS

安装 VirtualBox 地址:https://www.virtualbox.org/wiki/Downloads 版本: 6.1和7.0+版本都可以 安装: windows上安装需要admin权限,右键菜单选中 “Run as administrator” 安装 CentOS 6.10 地址:https://vault.centos.org/6.10/isos/x86_64/ 版本: 如果不需要GUI,选择…

LeetCode——622设计循环队列

. - 力扣&#xff08;LeetCode&#xff09;. - 备战技术面试&#xff1f;力扣提供海量技术面试资源&#xff0c;帮助你高效提升编程技能,轻松拿下世界 IT 名企 Dream Offer。https://leetcode.cn/problems/design-circular-queue/ 1.题目 设计你的循环队列实现。 循环队列是一…

银行渠道整合平台应用架构

渠道整合平台将 功能微服务化&#xff0c;将服务流程标准化。微服务 化的功能能够进行各种组合使用。而标准化的流程可同时作用于所有渠道&#xff0c;保证体验一致。未来在进行流程变更的时候可有效避免各渠道的重复开发。 • 渠道整合平台避免了各个渠道对于同一个业务的差异…

【HTML】简单制作一个动态3D正方体

目录 前言 开始 HTML部分 JS部分 CSS部分 效果图 总结 前言 无需多言&#xff0c;本文将详细介绍一段代码&#xff0c;具体内容如下&#xff1a; 开始 首先新建文件夹&#xff0c;创建两个文本文档&#xff0c;其中HTML的文件名改为[index.html]&#xff0c;JS的文件名改…

FreeRTOS学习 -- 移植

一、添加FreeRTOS源码 在基础工程中新建一个名为FreeRTOS的文件夹&#xff0c;创建FreeRTOS文件夹以后将FreeRTOS的源码添加到这个文件夹中。 portable 文件夹&#xff0c;只需要保留keil、MemMang 和 RVDS这三个文件夹&#xff0c;其他的都可以删除掉。 移植FreeRTOSConfig…

vue3新手笔记

setup&#xff08;&#xff09;{}函数&#xff0c;是启动页面后&#xff0c;自动执行的一个函数。所有数据&#xff08;常量、变量&#xff09;、函数等等&#xff0c;都要return 出去。 ref函数(可用于基本数据类型&#xff0c;也可以用于复杂数据类型)&#xff1a;让页面上的…