嵌入式实时操作系统的设计与开发(五)

news2025/1/20 13:29:28

线程退出

当线程代码执行完后,系统会隐式地调用acoral_thread_exit()函数进行线程退出相关的操作。
acoral_thread_exit()本质上是要执行acoral_kill_thread()。

void acoral_thread_exit(){
	acoral_kill_thread(acoral_cur_thread);
}
void acoral_kill_thread(acoral_thread_t *thread){
	acoral_sr CPU_sr;
	acoral_8 CPU;
	acoral_evt_t *evt;
	acoral_pool_t *pool;
	acoral_enter_critical();
	if(thread->state&ACORAL_THREAD_STATE_SUSPEND){
		evt = thread->evt;
		if(thread->state&ACORAL_THREAD_STATE_DELAY){
			acoral_list_del(&thread->waiting);
		}
		else{
			if(evt!=NULL){
				acoral_evt_queue_del(thread);
			}
		}
	}
	acoral_unrdy(thread);
	acoral_release_thread1(thread);
	acoral_exit_critical();
	acoral_sched();
}

释放线程就是回收当时创建线程时所分配的空间。

void acoral_release_thread1(acoral_thread_t *thread){
	acoral_list_t *head,*tmp;
	acoral_thread_t *daem;
	thread->state = ACORAL_THREAD_STATE_EXIT; // 将线程置为退出状态,如果是当前线程,只能是ACORAL_THREAD_STATE_EXIT状态,表明还不能释放该线程的资源,如TCB、堆栈,因为该线程尽管要退出了,但是还没完成退出使命,要继续走到线程切换HAL_SWITCH_TO函数,在该过程中,还有函数调用,故还需要堆栈,切换线程前还需要线程的TCB。
	tmp=head->prev;
	acoral_list_add2_tail(&thread->waiting,head);

	daem=(acoral_thread_t *)acoral_get_res_by_id(daemon_id);
	acoral_rdy_thread(daem);
}

挂起线程

操作系统在运行时,有时需要挂起某个线程。
例如,当某一线程运行时需要请求某一资源,而该资源正被其它线程所占用,此时用户线程需要挂起自己。
有两种线程挂起方式,对应的接口分别是acoral_suspend_thread()、acoral_unrdy_thread()。acoral_suspend_thread()比acoral_unrdy_thread()多一个acoral_sched(),也就是acoral_suspend_thread()会立刻将指定线程挂起(将其从就绪队列中取出),然后重新调度。而比acoral_unrdy_thread()只改变指定线程的状态,将其从就绪队列中取出,并不会马上触发重新调度。

void acoral_suspend_thread(acoral_thread_t *thread){
	acoral_sr cpu_sr;
	acoral_8 cpu;
	if(!(ACORAL_THREAD_STATE_READY&thread->state))
		return;
	acoral_enter_critical();
	acoral_rdyqueue_del(thread);
	acoral_exit_critical();
	acoral_sched();
}
/*将线程从就绪队列取下*/
void acoral_rdyqueue_del(acoral_thread_t *thread){
	acoral_rdy_queue_t *rdy_queue;
	rdy_queue = &acoral_ready_queues;
	acoral_prio_queue_del(rdy_queue, thread->prio, &thread->ready);
	thread->state &= ~ACORAL_THREAD_STATE_READY;
	thread->state &= ~ACORAL_THREAD_STATE_RUNNING;
	thread->state |= ~ACORAL_THREAD_STATE_SUSPEND;
	//设置线程所在的核可调度
	acoral_set_need_sched(true);
}
void acoral_prio_queue_del(acoral_rdy_queue_t *array, acoral_u8 prio, acoral_list_t *list){
	acoral_queue_t *queue;
	acoral_list_t *head;
	queue = array->queue + prio; //根据线程的优先级找到线程所在的优先级链表
	head = &queue->head;
	array->num--;
	acoral_list_del(list); //从链表上删除该线程
	if(acoral_list_empty(head))
		acoral_clear_bit(prio,array->bitmap);//如果该优先级不存在就绪线程,则清除该优先级就绪标志位。
}

任务挂起接口用到的地方很多,只要牵涉任务的等待都会调用该函数。
看线程TCB的waiting成员是否为空,如果因为等待时间或资源导致suspend,则waiting肯定挂在一个队列上,则是调用acoral_delay_self()导致线程suspend;否则是直接调用acoral_suspend_thread()导致suspend。

线程恢复

线程恢复是线程挂起的逆过程,和线程挂起类似,线程恢复也有两个接口acoral_resume_thread()、acoral_rdy_thread()。
前者比后者多了一个acoral_sched(),还多了一个判断,并且acoral_resume_thread()是用户能够调用的,而acoral_rdy_thread()由内核内部调用。

void acoral_resume_thread(acoral_thrad_t *thread){
	if(!(thread->state&ACORAL_THREAD_STATE_SUSPEND))
		return;
	acoral_enter_critical();
	acoral_rdyqueue_add(thread);
	acoral_exit_critical();
	acoral_sched();
}

改变线程优先级

当多个线程互斥地访问某一共享资源时,可能导致优先级反转,优先级反转将造成实时调度算法的不确定性。
解决优先级反转的方法是优先级继承和优先级天花板,使用这两种方式时,需要动态改变线程优先级。
aCoral描述线程优先级时,采用的是优先级队列,每个优先级是一个链表,因此改变线程的优先级,不是简单地将线程的TCB的prio变量更改,最终要通过acoral_thread_change_prio()实现,将线程挂到要设置的优先级的链表上去。

void thread_change_prio(acoral_thread_t *thread, acoral_u32 prio){
	acoral_enter_critical();
	if(thread->state&ACORAL_THREAD_STATE_READY){
		acoral_rdyqueue_del(thread);
		thread->prio = prio;
		acoral_rdyqueue_add(thread);
	}else{
		thread->prio = prio; //线程恢复时,会自动挂载到新的优先级队列上。
	}
	acoral_exit_critical();
}

调度策略时间处理函数

系统启动后,晶体振荡器源源不断地产生周期性信号,通过设置,晶体振荡器可以为系统产生稳定的Ticks,也称为心跳,Tick是系统的时基,也是系统中最小的时间单位,Tick的大小可以根据晶体振荡器的精度和用户的需求进行设置。
每当产生一个Tick,都对应着一个时钟中断服务程序ISR,时钟中断服务程序的具体实现是acoral_Ticks_entry()。其中包括了几项重要工作:延迟队列的处理time_delay_deal(),将挂到延迟队列中的线程的 TCB的delay成员值依次减一,如果某一线程的TCB的delay成员值减到了0,将触发aCoral重新调度acoral_sched();与调度策略相关的处理函数acoral_plicy_delay_deal(),如果采用周期性调度策略的线程或者采用时间片轮转调度策略的线程,acoral_plicy_delay_deal()将维护每个线程的周期和时间片,每当某一线程新的周期到达,或者时间片到达,都重新调度acoral_sched();超时处理函数timeout_delay_deal();用户也可以在acoral_Tciks_entry()扩展自己所需的函数。

void acoral_ticks_entry(acoral_vector vector){
	ticks++;
	if(acoral_start_sched == true){
		time_delay_deal();
		acoral_policy_delay_deal();
		timeout_delay_deal();
	}
}

对于acoral_policy_delay_deal(),不同的调度策略有不同的时间处理函数delay_deal(),这是在调度策略注册并初始化时绑定的,这充分体现了调度策略与调度机制分离的设计原则。

void acoral_policy_delay_deal(){
	acoral_list_t *tmp,*head;
	acoral_sched_policy_t *policy_ctrl;
	head = &policy_list.head;
	tmp = head;
	for(tmp=head->next;tmp!=head;tmp=tmp->next){
		policy_ctrl = list_entry(tmp,acoral_sched_policy_t,list);
		if(policy_ctrl->delay_deal!=NULL)
			policy_ctrl->delay_deal();
	}
}
slice_policy_init(){
	slice_policy.type = ACORAL_SCHED_POLICY_SLICE;
	slice_policy.policy_thread_release = slice_policy_thread_release;
	slice_policy.policy_thread_init = slice_policy_thread_init;
	slice_policy.delay_deal = slice_delay_deal;
	slice_policy.name = "slice";
	acoral_register_sched_policy(&slice_policy);
}
void slice_delay_deal(){
	acoral_thread_t *cur;
	slice_policy_data_t *data;
	if(cur->policy==ACORAL_SCHED_POLICY_SLICE){
		cur->slice--;
		if(cur->slice <= 0){
			data = (slice_policy_data_t *)cur->private_data;
			cur->slice = data->slice_ld;
			acoral_thread_move2_tail(cur);//把当前线程置为队列尾部
		}
	}
}
void acoral_thread_move2_tail(acoral_thread_t *thread){
	acoral_enter_critical();
	acoral_unrdy_thread(thread);// 将当前线程从就绪队列中移出
	acoral_rdy_thread(thread); // 将线程队列中的下一线程置为就绪线程
	acoral_exit_critical();
	acoral_sched();
}

事务处理机制

内核启动后,用户可以默认地创建几个线程。
在没有用户干预的情况下,哪个线程先执行?哪个线程后执行?哪个线程执行多长时间,完全由acoral_sched()根据用户指定的调度策略(如周期性策略、时间片轮转策略等)来决定,而周期性策略、时间片轮转策略等需要时钟来触发和维护。
如果用户线程执行完毕,内核将安排idle线程执行。
如果用户需要干预系统运行,并通过按钮、键盘等输入设备提出新的事务请求,内核将通过中断机制接收并进行相关处理,若事务过于复杂造成中断服务程序ISR难以处理,ISR的最后部分将创建新的线程来接收用户的请求,再acoral_sched()安排处理。
如果多个线程并发执行过程中因资源暂时无法获取、异常等内部原因造成当前线程无法继续执行,内核也将通过中断来挂起当前线程,再由acoral_sched()安排其它线程执行。

以上便是内核提供的事务处理机制,事务处理机制可分为事件触发机制(Event-triggered)和时间触发机制(Time-triggered)。

  • 通过中断响应、处理来触发内核重调度的机制属于事件触发机制。
  • 通过时钟维护、管理来触发内核重调度的机制属于时间触发机制。

中断管理

中断是一种硬件机制,其优先级高于系统中所有任务,它的产生将会中断某个线程或者正在执行程序的运行。

中断发生及响应

硬件抽象层HAL响应。中断请求IRQ被中断控制器汇集成中断向量(Interrupt Vector),每个中断向量对应一个中断服务程序ISR,中断向量存放了ISRs的入口地址或ISRs的第一条指令。
系统中通常包含多个中断向量,存放这些中断向量对应的ISRs入口地址的内存区域被称为中断向量表。
在Interl80x86处理器中,中断向量表包含256个入口地址,每个中断向量需要四字节存放ISR的首地址。
ARM处理器的中断请求IRQ将被中断控制器汇集到异常向量(每个异常向量四个字节),该异常向量位于ARM异常向量表的第7条记录,由于ARM异常向量表存放在内存0x00000000开始处,中断请求IRQ将被中断控制器汇集到0x00000018(第7条记录的起始地址)的内存地址,也就是说,当IRQ发生时,PC将通过硬件机制跳转到0x00000018开始执行。
在这里插入图片描述
根据ARM的中断机制,0x00到0x1C存放的是8条跳转指令。
当系统复位时,pc跳转到0x00(该地方是一条四字节的跳转指令),此时pc的值将变成“VECTOR_TABLE+0”。
VECTOR_TABLE是ARM处理器中断向量表的起始地址(用户可自己定义)。
当处理器发生IRQ时,pc将跳转到0x18,该地方是另一四字节的跳转指令LDR pc,VECTORE_TABLE+0x18,此时pc的值变为VECTORE_TABLE+0x18。

VECTORE_TABLE:
	...
	.long HAL_INTR_ENTRY
	...

VECTORE_TABLE+0x18存放的是HAL_INTR_ENTRY,它是指向所有IRQ的一个公共入口。当各种IRQ发生时,都将汇拢到HAL_INTR_ENTRY,进行与硬件相关的处理,也称为HAL层中断处理。

HAL_INTR_ENTRY:
    stmfd   sp!,    {r0-r12,lr}           @保护通用寄存器及PC 
    mrs     r1,     spsr
    stmfd   sp!,    {r1}                  @保护spsr,以支持中断嵌套

    msr     cpsr_c, #SVC_MODE|NOIRQ        @进入SVC_MODE,以便允许中断嵌套
    stmfd   sp!,    {lr}                  @保存SVc模式的专用寄存器lr

    ldr     r0,     =INTOFFSET 		  @读取中断向量号
    ldr     r0,     [r0]			@将INTOFFSET的值赋给r0
    mov     lr,    pc                     @求得lr的值
    ldr     pc,    =hal_all_entry 		@跳转到用C语言编写的中断公共入口函数hal_all_entry(),将RO作为参数传给hal_all_entry()

    ldmfd   sp!,    {lr}                    @恢复svc模式下的lr,
    msr     cpsr_c,#IRQ_MODE|NOINT       @更新cpsr,进入IRQ模式并禁止中断
    ldmfd   sp!,{r0}                    @spsr->r0
    msr     spsr_cxsf,r0                @恢复spsr
    ldmfd   sp!,{r0-r12,lr}
    subs    pc,lr,#4                    @此后,中断被重新打开

首先将寄存器R0~R12的值保存在中断模式下堆栈指针指向的内存地址,然后保存中断返回地址LR(LR存放的是中断发生时,被中断程序的下一条指令)到堆栈中,然后保存SPSR,以支持中断嵌套。上述压栈的顺序为:R0,R1,…,R12,LR,CPSR。
将ARM处理器切换到SVC模式,并且保存SVC模式连接寄存器LR到堆栈中,到此完成了中断现场保护的工作。
读取中断控制器中INTOFFSET寄存器的值,对INTOFFSET的读写操作与其它内存地址一致。当某个IRQ发生时,INTOFFSET会为该中断源分配一个整数(如时钟中断,对应的整数是0),这个整数也唯一地对应于该中断源。
这样可通过读取该寄存器分辨不同的中断源,进而执行不同的ISR。

/*
	中断入口,对中断复用进行展开
	vector 中断向量号,来自INTOFFSET寄存器
*/
	void hal_all_entry(acoral_vector vector){
		unsigned long eint;
		unsigned long irq=4;
		if(vector==4||vector==5){
			eint=rEINTPND;
			for(;irq<24;irq++){
				if(eint & (1<<irq)){
					acoral_intr_entry(irq);
					return;
				}
			}
		}
		if(vector>5){
			vector+=18;
		}
		if(vector==4)
			acoral_prints("DErr\n");
		acoral_intr_entry(vector);
	}

HAL层处理完成后,通过hal_all_entry进入内核层响应,hal_all_entry是与ARMS3C2440中断控制器密切相关的设置,经过相关处理后,会调用真正的中断公共服务入口函数acoral_intr_entry(),开始内核层的中断处理。

ARMS3C2440的中断机制中,第4号中断到第7号中断复用了中断号4,第8号中断到第23号中断复用了中断号5。
因此,当从INTOFFSET中读取的值为4或者5时,要根据寄存器EINTPND的值进一步区分中断源。
四号到七号之间包括四个中断,八号到二十三号之间包括十六个中断,再去除4号和5号本身,共有18个中断。
当从INTOFFSET中读取的值R0大于5时,实际对应在内核层中的中断号应该是R0+18。

/*中断公共服务入口函数*/
void acoral_intr_entry(acoral_vector vector){
	acoral_vector index;
	HAL_TRANSLATE_VECTOR(vector,index);
	acoral_intr_nesting_inc();
	//根据转换后的中断号index,从中断向量表中找到相应的中断向量,如果中断的类型属于专家模式,则根据内核中断标号找到对应的中断结构体,然后调用这个中断的ISR,并且通过acoral_intr_disable()除能中断关中断。
	if(intr_table[index].type == ACORAL_EXPERT_INTR){
		intr_table[index].isr(vector);
		acoral_intr_disable();
	}else{
		if(intr_table[vector].enter!=NULL){
			intr_table[vector].enter(vector);
			/*开中断*/
			acoral_intr_enable();// 在进入hal_all_entry()之前是要临时关中断的,但是中断处理程序往往较长,如果整个过程都关中断,必然影响中断的性能,因此为了保证中断性能和实时性,此时要开启中断。
			intr_table[vector].isr(vector);//ISR是在中断初始化时进行注册和绑定的
			/*关中断*/
			acoral_intr_disable();
			if(intr_table[vector].exit!=NULL){
				intr_table[vector].exit(vector);
			}
		}
	}
	acoral_intr_nesting_dec();
	acoral_intr_exit();
}

由于ARMS3C2440的中断复用机制,内核层的中断号比HAL层的中断号多。
另一方面,内核层的中断号也可少于HAL层的中断号,因为有些特殊中断或异常时不需要交给内核层处理,所以可能造成内核层的中断数减少,故HAL层中断与内核层中断的对应关系不一样。
因此,需要一个转换,将HAL层的中断号转换为内核层的中断号,该转换是通过HAL_TRANSLATE_VECTOR()实现的。
还要将中断嵌套数加1,因为aCoral只有在最后一层中断退出时才执行调度函数,因此需要一个变量来记录中断嵌套数。
aCoral通过acoral_intr_ctr_t表示内核层的中断向量表。

typedef struct{
	acoral_u32 index; // 内核层中断号
	acoral_u8 state;	//中断状态
	acoral_u8 type;
	void (*isr)(acoral_vector); //ISR
	void (*enter)(acoral_vector);	//中断服务程序执行之前执行的操作,aCoral中为hal_intr_ack函数。
	void (*exit)(acoral_vector); 	//中断服务程序执行后完成的操作
	void (*mask)(acoral_vector); 	//除能中断
	void (*unmask)(acoral_vector); 	//使能中断
}acoral_intr_ctr_t;

中断类型

#define ACORAL_COMM_INTR 1 //普通中断,会自动调用acoral_intr_ctr_t中的enter成员清楚中断,用户只需要关心isr即可。
#define ACORAL_EXPERT_INTR 2 //专家中断,不会调用enter成员,需要在isr中手动清除中断,如DM9000网卡。
#define ACORAL_RT_INTR 3	//实时中断

acoral_intr_ctr_t表示内核层的中断向量表,显然,它比HAL层的异常向量表丰富,如中断进入时处理(清除中断Pending标志位),中断退出时处理(置中断结束位)、中断屏蔽、中断开启等。

中断状态是内核层状态,其实中断在HAL层也有状态,如挂起、正在处理、处理完毕等,内核层的state除了包含这状态外,还可以增加一些状态以满足特殊需求。这样设计的目的是为了增加中断的灵活性和可扩展性。

acoral的三种中断模式

  1. 实时模式:中断处理程序直接调用,不经过HAL_INTR_ENTRY->acoral_intr_entry->中断处理程序。这种方式明显减少了中断响应时间,增加了中断的实时性。
  2. 专家模式:这种模式需要用户在中断处理程序中自己处理中断响应相关的操作,如清中断位等操作,这种模式主要应用于特殊中断,如aCoral的网卡中断,就需要在中断里关闭中断,这种方法没有办法调用统一中断模型。
  3. 普通模式:中断处理程序往往需要使用汇编,并对处理相关寄存器进行操作。aCoral的普通模式的中断处理程序不需要任何中断硬件相关的操作,只需要编写中断要做什么事情即可。

acoral_intr_enable()、acoral_intr_disable()是与硬件相关的操作。

HAL_INTR_ENABLE:
    mrs r0,cpsr
    bic r0,r0,#NOINT
    msr cpsr_cxsf,r0
    mov pc,lr
HAL_INTR_DISABLE:
    mrs r0,cpsr
    mov r1,r0
    orr r1,r1,#NOINT
    msr cpsr_cxsf,r1
    mov pc ,lr
/*中断退出函数*/
void acoral_intr_exit()
{
	if(!acoral_need_sched){// 中断退出时不需要调度,则直接退出
		return;
	}
	if(acoral_intr_nesting){// 中断如果属于嵌套中断,则直接退出
		return;
	}
	if(acoral_sched_is_lock){// 调度标志未开启,则直接退出
		return;
	}
	if(!acoral_start_sched){
		return;
	}
	// 如果有任务需要调度
	HAL_INTR_EXIT_BRIDGE();
}
#define HAL_INTR_EXIT_BRIDGE() hal_intr_exit_bridge_comm()
void hal_intr_exit_bridge_comm(){
	HAL_ENTER_CRITICAL();
	acoral_real_intr_sched();
	HAL_EXIT_CRITICAL();	
}

在这里插入图片描述
Save ri表示保存ARM S3C2440寄存器的值(保存现场)

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

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

相关文章

ccbill 代码分析

ccbill目录概述需求&#xff1a;设计思路实现思路分析1.BillState2.DBList3.DBListAttr4.DBListDBSrc5.DBListDBSrcsDBListsSurvive by day and develop by night. talk for import biz , show your perfect code,full busy&#xff0c;skip hardness,make a better result,wai…

每日一题:Leetcode59. 螺旋矩阵 II

文章目录 系列&#xff1a;数组专练 语言&#xff1a;java & go 题目来源&#xff1a;Leetcode59. 螺旋矩阵 II 难度&#xff1a;中等 考点&#xff1a;边界问题的处理 & 圈数处理 思路和参考答案文章目录题目描述思路java解法&#xff1a;java参考代码go参考代码&…

OceanBase 4.0解读:从TPC-H性能测评看4.0与3.x差异

关于作者 肖帆 OceanBase技术专家 OceanBase技术专家&#xff0c;开源生态团队成员。毕业于华中科技大学软件工程专业&#xff0c;从事数据库领域的质量保障工作&#xff0c;曾就职于有赞、网易&#xff0c;参与关系型数据库、缓存数据库、对象存储相关产品的测试开发&#x…

.net core api调用webserver接口(详细)

这里废话不多说&#xff0c;我就不简述什么事webserver了&#xff0c;相信点进本博客的大佬都是为了解决问题。 .net core 调用webserver的话还挺简单。首先我们先有个.net core api的项目。 1.我们先注入这个HttpClient 这个内置对象&#xff0c;一会要用到。 // 注入HttpC…

Vue Hook Event 解读

前言 Hook Event (钩子事件)相信很多 Vue 开发者都没有使用过&#xff0c;甚至没听过&#xff0c;毕竟 Vue 官方文档中也没有提及。 Vue 提供了一些生命周期钩子函数&#xff0c;供开发者在特定的逻辑点添加额外的处理逻辑&#xff0c;比如: 在组件挂载阶段提供了beforeMount 和…

代码随想录day34

1005. K 次取反后最大化的数组和 题目 给你一个整数数组 nums 和一个整数 k &#xff0c;按以下方法修改该数组&#xff1a; 选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。 重复这个过程恰好 k 次。可以多次选择同一个下标 i 。 以这种方式修改数组后&#xff0c;返回数…

linux笔记(9):MangoPi-MQ(芒果派麻雀D1s)Tina系统编译烧录

文章目录1.下载相关资料1.1 WhyCan Forum(哇酷开发者社区)提供的sdk1.1.1 SDK解压过程1.2 WhyCan Forum(哇酷开发者社区)提供的补丁1.2.1 补丁包含的文件1.2.2 补丁文件和D1下面的相同文件进行合并1.2.3 引脚PD17被复用&#xff0c;导致LCD变暗&#xff0c;修改设备树2. 编译ti…

【node.js 安装】linux下安装node.js

下面我们介绍安装包安装方法 nodejs官网下载地址1 nodejs官网下载地址2 我们以官网下载地址2打开 直接下载源代码&#xff0c;rz上传到/opt/tools/ 目录下 tar -xJvf node-v18.13.0-linux-x64.tar.xz配置环境变量&#xff0c;vim /etc/profile &#xff0c;配置内容如下&am…

SFP 收发器居然有那么多种?值得收藏学习

SFP 模块具有广泛的应用范围&#xff0c;可与大部分现代网络配合使用&#xff0c;大多数可以分为四大类&#xff1a;电缆类型、传输范围、传输速率、应用。 一、电缆类型 SFP 模块可以在光纤和铜线上工作&#xff0c;根据光纤的种类&#xff0c;SFP收发器可分为与单模光纤配合…

π122E60 5.0kVrms 200Mbps 双通道数字隔离器 兼容代替Si8622BT-IS

π122E60 5.0kVrms 200Mbps 双通道数字隔离器 兼容代替Si8622BT-IS 具有出色的性能特征和可靠性&#xff0c;整体性能优于光耦和基于其他原理的数字隔离器产品。 产品传输通道间彼此独立&#xff0c;可实现多种传输方向的配置&#xff0c;可实现 5.0kVrms 隔离耐压等级和 DC 到…

Loading 用户体验 - 加载时避免闪烁

&#x1f353; 前言 在切换详情页中有这么一个场景&#xff0c;点击上一条&#xff0c;会显示上一条的详情页&#xff0c;同理&#xff0c;点击下一条&#xff0c;会显示下一条的详情页。 伪代码如下所示&#xff1a; 我们定义了一个 switcher 模版&#xff0c; 用户点击上一…

TensorRT部署YOLOv5(03)-TensorRT介绍

TensorRT是本专栏中最重要的内容,绝大多数内容将围绕TensorRT来展开,本文对TensorRT进行一个基本的介绍,让不熟悉TensorRT的读者能够对TensorRT是什么,如何使用它有一个较为全面的认识 Nvidia TensorRT是一个用于Nvidia GPU上高性能机器学习推理的SDK,对开发者屏蔽了模型…

到底什么样的 REST 才是最佳 REST?

说起 REST API&#xff0c;小伙伴们多多少少都有听说过&#xff0c;但是如果让你详细介绍一下什么是 REST&#xff0c;估计会有很多人讲不出来&#xff0c;或者只讲出来其中一部分。 今天松哥就来和大家一起来聊一聊到底什么是 REST&#xff0c;顺便再来看下 Spring HATEOAS 的…

[算法与数据结构]——并查集

目录 1. 概论 定义&#xff1a; 主要构成&#xff1a; 作用&#xff1a; 2. 并查集的现实意义 故事引入&#xff1a; 数据结构的角度来看&#xff1a; 3. find( )函数的定义与实现 故事引入&#xff1a; 实现&#xff1a; 4. join( )函数的定义与实现 故事引入&#xff1a; 实现…

c++11 标准模板(STL)(std::forward_list)(三)

定义于头文件 <forward_list> template< class T, class Allocator std::allocator<T> > class forward_list;(1)(C11 起)namespace pmr { template <class T> using forward_list std::forward_list<T, std::pmr::polymorphic_…

Apollo本地快速部署

GitHub项目地址 Gitee项目地址 Apollo&#xff08;阿波罗&#xff09;是携程框架部门研发的分布式配置中心&#xff0c;能够集中化管理应用不同环境、不同集群的配置&#xff0c;配置修改后能够实时推送到应用端&#xff0c;并且具备规范的权限、流程治理等特性&#xff0c;适…

【计算机网络-数据链路层】介质访问控制协议(MAC协议)

文章目录1 静态划分信道——信道划分 MAC 协议1.1 频分多路复用&#xff08;FDM&#xff09;——“并行”1.2 时分多路复用&#xff08;TDM&#xff09;——“并发”1.2.1 同步时分多路复用1.2.2 异步时分多路复用1.3 波分多路复用&#xff08;WDM&#xff09;1.4 码分多路复用…

数据结构进阶 AVL树

作者&#xff1a;小萌新 专栏&#xff1a;数据结构进阶 作者简介&#xff1a;大二学生 希望能和大家一起进步&#xff01; 本篇博客简介&#xff1a;介绍高阶数据结构:AVL树 AVL树AVL树的概念AVL树节点类的定义AVL树的插入AVL树的旋转左单旋右单旋左右双旋右左单旋AVL树的验证…

多线程适用接口及常见类

日升时奋斗&#xff0c;日落时自省 目录 1、Callable接口 1.1、Callable方式 1.2、非Callable方式 2、JUC&#xff08;java.util.concurrent&#xff09;的常见类 2.1、ReentrantLock 2.2、信号量Semaphore 2.3、CountDownLatch 3、线程安全的集合类 3.1、多线程使用A…

【SpringMVC】使用SpringMVC处理JSON格式的数据

目录 一、前言 二、ResponseBody 三、RequestBody 四、HttpMessageConverter 相关文章&#xff08;可以关注我的SpringMVC专栏&#xff09; SpingMVC专栏SpingMVC专栏一、前言我们在使用Servlet处理前端请求&#xff0c;使用Json格式的数据&#xff0c;通常引入外部提供的一些…