I.MX RT1170双核学习(3):多核管理之MCMGR源码分析详解

本文通过SDK中最简单的hello_world例程来说明一下双核程序如何运行。在CM7和CM4的工程中都有一个MCMGR(Multicore Manager)文件夹,它是用来管理多核之间的操作的,当然也包括我们前面提到的那些寄存器的设置。

在这里插入图片描述

文章目录

  • 1 MCMGR_EarlyInit
    • 1.1 MCMGR_TriggerEvent
  • 2 MCMGR_Init函数
    • 2.1 MCMGR_RegisterEvent函数
    • 2.2 事件回调函数
    • 2.3 mcmgr_late_init_internal
  • 3 MCMGR_StartCore函数
  • 4 MCMGR_GetStartupData函数
  • 5 MU中断接收函数
  • 6 双核状态机交互过程详解
  • 7 总结

我们现在就以CM7核激活CM4核为例,先分析一下CM7核的代码。实际上在CM7的例程中就调用了 MCMGR_EarlyInitMCMGR_InitMCMGR_StartCore三个函数就能启动M4核了。下面就来分析一下这三个函数:

1 MCMGR_EarlyInit

无论是CM7还是CM4,都需要调用这个函数,它是用来初始化底层的多核管理库(MCMGR)的,这个函数应该尽可能在reset_handler附近调用,表示某个核已经启动并准备好执行任务。在这里,这个函数在ResetISR->SystemInit中就调用了。这个函数最终调用的是mcmgr_early_init_internal

mcmgr_status_t mcmgr_early_init_internal(mcmgr_core_t coreNum) 
{MU_Init(MUA);  //实际上就是使能MU的时钟(M7核初始化MUA,M4核这里的参数为MUB)return MCMGR_TriggerEvent(kMCMGR_RemoteCoreUpEvent, 0);
}

1.1 MCMGR_TriggerEvent

MCMGR_TriggerEvent函数实际上调用的是MCMGR_TriggerEventCommon

mcmgr_status_t MCMGR_TriggerEvent(mcmgr_event_type_t type, uint16_t eventData)
{return MCMGR_TriggerEventCommon(type, eventData, false);
}

所以来看一下MCMGR_TriggerEventCommon函数:

/*! @brief Type definition of event types. */
typedef enum _mcmgr_event_type_t
{kMCMGR_RemoteCoreUpEvent = 1,kMCMGR_RemoteCoreDownEvent,kMCMGR_RemoteExceptionEvent,kMCMGR_StartupDataEvent,kMCMGR_FeedStartupDataEvent,kMCMGR_RemoteRPMsgEvent,kMCMGR_RemoteApplicationEvent,kMCMGR_FreeRtosMessageBuffersEvent,kMCMGR_EventTableLength
} mcmgr_event_type_t;static mcmgr_status_t MCMGR_TriggerEventCommon(mcmgr_event_type_t type, uint16_t eventData, bool forcedWrite)
{uint32_t remoteData;remoteData = (((uint32_t)type) << 16) | eventData;return mcmgr_trigger_event_internal(remoteData, forcedWrite);
}

接着看一下mcmgr_trigger_event_internal函数:

/* MCMGR MU channel index - used for passing startupData */
#define MCMGR_MU_CHANNEL 3mcmgr_status_t mcmgr_trigger_event_internal(uint32_t remoteData, bool forcedWrite)
{/* When forcedWrite is false, execute the blocking call, i.e. wait until previouslysent data is processed. Otherwise, run the non-blocking version of the MU send function. */if (false == forcedWrite){MU_SendMsg(MUA, MCMGR_MU_CHANNEL, remoteData);// M7执行这条,M4的第一个参数MUA换为MUB}else{MU_SendMsgNonBlocking(MUA, MCMGR_MU_CHANNEL, remoteData);// M7执行这条,M4的第一个参数MUA换为MUB}return kStatus_MCMGR_Success;
}

这里的forcedWrite参数为false的时候,执行阻塞写函数,等待上一次发送的数据处理完了才发送;forcedWrite参数为true时,直接往寄存器中写数据,这个函数最好搭配中断使用。

从上面我们知道,mcmgr_early_init_internal实际上就是通过自己核对应的MU的通道3发送一个组合的32位数(高16位为type,低16位为eventData,这里typekMCMGR_RemoteCoreUpEventeventData为0)给对方核对应的MU的通道3。

  • 如果不知道MU的,建议看一下我之前介绍MU的文章:双核通信之MU消息单元详解

2 MCMGR_Init函数

接着就是调用MCMGR_Init函数:

mcmgr_status_t MCMGR_Init(void)
{// 通过OCOTP熔丝相应位可以判断当前是CM4(返回0)还是CM7核(返回1)mcmgr_core_t coreNum = MCMGR_GetCurrentCore();// 两个回调函数MCMGR_RegisterEvent(kMCMGR_StartupDataEvent, MCMGR_StartupDataEventHandler, (void *)&s_mcmgrCoresContext[coreNum]);MCMGR_RegisterEvent(kMCMGR_FeedStartupDataEvent, MCMGR_FeedStartupDataEventHandler, (void *)&s_mcmgrCoresContext[(coreNum == 0) ? 1 : 0]);return mcmgr_late_init_internal(coreNum);
}

MCMGR_RegisterEvent用来注册某个事件(参数一)的回调函数(参数二),其中参数三s_mcmgrCoresContext会传给回调函数作为其参数供其使用,它的定义如下:

typedef struct _mcmgr_core_context
{/*! @brief Current state of the core. */mcmgr_core_state_t state;/*! @brief Startup data, if state >= kMCMGR_RunningCoreState */uint32_t startupData;
} mcmgr_core_context_t;/*! @brief Type definition of possible core states. */
typedef enum _mcmgr_core_state
{kMCMGR_ResetCoreState = 0,kMCMGR_StartupGettingLowCoreState,kMCMGR_StartupGettingHighCoreState,kMCMGR_RunningCoreState,
} mcmgr_core_state_t;volatile mcmgr_core_context_t s_mcmgrCoresContext[2] = {{.state = kMCMGR_ResetCoreState, .startupData = 0}, {.state = kMCMGR_ResetCoreState, .startupData = 0}};

看样子似乎是一个状态机,其中:

  • s_mcmgrCoresContext[0]用于kMCMGR_StartupDataEvent事件的MCMGR_StartupDataEventHandler回调

  • s_mcmgrCoresContext[1]用于kMCMGR_FeedStartupDataEvent事件的MCMGR_FeedStartupDataEventHandler回调

具体完成了什么我们后面用到了再分析。

2.1 MCMGR_RegisterEvent函数

顾名思义就是用来注册回调函数的,实现也非常简单,就是定义了一个结构体数组,然后填充即可:

/*! @brief Type definition of structure with event handler and data. */
typedef struct _mcmgr_event
{/*! @brief Pointer to callback function. */mcmgr_event_callback_t callback;/*! @brief Context data for callback. */void *callbackData;
} mcmgr_event_t;mcmgr_event_t MCMGR_eventTable[kMCMGR_EventTableLength] = {0};mcmgr_status_t MCMGR_RegisterEvent(mcmgr_event_type_t type, mcmgr_event_callback_t callback, void *callbackData)
{if (type >= kMCMGR_EventTableLength){return kStatus_MCMGR_Error;}MCMGR_eventTable[type].callback = ((void *)0);MCMGR_eventTable[type].callbackData = callbackData;MCMGR_eventTable[type].callback = callback;return kStatus_MCMGR_Success;
}

mcmgr_event_type_t有8种事件,每个事件占据MCMGR_eventTable数组的一个索引。

接下来看一下两个回调函数完成了什么:

2.2 事件回调函数

下面来看一下MCMGR_StartupDataEventHandlerMCMGR_FeedStartupDataEventHandler

static void MCMGR_StartupDataEventHandler(uint16_t startupDataChunk, void *context)
{mcmgr_core_context_t *coreContext = (mcmgr_core_context_t *)context;switch (coreContext->state){case kMCMGR_StartupGettingLowCoreState:coreContext->startupData = startupDataChunk; /* Receive the low part */coreContext->state       = kMCMGR_StartupGettingHighCoreState;(void)MCMGR_TriggerEvent(kMCMGR_FeedStartupDataEvent, (uint16_t)kMCMGR_StartupGettingHighCoreState);break;case kMCMGR_StartupGettingHighCoreState:coreContext->startupData |= ((uint32_t)startupDataChunk) << 16;coreContext->state = kMCMGR_RunningCoreState;(void)MCMGR_TriggerEvent(kMCMGR_FeedStartupDataEvent, (uint16_t)kMCMGR_RunningCoreState);break;default:break;}
}static void MCMGR_FeedStartupDataEventHandler(uint16_t startupDataChunk, void *context)
{mcmgr_core_context_t *coreContext = (mcmgr_core_context_t *)context;switch ((mcmgr_core_state_t)startupDataChunk){case kMCMGR_StartupGettingLowCoreState:(void)MCMGR_TriggerEvent(kMCMGR_StartupDataEvent, (uint16_t)(coreContext->startupData & 0xFFFFU));coreContext->state = (mcmgr_core_state_t)startupDataChunk;break;case kMCMGR_StartupGettingHighCoreState:(void)MCMGR_TriggerEvent(kMCMGR_StartupDataEvent, (uint16_t)((coreContext->startupData) >> 16));coreContext->state = (mcmgr_core_state_t)startupDataChunk;break;case kMCMGR_RunningCoreState:coreContext->state = (mcmgr_core_state_t)startupDataChunk;break;default:break;}
}

这里的context就是前面注册回调函数时的第三个参数s_mcmgrCoresContext[0/1],前面我们看到默认的statekMCMGR_ResetCoreState,所以不会进入任何分支中,具体初始状态在何时改变的,我们后续分析。

我们看到这里两个Handler最后都是调用MCMGR_TriggerEvent函数,即通过MU发送一个32位数给对方核。

2.3 mcmgr_late_init_internal

MCMGR_Init最后调用mcmgr_late_init_internal打开MU的通道3的接收中断:
(下面代码为CM7核的,CM4核打开的是MUB)

mcmgr_status_t mcmgr_late_init_internal(mcmgr_core_t coreNum)
{MU_EnableInterrupts(MUA, (uint32_t)kMU_Rx3FullInterruptEnable);NVIC_SetPriority(MUA_IRQn, 2);NVIC_EnableIRQ(MUA_IRQn);return kStatus_MCMGR_Success;
}

在前面的MCMGR_TriggerEvent中,最后也是使用通道3发送的消息,所以在SDK中使用MU的通道3来完成双核执行的同步。

3 MCMGR_StartCore函数

对于CM7来说,注册完回调函数之后,还需要调用MCMGR_StartCore来启动CM4核。

MCMGR_StartCore(kMCMGR_Core1, (void *)(char *)CORE1_BOOT_ADDRESS, 2, kMCMGR_Start_Synchronous);

具体实现如下:

#define CORE1_BOOT_ADDRESS (void *)0x20200000
MCMGR_StartCore(kMCMGR_Core1, (void *)(char *)CORE1_BOOT_ADDRESS, 2, kMCMGR_Start_Synchronous); //kMCMGR_Core1=1mcmgr_status_t MCMGR_StartCore(mcmgr_core_t coreNum, void *bootAddress, uint32_t startupData, mcmgr_start_mode_t mode)
{mcmgr_status_t ret;/* 填充startupData */s_mcmgrCoresContext[coreNum].startupData = startupData;/* 设置相关寄存器 */ret = mcmgr_start_core_internal(coreNum, bootAddress);if (mode == kMCMGR_Start_Synchronous){/* 等待M4核读取和确认我们刚刚填充的startupData */while (s_mcmgrCoresContext[coreNum].state != kMCMGR_RunningCoreState){}}return kStatus_MCMGR_Success;
}

这里假设我们将CM4的程序通过CM7的映射地址0x20200000拷贝到CM4的TCM中了,如果CM4的程序在NOR Flash中,填写对应的地址即可。

mcmgr_start_core_internal就是我们上一篇文章双核相互激活和启动流程提到的CM7激活CM4相关寄存器的修改:

mcmgr_status_t mcmgr_start_core_internal(mcmgr_core_t coreNum, void *bootAddress)
{IOMUXC_LPSR_GPR->GPR0 = IOMUXC_LPSR_GPR_GPR0_CM4_INIT_VTOR_LOW(((uint32_t)(char *)bootAddress) >> 3u);IOMUXC_LPSR_GPR->GPR1 = IOMUXC_LPSR_GPR_GPR1_CM4_INIT_VTOR_HIGH(((uint32_t)(char *)bootAddress) >> 16u);SRC->CTRL_M4CORE = SRC_CTRL_M4CORE_SW_RESET_MASK;SRC->SCR |= SRC_SCR_BT_RELEASE_M4_MASK;return kStatus_MCMGR_Success;
}

4 MCMGR_GetStartupData函数

在CM4核启动后会调用MCMGR_GetStartupData函数,直到这个函数返回kStatus_MCMGR_Success

do{status = MCMGR_GetStartupData(&startupData);} while (status != kStatus_MCMGR_Success);

现在来看一下这个函数:

mcmgr_status_t MCMGR_GetStartupData(uint32_t *startupData)
{if (s_mcmgrCoresContext[1].state == kMCMGR_ResetCoreState){s_mcmgrCoresContext[1].state = kMCMGR_StartupGettingLowCoreState;if (kStatus_MCMGR_Success !=MCMGR_TriggerEvent(kMCMGR_FeedStartupDataEvent, (uint16_t)kMCMGR_StartupGettingLowCoreState)){return kStatus_MCMGR_Error;}}return mcmgr_get_startup_data_internal(1, startupData);
}mcmgr_status_t mcmgr_get_startup_data_internal(mcmgr_core_t coreNum, uint32_t *startupData)
{if (s_mcmgrCoresContext[1].state >= kMCMGR_RunningCoreState){*startupData = s_mcmgrCoresContext[1].startupData;return kStatus_MCMGR_Success;}return kStatus_MCMGR_NotReady;
}

实际上也是和刚刚的状态机相关。

5 MU中断接收函数

现在我们对CM7和CM4的交互过程还是一头雾水,前面注册的回调函数什么时刻被调用,CM7启动CM4后等待s_mcmgrCoresContext[coreNum].state变为kMCMGR_RunningCoreState,还有CM4启动后,CM4也要等待状态变化再往下执行程序,那么这些状态是在哪里被修改的呢?下面就来分析一下这个过程。

前面我们打开了中断,所以我们首先看一下中断处理回调函数,在通道三收到数据后将调用此回调函数:(下面为CM7核MUA的回调,MUB的类似)

void MU_Rx3FullFlagISR(void)
{uint32_t data;uint16_t eventType;uint16_t eventData;#if defined(FSL_FEATURE_MU_SIDE_A)data = MU_ReceiveMsgNonBlocking(MUA, 3);
#elif defined(FSL_FEATURE_MU_SIDE_B)data = MU_ReceiveMsgNonBlocking(MUB, 3);
#endif/* To be MISRA compliant, return value needs to be checked even it could not never be 0 */if (0U != data){eventType = (uint16_t)(data >> 16u);eventData = (uint16_t)(data & 0x0000FFFFu);if (((mcmgr_event_type_t)eventType >= kMCMGR_RemoteCoreUpEvent) &&((mcmgr_event_type_t)eventType < kMCMGR_EventTableLength)){if (MCMGR_eventTable[(mcmgr_event_type_t)eventType].callback != ((void *)0)){MCMGR_eventTable[(mcmgr_event_type_t)eventType].callback(eventData, MCMGR_eventTable[(mcmgr_event_type_t)eventType].callbackData);}}}
}
  • 在理论上我们的程序中没有发送数据内容为0的代码,但是为了符合MISRA规范,这里还是检查了0U != data

前面在MCMGR_TriggerEvent中,我们将typeevent组合成一个32位的数发送给对方,这里同样的,我们收到数据后取出高16位的type和低16位的event。然后调用我们使用MCMGR_RegisterEvent注册的回调函数,第一个参数为eventData,第二个参数为我们注册的时候提供的callbackData(这里为s_mcmgrCoresContext)。

6 双核状态机交互过程详解

看完了中断函数后,感觉两个核有一些联系了,我们先来看一下两个核的执行流程:
在这里插入图片描述
这些函数前面都分析过了,但是里面状态机的状态改变似乎有些复杂,而状态的改变是通过双核之间的通道3进行交互的,这里我们就来捋清里面的流程:

  • 这里通道间发送数据为32位,高16位为type,低16位为eventData,下面都表示为(type, eventData)

1、MCMGR_EarlyInit
发送32位数据,(kMCMGR_RemoteCoreUpEvent,0)。由于后面我们在MCMGR_Init函数中并没有注册kMCMGR_RemoteCoreUpEvent的回调函数,实际上这个消息会被忽略。

2、MCMGR_Init
这里没有发送任何数据,但是注册了两个回调函数,在回调函数中会发送数据:
在这里插入图片描述

3、启动CM4:MCMGR_StartCore

s_mcmgrCoresContext[1].startupData = 2;
while (s_mcmgrCoresContext[1].state != kMCMGR_RunningCoreState);

这里CM7将s_mcmgrCoresContext[1]startupData设置为了2,然后等待s_mcmgrCoresContext[1]state变为kMCMGR_RunningCoreState

4、CM4和CM7消息同步
(1)CM4在MCMGR_GetStartupData中将s_mcmgrCoresContext[1]state设置为了kMCMGR_StartupGettingLowCoreState,然后向CM7发送(kMCMGR_FeedStartupDataEvent, kMCMGR_StartupGettingLowCoreState)

(2)在CM7接收到这个32位消息后,将进入MCMGR_FeedStartupDataEventHandler中,向CM4发送(kMCMGR_StartupDataEvent, 2),然后将state设置为kMCMGR_StartupGettingLowCoreState

case kMCMGR_StartupGettingLowCoreState:(void)MCMGR_TriggerEvent(kMCMGR_StartupDataEvent, (uint16_t)(coreContext->startupData & 0xFFFFU));coreContext->state = (mcmgr_core_state_t)startupDataChunk;break;

此时双核的状态如下:
在这里插入图片描述

(3)CM4收到(kMCMGR_StartupDataEvent, 0),进入MCMGR_StartupDataEventHandler的下面分支:

case kMCMGR_StartupGettingLowCoreState:coreContext->startupData = startupDataChunk; /* Receive the low part */coreContext->state       = kMCMGR_StartupGettingHighCoreState;(void)MCMGR_TriggerEvent(kMCMGR_FeedStartupDataEvent, (uint16_t)kMCMGR_StartupGettingHighCoreState);break;

startupData设置为2,state设置为kMCMGR_StartupGettingHighCoreState,然后发送(kMCMGR_FeedStartupDataEvent, kMCMGR_StartupGettingHighCoreState)给CM7。

(4)CM7收到(kMCMGR_FeedStartupDataEvent, kMCMGR_StartupGettingHighCoreState),进入MCMGR_FeedStartupDataEventHandler的下面分支:

case kMCMGR_StartupGettingHighCoreState:(void)MCMGR_TriggerEvent(kMCMGR_StartupDataEvent, (uint16_t)((coreContext->startupData) >> 16));coreContext->state = (mcmgr_core_state_t)startupDataChunk;break;

这里发送(kMCMGR_StartupDataEvent, (uint16_t)(2>> 16))给CM4,然后设置自身的statekMCMGR_StartupGettingHighCoreState

此时双核的状态如下:
在这里插入图片描述

(5)CM4收到(kMCMGR_StartupDataEvent, 0)后,进入MCMGR_StartupDataEventHandler的下面分支:

case kMCMGR_StartupGettingHighCoreState:coreContext->startupData |= ((uint32_t)startupDataChunk) << 16;coreContext->state = kMCMGR_RunningCoreState;(void)MCMGR_TriggerEvent(kMCMGR_FeedStartupDataEvent, (uint16_t)kMCMGR_RunningCoreState);break;

startupData与之前收到的低16位进行组合,然后赋到startupData中,即CM7在MCMGR_StartCore函数中的第三个参数传给了CM4。然后将state设置为kMCMGR_RunningCoreState,并向CM7发送(kMCMGR_FeedStartupDataEvent, kMCMGR_RunningCoreState)

(6)CM7收到(kMCMGR_FeedStartupDataEvent, kMCMGR_RunningCoreState)后进入MCMGR_FeedStartupDataEventHandler中的kMCMGR_RunningCoreState分支:

case kMCMGR_RunningCoreState:coreContext->state = (mcmgr_core_state_t)startupDataChunk;break;

最终就将state设置为了kMCMGR_RunningCoreState。此时在MCMGR_StartCore中等待s_mcmgrCoresContext[1]state变为kMCMGR_RunningCoreState则成立,此时CM7知道CM4已经成功启动。

最终的状态如下:
在这里插入图片描述

7 总结

从上面状态机分析可知,CM7仅用了kMCMGR_FeedStartupDataEvent,而CM4仅用了kMCMGR_StartupDataEvent。在CM4启动后,先发送一个消息给CM7,然后CM7开始传startupData给CM4,最终CM7的状态都变为kMCMGR_RunningCoreState,表示CM7知道CM4已经启动了,就可以执行其它操作了。

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

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

相关文章

Java面试题一

1、JDK 和 JRE 有什么区别&#xff1f; JDK是Java的开发工具包&#xff1b;而JRE是Java的运行环境 其中JDK中包含JRE、JDK中有一个名为jre的目录&#xff0c;里面包含两个文件夹bin和lib&#xff0c;bin就是JVM&#xff0c; lib就是JVM工作所需要的类库 2、 和 equals 的区别是…

MFC 程序执行流程

目录 MFC 程序启动 MFC 入口函数 程序执行流程总结 在Win32课程中WinMain由程序员自己实现&#xff0c;那么流程是程序员安排&#xff0c;但到了MFC中&#xff0c;由于MFC库实现WinMain&#xff0c;也就意味着MFC负责安排程序的流程。 MFC 程序启动 程序的启动&#xff0c;…

Node.js初学习

目录 1、Node.js简介 2、npm是什么 3、node.js和vue是什么关系 1、Node.js简介 Introduction to Node.js | Node.js 根据官网的介绍&#xff1a;Node.js是一个开源的跨平台JavaScript运行时环境。Node.js在浏览器之外运行V8 JavaScript引擎&#xff0c;这是谷歌Chrome的核…

React 状态

大家好&#xff0c;欢迎来到 React 状态的课程。在这一课中&#xff0c;我们将学习如何在 React 中使用状态。 什么是状态&#xff1f; 状态是组件的数据。组件的状态可以通过 this.state 对象访问。 class ComponentName extends React.Component {constructor(props) {sup…

PCB设计规则中的经验公式_笔记

PCB设计规则中的经验公式 规则1 - 临界长度规则2 - 信号带宽与上升时间规则3- 时钟信号带宽规则4-信号传输速度规则5- 集肤 (效应) 深度规则6 - 50Ω传输线电容规则7 - 50Ω传输线电感规则8 - 回流路径电感规则9 - 地弹噪声规则10- 串行传输比特率与信号带宽规则11- PCB走线直流…

HIVE窗口函数

什么是窗口函数 hive中开窗函数通过over关键字声明&#xff1b;窗口函数&#xff0c;准确地说&#xff0c;函数在窗口中的应用&#xff1b;比如sum函数不仅可在group by后聚合&#xff0c;在可在窗口中应用&#xff1b; hive中groupby算子和开窗over&#xff0c;shuffle的逻辑…

面试 Java 算法高频题五问五答第一期

面试 Java 算法高频题五问五答第一期 作者&#xff1a;程序员小白条&#xff0c;个人博客 相信看了本文后&#xff0c;对你的面试是有一定帮助的&#xff01; ⭐点赞⭐收藏⭐不迷路&#xff01;⭐ 1&#xff09;括号生成: 数字 n 代表生成括号的对数&#xff0c;请你设计一个…

OpenSergo Dubbo 微服务治理最佳实践

*作者&#xff1a;何家欢&#xff0c;阿里云 MSE 研发工程师 Why 微服务治理&#xff1f; 现代的微服务架构里&#xff0c;我们通过将系统分解成一系列的服务并通过远程过程调用联接在一起&#xff0c;在带来一些优势的同时也为我们带来了一些挑战。 如上图所示&#xff0c;可…

C语言之枚举类型

目录 枚举类型 枚举常量 枚举类型的特征 命名空间 本节我们来学习表示一定整数值的集合的枚举类型。 枚举类型 老样子&#xff0c;我们先用一段程序引出&#xff1a; /*显示所选动物的叫声*/ #include<stdio.h>enum animal {Dog, Cat, Monkey, Invalid}; /*显示狗叫…

Zotero攻略

给大家分享一下我对于Zotero的使用。 1、下载链接 Zotero | Your personal research assistant 进入后直接下载即可 2、一些好用的插件 &#xff08;1&#xff09;Zotero Connector 下载地址&#xff1a;Zotero | Connectors 超级好用&#xff01;不用一篇一篇下PDF了&am…

Redis设计与实现之事务

一、事务 Redis 通过 MULTI 、DISCARD 、EXEC 和 WATCH 四个命令来实现事务功能&#xff0c;本章首先讨 论使用 MULTI 、DISCARD 和 EXEC 三个命令实现的一般事务&#xff0c;然后再来讨论带有 WATCH 的事务的实现。 因为事务的安全性也非常重要&#xff0c;所以本章最后通过…

【JavaWeb】Request(学习笔记)

一、Request概述 1、request对象和response对象的原理 tomcat服务器会根据请求u1中的资源路径&#xff0c;创建对应的ServletDemo1对象tomcat 服务器&#xff0c;会创建request对象和response对象&#xff0c;request对象中封装请求消息数据tomcat将request和response两个对象…

18个非技术面试题

请你自我介绍一下你自己&#xff1f; 这道面试题是大家在以后面试过程中会常被问到的&#xff0c;那么我们被问到之后&#xff0c;该如果回答呢&#xff1f;是说姓名&#xff1f;年龄&#xff1f;还是其他什么&#xff1f; 最佳回答提示&#xff1a; 一般人回答这个问题往往会…

为什么参数上必须加@RequestBody?

PostMapping("/login")public Result<Map<String,Object>> login(RequestBody User user){return Result.success();} 在这个 PostMapping("/login") 的控制器方法中&#xff0c;使用 RequestBody 注解的目的是将请求体中的 JSON 数据映射到方…

函数柯里化 剖析

定义一个函数&#xff0c;以闭包的方式内部函数调用外部函数,并返回内部函数 function calc(n){ //args是获取的第一次参数的值let argsArray.prototype.slice.call(arguments)// 上面的args写法等同于下方的for循环// for (var i 0; i < arguments.length; i) {// A…

neuq-acm预备队训练week 9 P1119 灾后重建

解题思路 本题可以用最短路算法——Floyd AC代码 #include<bits/stdc.h> #define inf 1e9 using namespace std; const int N 2e2 50; int n, m, q, now 0, a, b, c, t[N], G[N][N];int main() {scanf("%d%d", &n, &m);for(int i 0;i<n;i)sc…

设计模式-GOF对各个模式的定义

以下内容是对设计模式之父GOF的著作《设计模式——可复用面向对象软件的基础》定义的摘抄 1 抽象工厂 意图 提供一个接口以创建一系列相关或相互依赖的对象&#xff0c;而无须指定它们具体的类。 适用性 在以下情况下使用抽象工厂模式&#xff1a; 一个系统要独立于它的产…

2023新时代中国模特大赛总决赛在京落幕

12月16日&#xff0c;备受瞩目的2023新时代中国模特大赛圆满落幕。本次大赛旨在挖掘和培养具有新时代特色的模特人才&#xff0c;推动中国时尚产业的创新发展。 作为中国时尚界的重要赛事&#xff0c;新时代中国模特大赛吸引了来自全国各地的优秀模特选手45名参加全国总决赛。在…

ReactHooks大全—useState

React Hooks是React 16.8版本引入的一种新的编程范式&#xff0c;它可以让我们在不使用class的情况下&#xff0c;使用state和其他React特性&#xff0c;。React Hooks的出现&#xff0c;不仅提高了函数组件的功能和复用性&#xff0c;也简化了组件的编写和维护&#xff0c;让我…

Spring 依赖查找知识点总结

前言 源码在我github的guide-spring仓库中&#xff0c;可以克隆下来 直接执行。 我们本文主要来介绍依赖查找的使用示例 依赖查找 什么是依赖查找 依赖查找并不是 Spring 框架特有的概念&#xff0c;它是一种在软件开发中获取依赖对象的方式。它通常用于获取运行时需要的服…