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.mzph.cn/news/808024.shtml

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

相关文章

微信公众号第三方平台-公众号扫码授权接入代运营

文章目录 接入目的效果展示技术积累如何成为服务商如何搭建第三方后端服务传统模式V云服务模式如何完成商家授权授权逻辑介绍 环境准备注册开发者平台-个人类型 传统模式后端代码接收公众号个人三方平台的票据根据票据获取三方平台访问令牌根据访问令牌获取预授权码通过预授权码…

OJ 【难度1】【Python】完美字符串 扫雷 A-B数对 赛前准备 【C】精密计时

完美字符串 题目描述 你可能见过下面这一句英文&#xff1a; "The quick brown fox jumps over the lazy dog." 短短的一句话就包含了所有 2626 个英文字母&#xff01;因此这句话广泛地用于字体效果的展示。更短的还有&#xff1a; "The five boxing wizards…

网络——初识网络

在现如今&#xff0c;网络已经成了一种基础设施&#xff0c;大到国家&#xff0c;小到个人&#xff0c;网络已经充斥在我们每个人的身 边&#xff0c;如果一个人突然失去了网络&#xff0c;那么它的生活或多或少会出现一些不方便的地方&#xff0c;网络现在已 经伴随着我们的吃…

Solana主网使用自定义的RPC进行转账

1、引言 如果用 browser 连接主网的 RPC server 会收到 error code 403 message 為 Access forbidden, contact your app developer or supportrpcpool.com. 错误&#xff0c;因为主网的 RPC server 会检查 HTTP Header 如果判断出來是 browser 就会报告 403 錯誤。 要解決这…

N 皇后 - 蓝桥杯?-Lua 中文代码解题第6题

n 皇后问题 研究的是如何将 n 个皇后放置在 n n 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 给你一个整数 n &#xff0c;返回 n 皇后问题 不同的解决方案的数量。 示例 1&#xff1a; 输入&#xff1a;n 4 输出&#xff1a;2 解释&#xff1a;如上图所示&…

吴恩达2022机器学习专项课程(一) 5.2 向量化(1) 5.3 向量化(2)

问题预览/关键词 什么是向量化&#xff1f;向量化的好处是&#xff1f;如何向量化多元线性回归函数的参数&#xff1f;如何在Python中向量化参数&#xff1f;计算机底层是如何计算向量化的&#xff1f;向量化示例 笔记 1.向量化 一种在数学和计算中广泛使用的概念&#xff…

[Mac]安装App后“XX已损坏,无法打开“

问题&#xff1a; “xx.app”已损坏&#xff0c;无法打开。你应该将它移到废纸篓。 解决&#xff1a; 终端输入sudo xattr -r -d com.apple.quarantine 后将Applications中对应的问题app拖入生成路径&#xff0c;然后执行。 $ sudo xattr -r -d com.apple.quarantine /Appli…

备战蓝桥杯(日益更新)(刷题)

备战蓝桥杯&#xff08;日益更新&#xff09;&#xff08;刷题&#xff09; 文章目录 备战蓝桥杯&#xff08;日益更新&#xff09;&#xff08;刷题&#xff09;前言&#xff1a;一、二分&#xff1a;1. acwing503 借教室&#xff1a;&#xff08;二分 差分&#xff09;2. ac…

Socks5代理IP如何使用?详细教程解析

当我们在互联网上浏览网页、下载文件或者进行在线活动时&#xff0c;隐私和安全问题常常被提及。在这样的环境下&#xff0c;一个有效的解决方案是使用Sock5IP。本教程将向您介绍Sock5IP的使用方法&#xff0c;帮助您保护个人隐私并提升网络安全。 一、什么是Sock5IP&#xff1…

Mybatis-Plus使用入门

Mybatis-Plus 一、Mybatis-plus的简介 官方文档的地址&#xff1a; MyBatis-Plus &#xff08;一&#xff09;什么是Mybatis-Plus Mybatis-Plus是一个Mybatis&#xff08;opens new window&#xff09;的增强工具&#xff0c;在Mybatis的基础上只做增强不做改变&#xff0c…

ChatGPT在日常生活与工作中的应用,以及Hulu AI 的探索之旅

ChatGPT在日常生活与工作中的应用&#xff0c;以及Hulu AI 的探索之旅 &#x1f4ac;ChatGPT 的多面应用&#x1f4ac;Hulu AI&#xff1a;一个AI工具聚合平台的探索平台优势为何选择Hulu AI&#xff1f;珍稀优惠 &#x1f4ac;结束语 在数字化快速发展的当下&#xff0c;人工智…

冯喜运:4.11外汇黄金原油晚间行情分析及独家作家操作建议

【 黄金消息面分析】&#xff1a;周四(4月11日)亚市早盘&#xff0c;现货黄金窄幅震荡&#xff0c;周三金价从纪录高位下滑&#xff0c;盘中一度失守2320关口至2319.一线&#xff0c;收报2333附近&#xff0c;因此前强于预期的通胀数据削弱了美国提前降息的预期&#xff0c;美元…

Python初级第二次作业

一、 def reverse(num):anumt0b0cnumwhile a//10>0:if a%10>0:t1aa//10print(t)for i in range(t,-1,-1):if c%100:b0else:b(c%10)*(10**i)c//10print(b) if bnum:return Trueelse:return Falsedef isPalind(num):kreverse(num)if kTrue:print(f"{num}是回文&…

uniapp 轮播列表一排展示3个,左右滑动,滑动到中间放大

一、效果展示 二、代码实现 1.html代码&#xff1a; <!-- 轮播 --><view class"heade"><swiper class"swiper" display-multiple-items3 circulartrue previous-margin1rpx next-margin1rpxcurrent0 change"swiperChange">&l…

书生·浦语2.0(InternLM2)大模型实战--Day02 茴香豆 | 搭建RAG智能助理

视频地址&#xff1a;https://www.bilibili.com/video/BV1QA4m1F7t4/文档地址&#xff1a;https://github.com/InternLM/Tutorial/blob/camp2/huixiangdou/readme.md作业地址&#xff1a;https://github.com/InternLM/Tutorial/blob/camp2/huixiangdou/homework.md RAG 概述 R…

汇舟问卷:海外问卷怎么做?

最近美元升值了&#xff0c;但是想在国内赚取美金的途径很少&#xff0c;大多数人接触不到赚取美金的机会。目前汇舟问卷做的国外问卷调查就是一个赚取美金的机会。 操作步骤也比较简单&#xff0c;只需要先搭建好国外的ip环境&#xff0c;然后创建对应国家的人设&#xff0c;…

C语言 知识点 + 笔记(2w6千字 持续更新...)

前言 本篇以笔记为主的C语言详解,全篇一共十章内容,2万6千多字,会持续更新基础内容,争取做到更详细。多一句没有,少一句不行! 形而上学者谓之道,形而下学者谓之器 第 1 章 C语言的流程 (1) C程序经历的六个阶段 编辑(Edit)预处理(Preprocess)编译(Compile)汇编(Assemb…

Prometheus实现自定义脚本监控

#Prometheus# 监控路漫漫其修远兮&#xff0c;吾将上下而求索&#xff01; 一、前言 在监控工作过程中经常会收到大量的定制化的监控需求&#xff0c;Prometheus就提供了一个很强大的组件 --> Pushgateway&#xff0c;他不仅是网关的用途接收exporter的数据&#xff0c;还…

Docker 安装MySql并操作日志

一、在Linux系统里新建这几个文件夹 1.1 在conf.d文件夹下新建一个my.cnf文件 1.2 用vscode 打开&#xff08;防止乱码&#xff09;&#xff0c;复制以下内容 [mysqld] log_timestampsSYSTEM default-time-zone8:00server-id1log-binmysql-binbinlog-do-db mall # 要监听的库…