MCTS代码

news2024/11/14 6:11:08

这段代码的背景是玩一个游戏。游戏的参数有NUM_TURNS,在第i回合,你可以从一个整数[-2,2,3,-3]*(NUM_TURNS+1-i)中进行选择。例如,在一个4回合的游戏中,在第1回合,你可以从[-8,8,12,-12]中选择,在第2回合,你也可以从[-6,6,9,-9]中选择。在每一个转弯处,所选择的数字都会累积为一个聚合值。游戏的目标是使累积值尽可能接近0。

定义MCTS 标量。标量越大,利用(exploitation)更大;标量越小,探索( exploration)更大。


 接下来使用了logging模块,用于设置和获取日志记录器(logger)。

查看日志是开发人员日常获取信息、排查异常、发现问题的最好途径,日志记录中通常会标记有异常产生的原因、发生时间、具体错误行数等信息,这极大的节省排查时间,无形中提高了编码效率。

有关日志的更多信息。 

  1. logging.basicConfig(level=logging.WARNING)

    • logging.basicConfig():用于快速设置日志系统的基本配置。
    • level=logging.WARNING:设置日志系统的最低日志级别为WARNING。这意味着,只有WARNINGERRORCRITICAL级别的日志消息会被处理并显示。低于WARNING级别的消息(如DEBUGINFO)将被忽略。
    • 使用basicConfig方法后,所有后续的日志消息(除非明确指定了其他logger)都会使用这些基本设置。
  2. logger = logging.getLogger('MyLogger')

    • logging.getLogger('MyLogger'):这是获取或创建一个名为'MyLogger'的日志记录器的方法。
    • 如果名为'MyLogger'的logger已经存在,该方法将返回该logger的引用。如果不存在,它将创建一个新的logger。
    • 默认情况下,新创建的logger会继承root logger的配置(即basicConfig设置的配置)。但你可以为特定logger设置特定的处理器(handlers)、格式器(formatters)和级别(levels)。

这两个语句之间的关系是:basicConfig为整个日志系统(包括所有logger)设置了基本配置,而getLogger则允许你获取或创建一个特定的logger,并可以为其设置特定的配置(如果需要的话)。

一个简单的例子:

import logging  
  
# 设置基本配置,所有日志级别为WARNING及以上的消息都会被处理  
logging.basicConfig(level=logging.WARNING)  
  
# 获取一个名为'MyLogger'的logger  
logger = logging.getLogger('MyLogger')  
  
# 由于basicConfig已经设置了级别为WARNING,所以下面的消息将被记录  
logger.warning('This is a warning message.')  
  
# 下面的消息将不会被记录,因为级别低于WARNING  
logger.info('This is an info message.')

建立类State:

  1. 类变量

    • NUM_TURNS:表示游戏或决策过程中的回合数,这里默认是10。
    • GOAL:表示目标值,状态的值达到这个值时会获得最大的奖励,这里默认是0。
    • MOVES:表示每次状态变化时可能的移动值,这里默认是[2,-2,3,-3]
    • MAX_VALUE:根据公式计算得出,用于后续的归一化奖励。
    • num_movesMOVES列表的长度。
  2. __init__方法

    • 用于初始化一个State对象。
    • value:表示当前状态的值。
    • moves:一个列表,表示到目前为止所采取的所有移动。
    • turn:表示当前的回合数。
  3. next_state方法

    • 基于当前状态生成下一个状态。
    • 随机选择一个移动值,并更新状态的值、移动列表和回合数。
    • 返回新的State对象。
  4. terminal方法

    • 判断当前状态是否是终止状态(即回合数是否为0)。
    • 如果是终止状态,返回True;否则返回False
  5. reward方法

    • 计算并返回当前状态的奖励值,奖励值基于当前状态的值与目标值之间的差异来计算。
  6. __hash__方法

    • State对象提供一个哈希值,以便可以在哈希表中使用。
    • 使用MD5哈希算法对移动列表进行哈希。
  7. __eq__方法

    • 使用哈希值来判断两个State对象是否相等。
  8. __repr__方法

    • 返回一个字符串,包括状态的值和移动列表,表示State对象的状态。

建立类Node,用于表示决策树或搜索树中的节点。在MCTS算法中,每个节点代表一个可能的状态,并且每个节点都存储了关于该状态的信息,如访问次数、累积奖励等。

  1.  __init__方法
    • state:表示该节点对应的状态。
    • parent:表示该节点的父节点,默认为None
    • visits:表示该节点被访问的次数,初始化为1。
    • reward:表示从该节点开始到终止状态所获得的累积奖励,初始化为0.0。
    • children:一个列表,存储该节点的所有子节点。
  2. add_child方法
    • 用于向当前节点添加一个子节点。
    • child_state:子节点对应的状态。
    • 创建一个新的Node对象,并将其作为子节点添加到children列表中。
  3. update方法
    • 用于更新节点的rewardvisits属性。
    • reward:从该节点开始到终止状态所获得的奖励。
    • 该方法将奖励累加到self.reward,并将self.visits加1。
  4. fully_expanded方法
    • 检查当前节点是否已经完全扩展,即是否已经有了所有可能的子节点。
    • num_moves_lambda:一个可选的函数,用于计算给定节点的子节点数量。如果提供了这个函数,它将覆盖self.state.num_moves来计算子节点数量。
    • 如果当前节点的子节点数量等于可能的子节点数量,则返回True,否则返回False
  5. __repr__方法
    • 返回一个字符串,表示节点的简要信息。
    • 包括子节点的数量、访问次数和累积奖励。

建立函数UCTSEARCH ,实现了UCT(Upper Confidence Bound for Trees)搜索算法的函数。UCT是一种常用于蒙特卡洛树搜索(MCTS)的策略,它结合了随机模拟与基于树结构的搜索,以找到最优的决策序列。

参数:

  • budget:这是一个整数,表示模拟的总次数或预算。算法将运行这么多次模拟来寻找最佳策略。
  • root:这是搜索树的根节点,代表初始状态。
  • num_moves_lambda:这是一个可选的lambda函数,用于自定义计算给定节点子节点数量的方式。如果未提供,将使用节点状态中的num_moves属性。

主要步骤:

  1. 循环模拟:算法将进行budget次模拟。在每次模拟中,算法会从根节点开始,选择一系列的动作,直到达到终止状态。

  2. 日志记录:每进行10000次模拟,算法会记录当前的模拟次数,并输出根节点的信息。这有助于观察搜索过程的进展。

  3. 树策略(TREEPOLICY):在每次模拟中,算法使用TREEPOLICY函数从当前节点开始,根据UCT公式选择一个子节点作为下一步。这个步骤基于已有的信息(如节点的访问次数和奖励)来做出决策。

  4. 默认策略(DEFAULTPOLICY):算法使用DEFAULTPOLICY函数从选定的节点(front.state)开始,执行一系列动作直到达到终止状态,并计算累积奖励。

  5. 回溯(BACKUP):在模拟结束后,算法将累积的奖励回溯到搜索树中,更新每个访问过的节点的统计信息(如访问次数和累积奖励)。

  6. 选择最佳子节点(BESTCHILD):在所有模拟结束后,算法使用BESTCHILD函数从根节点开始,基于节点的统计信息选择一个最佳子节点作为下一步。

最终,UCTSEARCH函数返回从根节点开始的最佳子节点序列,这些子节点代表了一系列的动作,构成了在给定预算下找到的最优策略。


建立函数TREEPOLICY ,基于当前节点的信息和UCT公式来决定是扩展一个新节点(即执行一个新的动作),还是选择一个已经存在的子节点进行进一步的搜索。函数在搜索过程中平衡了探索(扩展新节点)和利用(选择已知的最佳子节点)之间的关系,以实现更有效的搜索。

参数:

  • node:当前搜索树中的节点,代表当前的状态。
  • num_moves_lambda:一个可选的函数,用于计算给定节点的子节点数量。如果未提供,将使用节点状态中的 num_moves 属性。

函数流程:

  1. 检查终止状态:首先,函数检查当前节点是否处于终止状态(即游戏是否结束)。如果是终止状态,则不再继续搜索,直接返回当前节点。

  2. 处理空子节点:如果当前节点没有子节点(即还没有进行过任何扩展),则调用 EXPAND 函数来扩展一个新的子节点,并返回这个新节点。

  3. 随机选择:如果当前节点有子节点,函数会随机决定是否选择一个最佳子节点还是扩展一个新的子节点。这是通过生成一个0到1之间的随机数来实现的。

    • 如果随机数小于0.5,则调用 BESTCHILD 函数选择当前节点中根据UCT公式计算出的最佳子节点,并将该节点设置为新的当前节点。
    • 如果随机数大于或等于0.5,则进入下一步。
  4. 检查是否完全扩展:使用 fully_expanded 方法检查当前节点是否已经完全扩展(即是否已经有了所有可能的子节点)。

    • 如果当前节点没有完全扩展,则调用 EXPAND 函数来扩展一个新的子节点,并返回这个新节点。
    • 如果当前节点已经完全扩展,则调用 BESTCHILD 函数选择当前节点中根据UCT公式计算出的最佳子节点,并将该节点设置为新的当前节点。
  5. 返回节点:函数最终返回选择的节点,这个节点可能是新扩展的节点,也可能是根据UCT公式选择的最佳子节点。


建立函数EXPAND,用于扩展当前节点的一个新的子节点,即在当前状态下尝试一个新的动作或决策。

参数:

  • node:当前搜索树中的节点,代表当前的状态。

函数流程:

  1. 获取已尝试的子节点状态:首先,函数创建一个列表 tried_children,其中包含当前节点所有已存在子节点的状态。这是为了确保不会重复扩展已经尝试过的相同状态。

  2. 生成新的状态:调用当前节点状态对象的 next_state() 方法来生成一个新的状态 new_state。这个方法通常代表在当前状态下可以采取的下一个动作或决策。

  3. 检查状态是否已尝试:函数使用一个 while 循环来检查新生成的状态 new_state 是否已经作为子节点尝试过。这是通过检查 new_state 是否存在于 tried_children 列表中,并且该状态不是终止状态来实现的。如果 new_state 已经尝试过或是一个终止状态,则继续生成新的状态。

  4. 添加新子节点:一旦找到一个新的、未尝试过的状态 new_state,函数使用 add_child 方法将其作为新的子节点添加到当前节点中。

  5. 返回新子节点:最后,函数返回新添加的子节点,这样调用者可以进一步处理或从这个新状态开始进行模拟。


建立函数BESTCHILD,用于从当前节点的子节点中选择一个最佳子节点来继续搜索。这个函数同样基于UCT公式来平衡探索和利用的权衡。

参数:

  • node:当前搜索树中的节点,代表当前的状态。
  • scalar:用于调整探索和利用之间的权衡。较大的 scalar 值将增加探索的倾向,而较小的值将更注重利用已知信息。

函数流程:

  1. 初始化最佳分数和最佳子节点列表:函数首先初始化 bestscore 为0.0,用于存储当前找到的最佳分数。同时,bestchildren 列表用于存储具有相同最佳分数的子节点。

  2. 遍历子节点:对于当前节点的每一个子节点 c,函数计算其UCT分数。

  3. 计算利用部分:exploit = c.reward / c.visits 是利用部分,它表示子节点 c 的平均奖励与其被访问次数之间的比率。这反映了子节点 c 的已知价值。

  4. 计算探索部分:explore = math.sqrt(2.0 * math.log(node.visits) / float(c.visits)) 是探索部分,它基于子节点 c 的访问次数和父节点 node 的总访问次数来计算。这反映了选择较少访问的子节点以进行进一步探索的倾向。

  5. 计算UCT分数:score = exploit + scalar * explore 是子节点 c 的UCT分数。它结合了利用部分和探索部分,其中 scalar 用于调整两者之间的平衡。

  6. 更新最佳分数和最佳子节点:如果子节点 c 的UCT分数大于当前的 bestscore,则更新 bestscore 为该分数,并重置 bestchildren 列表只包含当前子节点 c。如果分数等于 bestscore,则将子节点 c 添加到 bestchildren 列表中。

  7. 处理无最佳子节点的情况:如果遍历完所有子节点后 bestchildren 列表为空(这通常是一个错误情况,表明没有子节点被访问过),则记录一个警告日志。

  8. 返回最佳子节点:最后,函数从 bestchildren 列表中随机选择一个子节点作为最佳子节点返回。这允许在多个具有相同最佳分数的子节点之间进行随机选择。


 建立函数DEFAULTPOLICY,用于在没有其他特定策略可用时选择一个默认的动作。

参数:

  • state:当前的状态对象,代表游戏或决策过程的当前状态。

函数流程:

  1. 检查终止状态:函数首先检查当前状态 state 是否是终止状态(即游戏是否结束)。如果是终止状态,则不再继续执行,因为终止状态没有后续的动作或决策。

  2. 选择下一个状态:如果当前状态不是终止状态,函数使用 next_state() 方法来选择或生成下一个状态。这通常意味着在当前状态下执行一个默认的动作或决策。

  3. 迭代过程:函数将当前状态更新为下一个状态,并重复上述过程,直到达到一个终止状态。

  4. 返回奖励:当达到终止状态时,函数调用 reward() 方法来获取该状态的奖励值,并将其作为结果返回。


建立函数BACKUP,用于在模拟结束后更新从根节点到叶节点的路径上所有节点的统计信息。具体来说,它会遍历这条路径,增加每个节点的访问次数(visits)和累计奖励(reward)。

参数:

  • node:需要进行回溯的起始节点,通常是一个叶节点。
  • reward:模拟结束时获得的奖励值。

函数流程:

  1. 检查起始节点:函数首先检查起始节点 node 是否为 None。如果是 None,则不执行任何操作并直接返回。

  2. 遍历路径并更新统计信息:

    • 如果起始节点不是 None,函数进入一个 while 循环,该循环将继续执行,直到遍历到根节点或遇到 None 节点为止。
    • 在循环内部,函数首先增加当前节点 node 的访问次数(visits)和累计奖励(reward)。这通过 node.visits += 1 和 node.reward += reward 实现。
    • 然后,函数将当前节点更新为其父节点 node.parent,以便在下一次循环迭代中处理父节点。
  3. 返回:函数完成后返回 None


主程序入口:

  • if __name__=="__main__"::这是一个常见的Python模式,用于检查脚本是否作为主程序运行,而不是被导入为模块。
  • argparse.ArgumentParser():创建一个命令行参数解析器。
  • parser.add_argument('--num_sims', ...):添加一个命令行参数 --num_sims,它是一个必需的整数参数,用于指定模拟的次数。
  • parser.add_argument('--levels', ...):添加一个命令行参数 --levels,它也是一个必需的整数参数,用于指定搜索的层数。它的值必须在 0 到 State.NUM_TURNS 之间(包括两端)。
  • args = parser.parse_args():解析命令行参数,并将结果存储在 args 对象中。
  • current_node = Node(State()):创建一个根节点,其状态由 State() 确定。
  • for l in range(args.levels)::对于每个指定的搜索层数 l,执行以下操作:
    • current_node = UCTSEARCH(args.num_sims/(l+1), current_node):对当前节点执行UCT搜索,模拟次数随着层数的增加而减少(这是一个常见的策略,因为随着树深度的增加,每个节点的模拟次数通常会减少)。
    • 打印当前层数、当前节点的子节点数量以及每个子节点的信息。
    • print("Best Child: %s" % current_node.state):打印当前节点下具有最佳UCT分数的子节点的状态。
    • print("--------------------------------"):打印一个分隔符,以便于阅读输出。

 


#!/usr/bin/env python
import random
import math
import hashlib
import logging
import argparse


"""
A quick Monte Carlo Tree Search implementation.  For more details on MCTS see See http://pubs.doc.ic.ac.uk/survey-mcts-methods/survey-mcts-methods.pdf

The State is a game where you have NUM_TURNS and at turn i you can make
a choice from an integeter [-2,2,3,-3]*(NUM_TURNS+1-i).  So for example in a game of 4 turns, on turn for turn 1 you can can choose from [-8,8,12,-12], and on turn 2 you can choose from [-6,6,9,-9].  At each turn the choosen number is accumulated into a aggregation value.  The goal of the game is for the accumulated value to be as close to 0 as possible.

The game is not very interesting but it allows one to study MCTS which is.  Some features 
of the example by design are that moves do not commute and early mistakes are more costly.  

In particular there are two models of best child that one can use 
"""

#MCTS scalar.  Larger scalar will increase exploitation, smaller will increase exploration. 
SCALAR=1/(2*math.sqrt(2.0))

logging.basicConfig(level=logging.WARNING)
logger = logging.getLogger('MyLogger')


class State():
	NUM_TURNS = 10	
	GOAL = 0
	MOVES=[2,-2,3,-3]
	MAX_VALUE= (5.0*(NUM_TURNS-1)*NUM_TURNS)/2
	num_moves=len(MOVES)
	def __init__(self, value=0, moves=[], turn=NUM_TURNS):
		self.value=value
		self.turn=turn
		self.moves=moves
	def next_state(self):
		nextmove=random.choice([x*self.turn for x  in self.MOVES])
		next=State(self.value+nextmove, self.moves+[nextmove],self.turn-1)
		return next
	def terminal(self):
		if self.turn == 0:
			return True
		return False
	def reward(self):
		r = 1.0-(abs(self.value-self.GOAL)/self.MAX_VALUE)
		return r
	def __hash__(self):
		return int(hashlib.md5(str(self.moves).encode('utf-8')).hexdigest(),16)
	def __eq__(self,other):
		if hash(self)==hash(other):
			return True
		return False
	def __repr__(self):
		s="Value: %d; Moves: %s"%(self.value,self.moves)
		return s
	

class Node():
	def __init__(self, state, parent=None):
		self.visits=1
		self.reward=0.0	
		self.state=state
		self.children=[]
		self.parent=parent	
	def add_child(self,child_state):
		child=Node(child_state,self)
		self.children.append(child)
	def update(self,reward):
		self.reward+=reward
		self.visits+=1
	def fully_expanded(self, num_moves_lambda):
		num_moves = self.state.num_moves
		if num_moves_lambda != None:
		  num_moves = num_moves_lambda(self)
		if len(self.children)==num_moves:
			return True
		return False
	def __repr__(self):
		s="Node; children: %d; visits: %d; reward: %f"%(len(self.children),self.visits,self.reward)
		return s

def UCTSEARCH(budget,root,num_moves_lambda = None):
	for iter in range(int(budget)):
		if iter%10000==9999:
			logger.info("simulation: %d"%iter)
			logger.info(root)
		front=TREEPOLICY(root, num_moves_lambda)
		reward=DEFAULTPOLICY(front.state)
		BACKUP(front,reward)
	return BESTCHILD(root,0)

def TREEPOLICY(node, num_moves_lambda):
	#a hack to force 'exploitation' in a game where there are many options, and you may never/not want to fully expand first
	while node.state.terminal()==False:
		if len(node.children)==0:
			return EXPAND(node)
		elif random.uniform(0,1)<.5:
			node=BESTCHILD(node,SCALAR)
		else:
			if node.fully_expanded(num_moves_lambda)==False:	
				return EXPAND(node)
			else:
				node=BESTCHILD(node,SCALAR)
	return node

def EXPAND(node):
	tried_children=[c.state for c in node.children]
	new_state=node.state.next_state()
	while new_state in tried_children and new_state.terminal()==False:
	    new_state=node.state.next_state()
	node.add_child(new_state)
	return node.children[-1]

#current this uses the most vanilla MCTS formula it is worth experimenting with THRESHOLD ASCENT (TAGS)
def BESTCHILD(node,scalar):
	bestscore=0.0
	bestchildren=[]
	for c in node.children:
		exploit=c.reward/c.visits
		explore=math.sqrt(2.0*math.log(node.visits)/float(c.visits))	
		score=exploit+scalar*explore
		if score==bestscore:
			bestchildren.append(c)
		if score>bestscore:
			bestchildren=[c]
			bestscore=score
	if len(bestchildren)==0:
		logger.warn("OOPS: no best child found, probably fatal")
	return random.choice(bestchildren)
def DEFAULTPOLICY(state):
	while state.terminal()==False:
		state=state.next_state()
	return state.reward()


def BACKUP(node,reward):
	while node!=None:
		node.visits+=1
		node.reward+=reward
		node=node.parent
	return

if __name__=="__main__":
	parser = argparse.ArgumentParser(description='MCTS research code')
	parser.add_argument('--num_sims', action="store", required=True, type=int)
	parser.add_argument('--levels', action="store", required=True, type=int, choices=range(State.NUM_TURNS+1))
	args=parser.parse_args()

	current_node=Node(State())
	for l in range(args.levels):
		current_node=UCTSEARCH(args.num_sims/(l+1),current_node)
		print("level %d"%l)
		print("Num Children: %d"%len(current_node.children))
		for i,c in enumerate(current_node.children):
			print(i,c)
		print("Best Child: %s"%current_node.state)
		
		print("--------------------------------")	
			
	

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

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

相关文章

蓝桥杯-单片机组基础5——外部中断与LED的控制(附小蜜蜂课程代码)

蓝桥杯单片机组备赛指南请查看这篇文章&#xff1a;戳此跳转蓝桥杯备赛指南文章 本文章针对蓝桥杯-单片机组比赛开发板所写&#xff0c;代码可直接在比赛开发板上使用。 型号&#xff1a;国信天长4T开发板&#xff08;绿板&#xff09;&#xff0c;芯片&#xff1a;IAP15F2K6…

C语言:字符函数 字符串函数 内存函数

C语言&#xff1a;字符函数 & 字符串函数 & 内存函数 字符函数字符分类函数字符转换函数tolowertoupper 字符串函数strlenstrcpystrcatstrcmpstrstrstrtok 内存函数memcpymemmovememsetmemcmp 字符函数 顾名思义&#xff0c;字符函数就是作用于字符的函数&#xff0c;…

Linux 常用的文本处理工具

目录 cat 连接 more/less 分页 tail 实时 cat 连接 将一个或多个文件的内容连接并显示在终端上&#xff0c;创建新文件或将内容追加到已有文件。 不会分屏显示文件内容&#xff0c;适用于较小的文件。 cat 文件1.txt 文件2.txt # 连接并显示文件1.txt和文件2.txt的内容 …

qsort函数的模拟实现(冒泡排序模拟)

冒泡排序&#xff1a; 从第一个元素开始&#xff0c;依次比较相邻的两个元素&#xff0c;如果顺序不对就交换它们。 经过一轮遍历后&#xff0c;最大&#xff08;或最小&#xff09;的元素会排在最后。 重复进行上述步骤&#xff0c;直到没有任何元素需要交换&#xff0c;即…

C++/数据结构:AVL树

目录 一、AVL树的概念 二、AVL树的实现 2.1节点定义 2.2节点插入 三、AVL树的旋转 3.1新节点插入较高左子树的左侧&#xff1a;右单旋 3.2新节点插入较高右子树的右侧&#xff1a;左单旋 3.3新节点插入较高左子树的右侧---左右&#xff1a;先左单旋再右单旋 3.4新节点插…

3. kubectl 命令行交互

在维护 Kubernetes 系统集群时&#xff0c;kubectl 应该是最常用的工具之一。从 Kubernetes 架构设计的角度看&#xff0c;kubectl 工具是 Kubernetes API Server 的客户端。它的主要工作是向 Kubernetes API Server 发起 HTTP 请求。Kubernetes 是一个完全以资源为中心的系统&…

前端Vue3项目如何打包成Docker镜像运行

将前端Vue3项目打包成Docker镜像并运行包括几个主要步骤&#xff1a;项目打包、编写Dockerfile、构建镜像和运行容器。下面是一个基本的流程&#xff1a; 1. 项目打包 首先&#xff0c;确保你的Vue3项目可以正常运行和打包。在项目根目录下执行以下命令来打包你的Vue3项目&am…

Redis 之七:穿透、击穿、雪崩

&#xff08;本内容部分来自知乎网等网络&#xff09; Redis 缓存的使用&#xff0c;极大的提升了应用程序的性能和效率&#xff0c;特别是数据查询方面。但同时&#xff0c;它也带来了一些问题。其中&#xff0c;最要害的问题&#xff0c;就是数据的一致性问题&#xff0c;从严…

大模型(LLM)的量化技术Quantization原理学习

在自然语言处理领域&#xff0c;大型语言模型&#xff08;LLM&#xff09;在自然语言处理领域的应用越来越广泛。然而&#xff0c;随着模型规模的增大&#xff0c;计算和存储资源的需求也急剧增加。为了降低计算和存储开销&#xff0c;同时保持模型的性能&#xff0c;LLM大模型…

MBR20100FCT-ASEMI适配开关电源MBR20100FCT

编辑&#xff1a;ll MBR20100FCT-ASEMI适配开关电源MBR20100FCT 型号&#xff1a;MBR20100FCT 品牌&#xff1a;ASEMI 封装&#xff1a;ITO-220AB 最大平均正向电流&#xff08;IF&#xff09;&#xff1a;20A 最大循环峰值反向电压&#xff08;VRRM&#xff09;&#xf…

DangZero:通过直接页表访问的高效UAF检测(DangZero实现IMPLEMENTATION翻译)

We implement DangZero as a shared library that overlays the de- fault memory allocator via LD_PRELOAD. Additionally, DangZero requires a backend to be available for direct page table access, which we describe in detail in the following section. 我们将DangZ…

mac苹果电脑c盘满了如何清理内存?2024最新操作教程分享

苹果电脑用户经常会遇到麻烦:内置存储器(即C盘)空间不断缩小&#xff0c;电脑运行缓慢。在这种情况下&#xff0c;苹果电脑c盘满了怎么清理&#xff1f;如何有效清理和优化存储空间&#xff0c;提高计算机性能&#xff1f;成了一个重要的问题。今天&#xff0c;我想给大家详细介…

Neo4j aura 官方网站快速入门新手教精读-从官方教程学习知识图谱

Neo4j 官方网站快速入门新手教精读 本文旨在为Neo4j新手提供一份全面的入门指南。除了基础的文本解释&#xff0c;我在里面还插入了每一步骤的详细截图或者自己画的图&#xff0c;从官方了解知识肯定比自己乱看要权威一些&#xff0c;有看不懂的不要纠结了解大概意思即可&#…

springboot支持的常用日志框架介绍

日志系统是计算机系统中用于记录和跟踪事件、错误和信息的软件组件。在软件开发和维护过程中&#xff0c;日志系统起着至关重要的作用。它可以帮助开发人员了解软件的运行情况&#xff0c;快速定位和解决问题。本文将从以下几个方面介绍日志系统&#xff1a;日志系统概述、Spri…

C/C++内存管理及内存泄漏详解

目录 C/C内存分布 C语言中动态内存管理方式&#xff1a;malloc/calloc/realloc/free C内存管理方式 new/delete操作内置类型 new和delete操作自定义类型 operator new与operator delete函数 new和delete的实现原理 内置类型 自定义类型 内存泄漏 概念 内存泄漏分类 ⭐…

Vins-Moon配准运行

Vins-Moon运行 源码地址电脑配置环境配置编译适配Kitti数据集运行结果Euroc数据集kitti数据集 evo评估&#xff08;KITTI数据&#xff09;输出轨迹(tum格式)结果 源码地址 源码链接&#xff1a;https://github.com/HKUST-Aerial-Robotics/VINS-Mono.git 电脑配置 Ubuntu 18.…

机器学习:模型评估和模型保存

一、模型评估 from sklearn.metrics import accuracy_score, confusion_matrix, classification_report# 使用测试集进行预测 y_pred model.predict(X_test)# 计算准确率 accuracy accuracy_score(y_test, y_pred) print(f"Accuracy: {accuracy*100:.2f}%")# 打印…

MySQL 表的基本操作,结合项目的表自动初始化来讲

有了数据库以后&#xff0c;我们就可以在数据库中对表进行增删改查了&#xff0c;这也就意味着&#xff0c;一名真正的 CRUD Boy 即将到来&#xff08;&#x1f601;&#xff09;。 查表 查看当前数据库中所有的表&#xff0c;使用 show tables; 命令 由于当前数据库中还没有…

CrossOver2024电脑虚拟机软件详细介绍概述

CrossOver是由CodeWeavers开发的一款系统兼容软件&#xff0c;它能够在Mac和Linux操作系统上直接运行Windows应用程序&#xff0c;而无需创建或启动完整的Windows虚拟机。CrossOver通过模拟Windows应用程序所需的运行环境&#xff0c;实现了跨平台的无缝集成和高效运行。 Cross…

界面控件Telerik UI for ASP. NET Core教程 - 如何为网格添加上下文菜单?

Telerik UI for ASP.NET Core是用于跨平台响应式Web和云开发的最完整的UI工具集&#xff0c;拥有超过60个由Kendo UI支持的ASP.NET核心组件。它的响应式和自适应的HTML5网格&#xff0c;提供从过滤、排序数据到分页和分层数据分组等100多项高级功能。 上下文菜单允许开发者为应…