Stm32-使用TB6612驱动电机及编码器测速

news2024/11/26 14:23:56

这里写目录标题

    • 起因
    • 一、电机及编码器的参数
    • 二、硬件
    • 三、接线
    • 四、驱动电机
      • 1、TB6612电机驱动
      • 2、定时器的PWM模式驱动电机
    • 五、编码器测速
      • 1、定时器的编码器接口模式
      • 2、定时器编码器模式测速的原理
      • 3、编码器模式的配置
      • 4、编码器模式相关代码
      • 5、测速方法
    • 六、相关问题以及解答
      • 1、编码器模式下的自动重装值ARR和预分频PSC应该如何设置
      • 2、如何判断正反转
      • 3、圈数如何计算
      • 4、转速如何计算
      • 5、为什么我的编码器没有输出,获取到的脉冲数是0
    • 七、测速硬件展示及测速现象
    • 八、总结
    • 九、大家可以参考
      • 参考链接1
      • 参考链接2
      • 参考链接3
      • 参考链接4

起因

最近在学习编码电机以及尝试使用编码电机测速。遇到了很多问题,花费了很多时间,在这里做一个记录,对自己学习到的知识进行一个总结

找了很多资料,看了很多视频,这些太多了,以至于让我不知道究竟哪一个是正确的,今天看这个,明天看这个,导致自己的学习效率低下

当然,有很多大佬的文章和资料给了我很大的启发

这个电机我玩了四天,把自己觉得重要的东西和大家分享一下

现在一般都是用编码器电机,参加比赛啥的,编码电机常用于测速,所以编码电机就成了一个必须学习的知识点

编码器被广泛应用于电机测速,实现电机闭环控制。

相关的知识点有:定时器的输出比较(输出PWM)、定时器的输入捕获,定时器的编码器接口、速度控制


一、电机及编码器的参数

编码电机其实就是一个带有编码器的电机,我的这个电机是一个增量式的带霍尔传感器的电机

电机的型号是JGB37-520电机

下方是电机的参数
主要关注的就是电机的额定电压 12V
电机的减速比 30(这个很重要)

编码器的参数
主要关注编码器的线数 11线 (也就是说电机转一圈会产生11个脉冲)
供电电压 5V
输出类型 方波
在这里插入图片描述

编码器的连接

一般这种编码器都有六根线
两边靠外的两根线是电机电源线
往里两根是编码器的电源线
中间两根是编码器的A,B相
在这里插入图片描述

具体大概是啥意思呢?
就是电机转动的时候编码器会通过编码电机的A相和B相输出两个正交的方波

通过输出的两个方波就可以对电机进行测速识别电机的方向


二、硬件

整体结构采用洞洞板+TB6612+Stm32C8T6+编码电机(起初采用的是这种结构)

后面采用Stm32ZET6+TB6612+洞洞板+编码电机+12V电源(原因是C8T6烧坏了,哭😥)

主控Stm32C8T6 or Stm32ZET6
电机驱动 TB6612(由于上一个L298N烧了)
520霍尔编码电机
12V电源


三、接线

这里展示驱动一个编码电机的示例,毕竟先从一个电机玩起,弄懂后后面就会使用的更加得心应手啦

主要使用到了定时器的PWM模式(输出比较)功能

大家一定要认真接线,看清出每根线的作用,不要随便接线,一不小心电机驱动就烧了,或者是单片机烧了(在学习的时候就烧了一个单片机,人民币-15)

注意这个是我实现的接线,大家可以根据自己单片机的片上资源合理选择,选择合适的IO口

电机驱动

TB6612C8T6
STBY高电平(+3.3V)
AIN1PB14
AIN2PB15
PWMAPA8 (TIM1-CH1)
AO1电机电源+
AO2电机电源-
VM12V
VCC3.3V
GND和单片机共地

编码器

编码器的A、B相C8T6
A相PA0 (TIM2-CH1)
B相PA1 (TIM2-CH2)

四、驱动电机

1、TB6612电机驱动

首先了解一下TB6612

下图是TB6612驱动模块
原理图
在这里插入图片描述
STBY接高电平 清零电机全部停止
置 1 通过 AIN1 AIN2, BIN1,BIN2 引脚来控制正反转
PWM引脚控制占空比
VM: 接 12V 以内电源
VCC: 接 5V 电源
GND: 接电源负极

下图是驱动逻辑
可以看出IN引脚控制正反转,PWM引脚控制速度

2、定时器的PWM模式驱动电机

使用定时器的PWM模式 生成一个需要的 占空比可调的 频率 符合要求的方波信号。

方波信号的频率不宜过高或者过低,过高容易导致电机驱动的晶闸管经常处于开关状态–发热巨大过低则容易产生噪音,对电机也低频的冲击

这里输出PWM信号的定时器是TIM1-CH1

设置成PWM模式,频率和占空比可调

有关定时器PWM模式,可以看其他大佬的文章和资料,看看手册

可以看看江科大的教学视频,比我讲的详细多了,也很好理解

我贴出视频链接,大家学习32的时候可以跟他
TIM输出比较,PWM模式

下方的PWM模式的代码作为一个参考

Motor.h

#ifndef __MOTOR_H
#define __MOTOR_H
#include "sys.h"	 

#define PWMA   TIM1->CCR1  //PA8  PWMA  TIM1_CH1
#define AIN2   PBout(15)
#define AIN1   PBout(14)

void Motor_PWM_Init(u16 arr,u16 psc);
void Motor_SetSpeed(u8 mode ,u16 speed);

#endif

Motor.c

void Motor_Init(void)	//IN引脚初始化
{
    GPIO_InitTypeDef GPIO_InitStructure;
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能PB端口时钟
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14|GPIO_Pin_15;	//端口配置
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;      //推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;     //50M
    GPIO_Init(GPIOB, &GPIO_InitStructure);				 //根据设定参数初始化GPIOB
}
void Motor_PWM_Init(u16 arr,u16 psc)	//PWM引脚初始化
{
    GPIO_InitTypeDef GPIO_InitStructure;
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    TIM_OCInitTypeDef  TIM_OCInitStructure;

    Motor_Init();

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_TIM1 | RCC_APB2Periph_AFIO,ENABLE);//开启时钟

    //输出TIM1 CH1 
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; //TIM_CH1 
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //复用推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);//初始化定时器。
    TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
    TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值  不分频
    TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
    TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位


    TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //选择定时器模式:TIM脉冲宽度调制模式1
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
    TIM_OCInitStructure.TIM_Pulse = 0;                            //设置待装入捕获比较寄存器的脉冲值
    TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;     //输出极性:TIM输出比较极性高
    TIM_OC1Init(TIM1, &TIM_OCInitStructure);  //根据TIM_OCInitStruct中指定的参数初始化外设TIMx

    TIM_CtrlPWMOutputs(TIM1,ENABLE);	//MOE 主输出使能		高级定时器一定要写这个语句

    TIM_ARRPreloadConfig(TIM1, ENABLE); //使能TIMx在ARR上的预装载寄存器

    TIM_Cmd(TIM1, ENABLE);  //使能TIM1
}

void Motor_SetSpeed(u8 mode ,u16 speed)		//mode 代表正反转 speed PWM占空比即速度
{
    PWMA = speed;
    if(mode==1)
    {
        AIN1 = 1;
        AIN2 = 0;
    }
    else {
        AIN1 = 0;
        AIN2 = 1;
    }
}

五、编码器测速

编码器一般应用于电机控制,使用PWM驱动电机,然后再使用编码器测量速度,再使用PID算法进行闭环控制

记住下面这句话

在一定的时间内,电机转动一圈,通过霍尔传感器的A、B两相输出一定数量的脉冲,我们可以根据一定时间内的脉冲数计算出电机的瞬时速度。

1、定时器的编码器接口模式

采用的是定时器的编码器接口模式,Stm32中的定时器只有TIM1-5和TIM8才有编码器接口功能,而且只有CH1通道和CH2通道有用。

2、定时器编码器模式测速的原理

原理:接收编码器的A、B相产生的正交信号,根据编码器产生的正交信号脉冲,自动控制CNT自增或自减,根据计数方向和编码器的信号关系来指示编码器的位置、旋转方向和旋转速度利用脉冲值来计算电机的转动位移

这个可以参考手册里定时器的编码器模式,比我讲的清楚多了
定时器的编码器接口托管了输入捕获的前两个接口

还有一句话记住,编码器模式下就相当于一个带有方向选择的外部时钟

3、编码器模式的配置

具体配置流程就是

时钟–>GPIO–>时基单元配置–>编码器接口配置–>开启定时器–>读取一个时间段内的脉冲–>计算电机旋转轴转速

使用这个函数把定时器设置为编码器接口模式

TIM_EncoderInterfaceConfig(TIM3, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);

采用的是编码器模式3,在TI1和TI2边沿都计数,也就是在一个周期内对A相和B相的上升沿下降沿都计数一个周期内计4次,所以采用这种模式后,相应的计数值(CNT)就会变成4倍,这就是很多资料里说的四倍频计数

4、编码器模式相关代码

采用的是定时器2的编码器接口模式,通道1和通道2捕获
encoder.c

/**
 * @brief 把TIM2初始化为编码器接口模式
 * @param psc 预分频系数
 * @param arr 自动重装载值
 * @retval None
 */
void Encoder_Init_TIM2(uint16_t psc,uint16_t arr)	
{
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    TIM_ICInitTypeDef TIM_ICInitStructure;
    GPIO_InitTypeDef GPIO_InitStructure;
    //NVIC_InitTypeDef  NVIC_InitStructure;



    //使能定时器2的时钟
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
    //使能PB端口时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);



    //端口配置
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
    //浮空输入
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    //根据设定参数初始化GPIOB
    GPIO_Init(GPIOA, &GPIO_InitStructure);



    TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
    // 预分频器
    TIM_TimeBaseStructure.TIM_Prescaler = psc;
    //设定计数器自动重装值
    TIM_TimeBaseStructure.TIM_Period = arr;
    //选择时钟分频:不分频
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    //TIM向上计数
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
    TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);



    //使用编码器模式3
    TIM_EncoderInterfaceConfig(TIM2, \
                               TIM_EncoderMode_TI12, \
                               TIM_ICPolarity_Rising, \
                               TIM_ICPolarity_Rising);
    TIM_ICStructInit(&TIM_ICInitStructure);
    TIM_ICInitStructure.TIM_ICFilter = 10;
    TIM_ICInit(TIM2, &TIM_ICInitStructure);



    //清除TIM的更新标志位
    TIM_ClearFlag(TIM2, TIM_FLAG_Update);
    TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);

    //Reset counter
    TIM_SetCounter(TIM2,0);
    TIM_Cmd(TIM2, ENABLE);
}

/**
 * @brief 单位时间读取编码器计数
 * @param TIMX 定时器
 * @retval 速度值	是编码器返回的脉冲
 */
int Read_Encoder()
{
    	int Encoder_TIM;
        Encoder_TIM= (short)TIM2 -> CNT;
        //Encoder_TIM= (int)((int16_t)(TIM4->CNT));;
        TIM2 -> CNT=0;
   	    return Encoder_TIM;
}

encoder.h

#ifndef __ENCODER_H
#define __ENCODER_H


#include "sys.h"
	 
void Encoder_Init_TIM2(uint16_t psc,uint16_t arr);
int Read_Encoder();


#endif

5、测速方法

在一个时间周期T0内,定时的读取编码器产生的脉冲,以我的编码器为例(11线,减速比30),转一圈会产生1320个脉冲(因为采用的是编码模式3)
这个1320 = 11 * 30 * 4

通过在固定的周期T0内,产生的脉冲就相当于路程,而这个固定的周期就相当于时间

所以速度就等于 在T0这段时间获取到的脉冲总数/(编码器单圈产生的总脉冲数*T0)

对于我的电机就是

T0这段时间获取到的脉冲总数/(1320*T0)

六、相关问题以及解答

在看了前面之后,应该对编码器模式和编码器测速有了一个大概的认识,知道了它测速的原理,但肯定有好多疑问,我把我学习过程中遇到的问题和解决方法做一个总结,你肯定也有这些疑问,不要着急,看下去。

1、编码器模式下的自动重装值ARR和预分频PSC应该如何设置

ARR,自动重装值,指的是CNT计数自增或自减到ARR就会溢出,(可以产生中断),然后继续从0开始计数。

PSC,预分频系数,(前面不是说过编码器相当于一个外部时钟吗),PSC相当于外部时钟的频率,如果分频(假设PSC=2)的话,就会(比如电机转一圈产生100个脉冲),此时编码器模式下只能计数50个脉冲。

所以我们应该如何设置?

PSC呢? PSC没有必要设置,因为我要计数的本来就是电机转动一圈产生的真实脉冲,所以PSC给0就好啦

ARR呢? 目前在各种论坛和博客和资料中有两种版本。

第一种,根据电机的线数和减速比来设置,比如我的电机是11线,减速比30,转动一圈的脉冲数是1320,这个值就可以设置为1320。

产生的脉冲数恰好是你定时器溢出的时候,溢出一次记录一次,这个的次数就是电机的圈数(当然这种误差很大)

也就是说电机转一圈正好是1320,当CNT计数到ARR时,计数器就会清零并且重新计数,所以这个ARR就是电机转一圈产生的脉冲数的最大值。

第二种,直接设置成定时器ARR的最大值,也就是65535(2^16-1),这样设置的目的就是无论你电机产生多少脉冲,都可以记录,且不会溢出。

不过使用65535的话,就要在最开始的时候初始化编码器模式提前把CNT清零,然后再开始计数。再在一个周期内定时读取脉冲数,再清零,这个脉冲也是周期内读取到的脉冲值。

电机旋转一圈能产生脉冲,那么我们就能记录一段时间产生的脉冲数来计算速度

2、如何判断正反转

可以通过判断CR1寄存器中的DIR位,这个位是计数方向位

在这里插入图片描述

正转就是CNT向上计数(DIR==0)

反转就是CNT向下计数(DIR==1)

3、圈数如何计算

就是上方说的,把ARR设置为电机旋转一周产生的脉冲数

电机转一圈,CNT达到ARR,溢出,进中断,设置一个变量++(正转),–(反转)

4、转速如何计算

规定某个某个时钟周期内,读取有多少脉冲,从而计算转速

这里采用的是M法测速,测出的是电机是多少转/s

脉冲相当于路程,某个时钟周期相当于时间

这个上方有描述,可以往上翻翻

5、为什么我的编码器没有输出,获取到的脉冲数是0

检查一下接线,从硬件开始,一步一步排查,对应的引脚是否正确

检查电源,编码器的电源是否打开,相应的PWM波是否有效

硬件确认没有错误,检查软件,编码器接口是否打开,PWM模式是否输出,电机IN引脚是否配置

七、测速硬件展示及测速现象

硬件展示
在这里插入图片描述
测速展示
可以看到当转速为1的时候,产生的脉冲是66-68,而我设置的闸门时间是50ms,结合电机转一圈是1320个脉冲,也就是说测量的脉冲数几乎正确。

(66*2=132) 和编码器的线数完全吻合,测速成功!!!
在这里插入图片描述

八、总结

这个编码器花费了我几乎四天的时间,也可能是自己比较小白,不懂得如何通过电机转一圈产生的脉冲数来计算速度,第一天就实现了读取脉冲

但是后面几天执着于测速,没有采取正确的方法,导致自己无线内耗,浪费了大量时间。

由于网上资料繁杂,找不到自己想要的,浪费了很多时间,很多只是一笔带过,没有系统的讲解原理和方法,我也不知道自己这篇文章是否正解

所以将这个学习总结分享给大家

当然,这都是在借鉴了前人的肩膀下,谢谢各位大佬和优秀的文章,我会在下方贴出自己觉得值得一看大佬们的链接,大家可以一看

欢迎大家指错,看到了就会修改,大家一起共同进步

九、大家可以参考

参考链接1

带霍尔传感器编码器的直流减速电机测速原理讲解(附源码)-OpenEdv-开源电子网

参考链接2

编码器分类及原理和测速应用(含代码) - 古月居

参考链接3

stm32平衡小车(2)-----编码器电机驱动

参考链接4

带编码器的直流减速电机——基于STM32F407_编码器直流减速电机磁极数目_谁还不是个程序猿的博客

谢谢各位大佬的文章,让我受益匪浅,站在前人的肩膀下才能看的更远

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

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

相关文章

关于Cesium的常见需求整理之点位和弹窗(点位弹窗)

一、点位上图 ①在Cesium中,每个自定义的地图元素被视为一个entity对象,如果我们要添加点位到地图上,那就必须先创建一个entity对象。 var entity new Cesium.Entity({position: position, });以上代码我们创建了一个entity对象&#xff0…

Autosar通信入门系列06-聊聊CAN通信的线与机制与ACK应答

本文框架 1. 概述2. CAN通信的线与机制3. ACK应答机制理解 1. 概述 本文为Autosar通信入门系列介绍,如您对AutosarMCAL配置,通信,诊断等实战有更高需求,可以参见AutoSar 实战进阶系列专栏,快速链接:AutoSa…

数据库基础(增删改查)

目录 MySQL 背景知识 数据库基础操作 1.创建数据库 2.查看所有数据库 3.选中指定的数据库 4.删除数据库 数据库表操作 MySQL的数据类型 1.创建表 3.查看指定表的结构 4.删除表 增删改 新增操作 修改(Updata) 删除语句 面试题 查询操作 指定列查询 查询的列为表达式…

系统设计:通用思路之4S分析法

1.系统设计 系统设计是一个定义系统架构、功能模块、服务及接口和数据存储等满足特定需求的过程。 与面向对象设计不同的是,面向对象设计通常是对于某个特定功能模块的设计,通常要求设计类图关系、接口关系、实现关系等涉及具体代码层面的设计&#xff…

C语言库函数之 qsort 讲解、使用及模拟实现

引入 我们在学习排序的时候&#xff0c;第一个接触到的应该都是冒泡排序&#xff0c;我们先来复习一下冒泡排序的代码&#xff0c;来作为一个铺垫和引入。 代码如下&#xff1a; #include<stdio.h>void bubble_sort(int *arr, int sz) {int i 0;for (i 0; i < sz…

基于chatgpt动手实现一个ai_translator

动手实现一个ai翻译 前言 最近在极客时间学习《AI 大模型应用开发实战营》&#xff0c;自己一边跟着学一边开发了一个进阶版本的 OpenAI-Translator&#xff0c;在这里简单记录下开发过程和心得体会&#xff0c;供有兴趣的同学参考&#xff1b; ai翻译程序 版本迭代 在学习…

C语言必会题目(2)

W...Y的主页 &#x1f60a; 代码仓库分享&#x1f495; 今天继续分享C语言必会的题目&#xff0c;上一篇文章主要是一些选择题&#xff0c;而今天我们主要内容为编程题的推荐与讲解 准备好迎接下面的题了吗&#xff1f;开始发车了&#xff01;&#xff01;&#xff01; 输入…

pytest运行时参数说明,pytest详解,pytest.ini详解

一、Pytest简介 1.pytest是一个非常成熟的全功能的Python测试框架&#xff0c;主要有一下几个特点&#xff1a; 简单灵活&#xff0c;容易上手&#xff0c;支持参数化 2.能够支持简单的单元测试和复杂的功能测试&#xff0c;还可以用来做selenium、appium等自动化测试&#xf…

zookeeper案例

目录 案例一&#xff1a;服务器动态上下线 服务端&#xff1a; &#xff08;1&#xff09;先获取zookeeper连接 &#xff08;2&#xff09;注册服务器到zookeeper集群&#xff1a; &#xff08;3&#xff09;业务逻辑&#xff08;睡眠&#xff09;&#xff1a; 服务端代码…

提高生产力 | Apifox 数据结构验证最佳实践

目录 实践场景 定义返回响应 场景数据准备 校验响应数据 总结 在设计接口的过程中&#xff0c;响应数据需要和返回响应规范一一对应。这样能够确保接口的一致性和可靠性&#xff0c;并且方便接口的使用和维护&#xff0c;即使在后续迭代过程中出现问题&#xff0c;开发人员…

zabbix监控安装部署

目录 一、环境 二、配置 1.配置yum源&#xff0c;这里用的清华的 2.过滤一下安装包&#xff0c;查看依赖包 安装依赖包 3.配置数据库 开机自启 创建数据库 创建用户 授权 导入数据到数据库 查看zabbix数据库有没有表和数据 4.修改zabbix配置文件 1.修改zabbix配置…

【Java】常见面试题:多线程

文章目录 1. 谈谈进程和线程之间的区别【高频】2. java中有哪些方式来创建线程&#xff1f;3. run和start的区别【经典面试题】4. Java线程的状态5. 【线程不安全的原因】6. 就以count为例&#xff1a;一个线程加锁、一个线程不加锁&#xff0c;此时能否保证线程的安全呢&#…

client-go实战之十二:选主(leader-election)

欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码)&#xff1a;https://github.com/zq2599/blog_demos 本篇概览 本文是《client-go实战》系列的第十二篇&#xff0c;又有一个精彩的知识点在本章呈现&#xff1a;选主(leader-election)在解释什么是选主之前&…

Keepalived源码安装

文章目录 Keepalived源码安装安装准备缺少OpenSSL解决方法 Keepalived 源码安装 安装准备 tar zxf keepalived-2.2.8.tar.gz /root/ ll drwxrwxr-x. 10 1000 1000 4096 Aug 9 18:29 keepalived-2.2.8 #进入目录执行以下命令查看帮助 ./configure --help #重要编译参数 -…

QT学习笔记-oracle oci数据库驱动交叉编译并移植到ARM开发板

QT学习笔记-oracle oci数据库驱动交叉编译并移植到RK3568ARM开发板 0、背景1、搭建交叉编译环境2、交叉编译过程3、把数据库驱动部署到目标系统中 0、背景 在上一文《QT学习笔记-QT安装oracle oci驱动》中介绍了在Windows环境下使用QT访问oracle数据库时遇到驱动无法加载问题的…

kingbase:数据库启动状态

1 启停KingbaseES数据库 Linux下通过系统服务&#xff1a; root用户执行&#xff1a; service kingbase8d stop/start/restart ——注册服务的情况下 Linux下通过安装用户&#xff1a; 安装用户执行&#xff1a; sys_ctl stop/start/restart -D data路径 2 查看数据库当…

UI自动化环境的搭建(python+pycharm+selenium+chrome)

最近在做一些UI自动化的项目&#xff0c;为此从环境搭建来从0到1&#xff0c;希望能够帮助到你&#xff0c;同时也是自我的梳理。将按照如下进行开展&#xff1a; 1、python的下载、安装&#xff0c;python环境变量的配置。 2、pycharm开发工具的下载安装。 3、selenium的安装。…

【Java】一只小菜坤的编程题之旅【3】

文章目录 1丶判定是否互为字符重排2、杨辉三角3丶某公司的1个面试题&#xff08;字符串包含问题&#xff09; 1丶判定是否互为字符重排 这个题我们用一个非常简单的思想就能实现&#xff0c;我们先将字符串转换为字符数组&#xff0c;然后对字符数组进行排序&#xff0c;然后再…

Codeforces Round 893 (Div. 2)ABC

Codeforces Round 892 (Div. 2) 目录 A. United We Stand题目大意思路代码 B. Olya and Game with Arrays题目大意思路代码 C. Another Permutation Problem题目大意思路代码 A. United We Stand 题目大意 给你一个数组&#xff0c;把这个数组分成两个数组a和b&#xff0c;使…

03_013内存分配api以及页表详解

前言 之前文章中物理ram中的最小单位一直用页来表示 这次又描述的详细了点 物理ram的最小单位 有的地方叫 块,框,页帧 在虚拟空间中最小单位也叫页 需要好好区分 不过后来想想管你虚拟页还是物理ram页 都存在物理ram上 都能想成一 一对应的关系 所以大家都叫页好像也行 内存分…