零基础入门学用Arduino 第二部分(一)

news2024/11/28 0:51:55

重要的内容写在前面:

  1. 该系列是以up主太极创客的零基础入门学用Arduino教程为基础制作的学习笔记。
  2. 个人把这个教程学完之后,整体感觉是很好的,如果有条件的可以先学习一些相关课程,学起来会更加轻松,相关课程有数字电路(强烈推荐先学数电,不然可能会有一些地方理解起来很困难)、模拟电路等,然后就是C++(注意C++是必学的)
  3. 文章中的代码都是跟着老师边学边敲的,不过比起老师的版本我还把注释写得详细了些,并且个人认为重要的地方都有详细的分析。
  4. 一些函数的介绍有参考太极创客官网给出的中文翻译,为了便于现查现用,把个人认为重要的部分粘贴了过来并做了一些修改。
  5. 如有错漏欢迎指正。

视频链接:2-1 MeArm项目概述_哔哩哔哩_bilibili

太极创客官网:太极创客 – Arduino, ESP8266物联网的应用、开发和学习资料

一、伺服电机(舵机)

1、认识伺服电机

(1)伺服电机是一种根据输入PWM信号占空比来控制输出角度的装置(二者具体有什么对应关系取决于电机的型号),其硬件结构如下图所示,电位器与转轴相连,转轴每一个转动角度都分别对应电位器的一个旋转角度,控制电路负责将电位器部分传来的信号和输入的PWM信号作比对,如果当前转轴的转动角度不符合输入PWM信号占空比对应的输出角度,则控制电路将控制电机带动齿轮组旋转,直到转轴的旋转角度与输入的PWM信号相对应。

(2)舵机一般有三根引线需要连接,且引线的颜色一般有统一标准,如下所示,其中橙色线负责给电机输入PWM信号,它不一定必须连接在Arduino的9号引脚,只要带PWM输出的引脚都可以连接。

(3)通常舵机需要的供电功率比较高,如果需要控制1个以上的舵机,那么必须使用单独的电源为舵机供电,而不能使用Arduino开发板的+5V引脚为1个以上的舵机供电。另外,如果使用独立电源为舵机供电,那么独立电源与Arduino必须共地(电源GND与Arduino的GND相互连接在一起)。

2、Servo库概述

(1)通过Servo库可以用Arduino控制舵机(伺服电机)常见的Arduino舵机有两种,一种舵机轴可以旋转180度,还有一种舵机轴可以连续旋转,这种舵机的旋转速度是可以控制的

(2)Servo库允许大多数Arduino开发板(如Arduino Uno)同时控制12个舵机,对于Arduino Mega,可以用Servo库同时控制48个舵机。

(3)请注意:使用Servo库可能会影响Arduino开发板的某些引脚的PWM功能。

(4)对于除Mega开发板以外的Arduino开发板,当使用Servo库时,开发板的9号和10号引脚的PWM功能是无法使用的,也就是说,Arduino程序如果使用了Servo库,无论9号和10号引脚上是否连接了舵机,这两个引脚都不能使用analogWrite进行控制。

(5)对于Mega开发板,使用12个以下的舵机是不会影响开发板的PWM功能的,但如果使用12个以上的舵机,那么Mega开发板上的引脚11和引脚12是要受到影响的。

3、Servo库提供的函数

(1)servo.attach(pin):告知Arduino舵机的数据线连接在引脚pin上。

(2)servo.write(angle):对于标准舵机,write函数会将舵机轴旋转到角度angle相应的位置;对于连续旋转类型的舵机,write函数可以设置舵机的旋转速度(0指示舵机向着一个方向全速旋转,180指示舵机向着另一个方向全速旋转,90指示舵机静止不动,以此类推)。

(3)servo.read():获取当前舵机轴的角度(即上一次使用write()函数设置舵机轴的角度信息)。

(4)servo.attached():检查某一个舵机对象是否连接在开发板引脚上,如果某一个舵机对象的确连接在开发板上则返回true,否则返回false。

(5)servo.deattach():将舵机对象与Arduino开发板断开连接,舵机一旦与Arduino开发板断开连接,则引脚9和引脚10的PWM输出(analogWrite)功能将会恢复。

4、例——舵机自动旋转

(1)根据上图所示将电路连接好。

(2)将下面的程序下载到开发板中,会发现舵机进行重复的来回旋转。值得注意的是,C++采用面向对象的编程思想,Servo库的编写亦是如此,所以可以为Arduino控制的舵机分别创建Servo对象,通过Servo对象调用Servo类的成员函数以达到让对象(舵机)执行相应操作的效果。

#include <Servo.h>  //包含Servo库的头文件
 
Servo myservo;  //创建Servo对象用以控制伺服电机
int pos = 0;    //存储伺服电机角度信息的变量
 
void setup() 
{
  myservo.attach(9);  //Servo对象(其控制的电机)连接在9号引脚 
  Serial.begin(9600);
}
 
void loop() 
{
  for (pos = 0; pos < 180; pos += 45)  //从0度转到180度,每一步旋转45度
  {
    myservo.write(pos);              //舵机旋转轴旋转到'pos'变量的角度
    Serial.println(pos);
    delay(1500);                     //等待1.5秒以确保伺服电机可以达到目标角度
  }
  for (pos = 180; pos > 0; pos -= 45)  //从180度转到0度,每一步旋转45度
  {
    myservo.write(pos);              //舵机旋转轴旋转到'pos'变量的角度
    Serial.println(pos);
    delay(1500);                     //等待1.5秒以确保伺服电机可以达到目标角度
  }
}

二、串口通讯

1、概述

(1)串行端口用于Arduino和个人电脑或其设备进行通信(若没有特殊说明,所提及的串口通信双方均为Arduino与个人电脑),所有Arduino控制器都有至少一个串行端口(也称为UART或者USART)。

(2)个人电脑可以通过USB端口与Arduino的引脚0RX和引脚1TX进行通信,所以当Arduino的引脚0和引脚1用于串行通信功能时,Arduino的引脚0和引脚1是不能做其它工作的。引脚RX和引脚TX工作时,开发板上会有相应的两盏指示灯分别闪烁。

①开发板上的RX和TX如下图所示。

②电脑端的RX和TX如下图所示。

③Arduino和计算机的连接如下图所示,计算机可通过TX向Arduino发送以字节为单位的数据,Arduino可借助RX接收计算机发送来的数据,同理,Arduino可通过TX向计算机发送以字节为单位的数据,计算机可借助RX接收Arduino发送来的数据

(3)串口通信中有一个概念是波特率,可认为是数据传输速率,通信双方的波特率应相同,否则无法正常通信。

(4)如果想深入了解串口通信的底层原理,可以参考课程《计算机网络》,虽然该课程不会详细介绍甚至不会介绍串口通信,但是通过对通信协议的学习,也有助于理解串口通信的原理。

2、例1——串行通讯基本演示

(1)本例仅用USB线将单片机和个人电脑连接即可。

(2)begin函数:

①该函数的作用是设置电脑与Arduino进行串口通讯时的数据传输速率(每秒传输字节数,也称波特率),可使用300、600、1200、2400、4800、9600、14400、19200、28800、38400、57600或115200,也可以根据所使用的设备而设置其它传输速率。

②使用格式为“Serial.begin(speed)”,其中参数speed为传输速率。

(3)available函数:

①当计算机给Arduino传输数据时,Arduino不一定能及时将该数据接收并处理,为了防止数据丢失,Arduino设置了串口接收缓存用于存放接收到的数据,当然,这个缓存不会很大,如果不及时将缓存中的数据读走,同样会造成数据丢失。另外,缓存中的字节数据是以ASCII码的形式存储的,比如字符’1’对应的ASCII码为49,转换为二进制就是00011001。

②available函数无函数参数,可用于检查设备是否接收到数据,该函数将会返回等待读取的数据字节数

③available函数属于Stream类,该函数可被Stream类的子类所使用(Stream类的子类继承了Stream类的函数),如Serial、WiFiClient、File等,本例使用的是Serial.available(),用于检查串口接收缓存中是否有未处理的数据。

(4)read函数:

①read函数无参数,可用于从设备接收到数据中读取一个字节的数据,设备没有接收到数据时返回值为-1,设备接收到数据时返回值为接收到的数据流中的1个字符(先接收先读取,后接收后读取)。

②本函数属于Stream类,该函数可被Stream类的子类所使用,如Serial、WiFiClient、File等,本例使用的是Serial.read(),用于读取串口接收缓存中的数据。

③一个字节数据被读取走后,其在串口接收缓存中随之被移除。

(5)print函数:

①该函数的作用是以人类可读的ASCII码形式向串口发送数据,该函数有多种格式。

[1]Serial.print(val):val是要发送的数据,可以是任意数据类型。

[2]Serial.print(val, format):val是要发送的数据,可以是任意数据类型;format是指定数字的基数(用于整型数)或者小数的位数(用于浮点数),对于整数,这个允许的值为BIN(二进制)、OCT (八进制)、DEC(十进制)或HEX(十六进制),对于浮点数,该参数可以指定小数点的位数。

②整数的每一数位将以ASCII码形式发送;浮点数同样以ASCII码形式发送,默认保留小数点后两位;字节型数据将以单个字符形式发送;字符和字符串会以其相应的形式发送。

③该函数的返回值为发送的字节数,一般来说可以摒弃,即这个返回值没什么实际用途。

(6)println函数:

①该函数的作用是以人类可读的ASCII码形式向串口发送数据,与print函数极其类似,不同的是该函数会在数据发送完毕后发送一个换行符

②该函数的返回值为发送的字节数,一般来说也可以摒弃,即这个返回值也没什么实际用途。

(7)将下面的程序下载到开发板中,然后开始人工调试。

void setup() 
{
  Serial.begin(9600);
  Serial.println("Please input serial data.");
}
 
void loop() 
{
  if (Serial.available() > 0 )    //检查串口缓存中是否有数据等待读取
  {
    Serial.println("======================");        
    char serialData = Serial.read();     //读取串口缓存中等待的一个字符
    Serial.print("Your serialData: '");
    Serial.print(serialData);            //发送刚刚读取到的字符给电脑
    Serial.println("'");
 
    Serial.print("serialData in DEC ");
    Serial.println(serialData, DEC);   //发送刚刚读取到字符的ASCII编码(十进制)
    Serial.print("serialData in BIN ");
    Serial.println(serialData, BIN);   //发送刚刚读取到字符的ASCII编码(二进制)
    Serial.print("serialData in OCT ");
    Serial.println(serialData, OCT);   //发送刚刚读取到字符的ASCII编码(八进制)
    Serial.print("serialData in HEX ");
    Serial.println(serialData, HEX);   //发送刚刚读取到字符的ASCII编码(十六进制)
    Serial.println("======================");
  }
}

①打开Arduino IDE的串口监视器,将内容“a”(结尾不带换行符)发送给Arduino,Arduino接收到数据后,Serial.available()的返回值为1,程序检测到缓存中有数据需要读取,然后进行读取操作,并将读取到的1字节数据以不同格式通过串口发送给电脑(同时附有提示信息)。

②打开Arduino IDE的串口监视器,将内容“bc”(结尾不带换行符)发送给Arduino,Arduino接收到数据后,Serial.available()的返回值为2,程序检测到缓存中有数据需要读取,然后进行读取操作,首先读取1字节数据——字符’b’并以不同格式通过串口发送给电脑(同时附有提示信息),再读取1字节数据——字符’c’并以不同格式通过串口发送给电脑(同时附有提示信息)。

3、例2——利用串行通讯控制一个舵机

(1)按照下图所示将电路连接好,其中Arduino、舵机和电源模块需要共地,电源模块选择5V的输出,另外需要注意电源模块的正负极不能短接。

(2)parseInt函数:

①parseInt函数没有参数,可用于从设备接收到的数据中寻找整数值并将其返回,返回值类型为长整型(long int),若找不到整数数据则返回0。

②本函数属于Stream类,可被Stream类的子类所使用,本例使用的是Serial.parseInt()。

(3)parseFloat函数:

①parseFloat函数没有参数,可用于从设备接收到的数据中寻找浮点数值并将其返回,返回值类型为浮点型(float)。

②本函数属于Stream类,可被Stream类的子类所使用,本例不使用该函数。

(4)将下面的程序下载到开发板中,然后开始人工调试(示例程序没有限定输入数值的取值范围,在实际应用中应用if语句对Arduino接收到的数值进行越界判断后再做处理)。

#include <Servo.h>
 
Servo myServo;         //为舵机创建Servo对象myServo
int dataIndex = 0;     //创建整数型变量,存储输入数据序列号

void setup() 
{
  myServo.attach(6);    //舵机连接在引脚6上
  Serial.begin(9600);   //启动串口通讯,配置传输波特率为9600
  Serial.println("Please input serial data.");
}
 
void loop() 
{           
  if ( Serial.available() > 0 ) // 检查串口缓存是否有数据等待传输
  {                             //处理数据序列号并通过串口监视器显示
    dataIndex++;  
    Serial.print("dataIndex = ");
    Serial.print(dataIndex);
    Serial.print(" , ");      
     
    int pos = Serial.parseInt();   //解析串口数据中的整数信息并赋值给变量pos
    Serial.print("Set servo position: ");
    Serial.println(pos);           //通过串口发送变量pos数值
    myServo.write(pos);            //使用pos变量数值设置伺服电机
    delay(15);
  }
}

①通过串口助手向Arduino发送内容“90”,程序检测到串口接收缓存有未处理数据后,调用parseInt函数对其中的数据进行处理,读取并分析出数据中的内容为整数90,接着Arduino通过串口将“90”发回给计算机,同时控制舵机旋转至90°的位置。

②通过串口助手向Arduino发送内容“*#AB45”,程序检测到串口接收缓存有未处理数据后,调用parseInt函数对其中的数据进行处理,读取并分析出数据中的内容含有整数45,接着Arduino通过串口将“45”发回给计算机,同时控制舵机旋转至45°的位置。

③通过串口助手向Arduino发送内容“*#AB135CV”,程序检测到串口接收缓存有未处理数据后,调用parseInt函数对其中的数据进行处理,读取并分析出数据中的内容含有整数135,“135”之前的字符全部省略,“135”之后的字符在本次parseInt函数的调用中不进行读取,这部分内容仍存储在缓存中,接着Arduino通过串口将“135”发回给计算机,同时控制舵机旋转至135°的位置,这时串口缓存中还有内容“CV”未处理,故程序再次调用parseInt函数对其中的数据进行处理,但在剩余内容“CV”中找不到整数数据,故返回0,Arduino通过串口将“0”发回给计算机,同时控制舵机旋转至0°的位置。

④通过串口助手向Arduino发送内容“*#AB90CV45”,程序检测到串口接收缓存有未处理数据后,调用parseInt函数对其中的数据进行处理,首先parseInt读取到整数数据“90”,“90”以后的数据不参与本次读取,Arduino控制舵机旋转至90°的位置,这时串口接收缓存中还有数据未处理,则程序再次调用parseInt函数,读取到整数数据“45”,Arduino控制舵机旋转至45°的位置。

4、例3——利用串行通讯控制四个舵机

(1)按照下图所示将电路连接好,其中Arduino、舵机和电源模块需要共地,电源模块选择5V的输出,另外需要注意电源模块的正负极不能短接。

(2)将下面的程序下载到开发板中,然后开始人工调试(示例程序没有限定输入数值的取值范围,在实际应用中应用if语句对Arduino接收到的数值进行越界判断后再做处理)。

#include <Servo.h>
 
Servo base, fArm, rArm, claw;  //建立4个电机对象

void setup() 
{
  base.attach(11);     //base伺服电机连接引脚11 电机代号为'b'
  rArm.attach(10);     //rArm伺服电机连接引脚10 电机代号为'r'
  fArm.attach(9);      //fArm伺服电机连接引脚9  电机代号为'f'
  claw.attach(6);      //claw伺服电机连接引脚6  电机代号为'c'
  Serial.begin(9600);
  Serial.println("Please input serial data.");
}
 
void loop() 
{
  if (Serial.available())   //检查串口缓存是否有数据等待传输
  { 
    char servoName = Serial.read();   //获取电机指令中电机编号信息
 
    Serial.print("servoName = ");
    Serial.print(servoName);    
    Serial.print(" , ");      
     
    int data = Serial.parseInt();   //获取电机指令中电机角度信息
 
    switch(servoName)  //根据电机指令中电机信息决定对哪一个电机进行角度设置
    {
      case 'b':   //电机编号为b,设置base电机的角度
        base.write(data); Serial.print("Set base servo value: ");
        Serial.println(data); break;
      case 'r':   //电机编号为r,设置rArm电机的角度  
        rArm.write(data); Serial.print("Set rArm servo value: ");
        Serial.println(data); break;
      case 'f':   //电机编号为f,设置fArm电机的角度  
        fArm.write(data); Serial.print("Set fArm servo value: ");
        Serial.println(data); break;
      case 'c':   //电机编号为c,设置claw电机的角度  
        claw.write(data); Serial.print("Set claw servo value: ");
        Serial.println(data); break;
    }
  }
}

①通过串口助手向Arduino发送内容“b45”,程序检测到缓存中有未处理数据,遂进行处理,首先获取第一个字节,其ASCII码对应的字符’b’为电机编号,接着从后续数据中获取到整数数据“45”,经过switch语句的逻辑控制将编号为“b”的电机控制旋转至45°。

②通过串口助手向Arduino发送内容“r90”,程序检测到缓存中有未处理数据,遂进行处理,首先获取第一个字节,其ASCII码对应的字符’r’为电机编号,接着从后续数据中获取到整数数据“90”,经过switch语句的逻辑控制将编号为“r”的电机控制旋转至90°。

③通过串口助手向Arduino发送内容“f135”,程序检测到缓存中有未处理数据,遂进行处理,首先获取第一个字节,其ASCII码对应的字符’f’为电机编号,接着从后续数据中获取到整数数据“135”,经过switch语句的逻辑控制将编号为“f”的电机控制旋转至135°。

④通过串口助手向Arduino发送内容“c180”,程序检测到缓存中有未处理数据,遂进行处理,首先获取第一个字节,其ASCII码对应的字符’c’为电机编号,接着从后续数据中获取到整数数据“180”,经过switch语句的逻辑控制将编号为“c”的电机控制旋转至180°。

⑤如果发送一个非法的电机编号(比如“a”),四个舵机均不会有任何响应,因为switch语句中没有实现对这种异常情况进行处理。

5、更多Serial(串行通信)函数

(1)Serial.end():终止串行通讯,让RX和TX引脚用于Arduino的输入(INPUT)或输出(OUTPUT)功能。(可调用Serial.begin()重新打开串行通讯)

(2)Serial.find(target):可用于从设备接收到的数据中寻找指定字符串信息,target为被查找字符串,允许使用String或char类型,当函数找到了指定字符串信息后将会立即结束函数执行并且返回“真”,否则将会返回“假”。

(3)Serial.findUntil(target, terminator):可用于从设备接收到的数据中寻找指定字符串信息,当函数找到了指定字符串信息后将会立即结束函数执行并且返回“真”,否则将会返回“假”。

①该函数在满足以下任一条件后都会停止函数执行:读取到指定终止字符串、找到了指定字符串信息、达到设定时间(可使用setTimeout来设置)。

②target是被查找字符串,允许使用String或char类型;terminator是终止字符串,用于设置终止函数执行的字符串信息,设备在读取数据时一旦读取到此终止字符串,将会结束函数执行并返回。

(4)Serial.flush():让开发板在所有待发数据发送完毕前,保持等待状态。

(5)Serial.peek():从设备接收到的数据中读取一个字节的数据,设备没有接收到数据时返回值为-1,设备接收到数据时返回值为接收到的数据流中的第1个字符;与read函数不同的是,使用peek函数读取数据后,被读取的数据不会从数据流中消除。

(6)Serial.readBytes(buffer, length):用于从设备接收的数据中读取信息,读取到的数据信息将存放在缓存变量中。

①该函数在读取到指定字节数的信息或者达到设定时间(该设定时间可使用setTimeout来设置)后都会停止函数执行并返回。

②buffer是缓存变量/数组,用于存储读取到的信息,允许使用char或者byte类型的变量或数组;length是读取字节的数量,readBytes函数在读取到length所指定的字节数量后就会停止运行,允许使用int类型。

(7)Serial.readBytesUntil(character, buffer, length):用于从设备接收到数据中读取信息,读取到的数据信息将存放在缓存变量中。

①该函数在满足以下任一条件后都会停止函数执行并且返回:读取到指定终止字符、读取到指定字节数的信息、达到设定时间(可使用setTimeout来设置)。

②当函数读取到终止字符后,会立即停止函数执行,此时buffer(缓存变量/数组)中所存储的信息为设备读取到终止字符前的字符内容。

③character是终止字符,用于设置终止函数执行的字符信息,允许使用char类型;buffer是缓存变量/数组,用于存储读取到的信息,允许使用char或者byte类型的变量或数组;length是读取字节数量,readBytes函数在读取到length所指定的字节数量后就会停止运行,允许使用int类型。

(8)Serial.write有两种形式:

①Serial.write(val):将字节数据val通过串口发送,val可以是一个字节数据,也可以是由一系列字节数据组成的字符串。

②Serial.write(buf, len):将同一系列字节组成的数组buf通过串口发送,len为数组长度。

(9)Serial.readString():用于从设备接收到数据中读取数据信息,读取到的信息将以字符串格式返回。

(10)Serial.readStringUntil(terminator):用于从设备接收到的数据中读取信息,读取到的数据信息将以字符串形式返回,该函数在读取到指定终止字符terminator或达到设定时间(可使用setTimeout来设置)时都会停止函数执行并返回。

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

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

相关文章

在windows下使用本地AI模型提供翻译、对话、文生图服务

文章目录 在windows下使用本地AI模型提供翻译、对话、文生图服务ollama简介下载安装配置环境变量模型安装目录服务监听地址跨域配置我的配置注意事项 开机自启 使用运行模型对话时的命令 查看本地已安装模型删除模型 查看ollama支持的模型 Docker Desktop简介下载安装配置开机自…

Linux驱动应用编程(四)IIC(获取BMP180温度/压力数据)

本文目录 一、基础1. 查看开发板手册&#xff0c;获取可用IIC总线2. 挂载从机&#xff0c;查看从机地址。3. 查看BMP180手册&#xff0c;使用命令读/写某寄存器值。4. 查看BMP180手册通信流程。 二、IIC常用API1. iic数据包/报2. ioctl函数 三、数据包如何被处理四、代码编写流…

(2024,Vision-LSTM,ViL,xLSTM,ViT,ViM,双向扫描)xLSTM 作为通用视觉骨干

Vision-LSTM: xLSTM as Generic Vision Backbone 公和众与号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0. 摘要 2 方法 3 实验 3.1 分类设计 4 结论 0. 摘要 Transformer 被广泛用作计算…

FM148R,FM147A和利时卡件

FM148R,FM147A和利时卡件。软件组成及各部分功能软件组成---各组件功能注意事项&#xff1a;仿真功能&#xff1a;仿真系统可以用于在单机上对组态完成的工程内容进行模拟运行。FM148R,FM147A和利时卡件。便于对这些组态内容的正确性和合理性进行初步调试。二、FM148R,FM147A和…

【Vue】面经基础版-案例效果分析

面经效果演示 功能分析 通过演示效果发现&#xff0c;主要的功能页面有两个&#xff0c;一个是列表页&#xff0c;一个是详情页&#xff0c;并且在列表页点击时可以跳转到详情页底部导航可以来回切换&#xff0c;并且切换时&#xff0c;只有上面的主题内容在动态渲染 实现思路…

C++基础四:C++模板编程

目录 一:函数模板 二:类模板 空间配置器allocator 一:函数模板 模板代码只能同一实现,不能先声明,再在另一文件实现,模板代码都是放在头文件当中的,在头文件中直接实现 二:类模板 template<typename T=int> class SeqStack // 模板名称+类型参数列表 = 类名称…

8.3 Go 包的组织结构

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」…

list模拟与实现(附源码)

文章目录 声明list的简单介绍list的简单使用list中sort效率测试list的简单模拟封装迭代器insert模拟erase模拟头插、尾插、头删、尾删模拟自定义类型迭代器遍历const迭代器clear和析构函数拷贝构造&#xff08;传统写法&#xff09;拷贝构造&#xff08;现代写法&#xff09; 源…

LabVIEW控制PLC的实现方式

LabVIEW与PLC的结合可以充分发挥两者的优点&#xff0c;实现更高效、灵活和可靠的自动化控制系统。本文将详细介绍LabVIEW控制PLC的实现方式&#xff0c;包括通信接口、数据交换、编程方法及实际应用案例&#xff0c;帮助用户理解并应用这一技术。 通信接口 常见通信协议 La…

Swift 序列(Sequence)排序面面俱到 - 从过去到现在(二)

概览 在上篇 Swift 序列(Sequence)排序面面俱到 - 从过去到现在(一)博文中,我们讨论了 Swift 语言中序列和集合元素排序的一些基本知识,我们还给出了以自定义类型中任意属性排序的“康庄大道”。 不过在实际的撸码场景中,我们往往需要的是“多属性”同时参与到排序的考…

Unity HDRP水系统

1.开启水系统 Script Interactions启用后&#xff0c;HDRP会为CPU上的水模拟分配内存。这允许您在水面上启用CPU模拟来查询高度信息。 2.Sky and Fog Volume中添加Water Rendering并勾选State 3.点击GameObject选择WaterSurface&#xff0c;选择要添加的水类型&#xff08;海…

【Vue】vuex 的使用 - 创建仓库

通用的地方我们一般会称之为仓库 1.安装 vuex 安装vuex与vue-router类似&#xff0c;vuex是一个独立存在的插件&#xff0c;如果脚手架初始化没有选 vuex&#xff0c;就需要额外安装。 yarn add vuex3 或者 npm i vuex32.新建 store/index.js 专门存放 vuex ​ 为了维护项目…

基于协调过滤算法商品推荐系统的设计#计算机毕业设计

基于协调过滤算法商品推荐系统的设计#计算机毕业设计 管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;商品管理&#xff0c;论坛管理&#xff0c;商品资讯管理 前台账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;论坛&#xf…

HIP的应用可移植性

Application portability with HIP — ROCm Blogs (amd.com) 许多科学应用程序在配备AMD的计算平台和超级计算机上运行&#xff0c;包括Frontier&#xff0c;这是世界上第一台Exascale系统。这些来自不同科学领域的应用程序通过使用Heterogeneous-compute Interface for Portab…

Electron qt开发教程

模块安装打包 npm install -g electron-forge electron-forge init my-project --templatevue npm start //进入目录启动 //打包成一个目录到out目录下&#xff0c;注意这种打包一般用于调试&#xff0c;并不是用于分发 npm run package //打出真正的分发包&#xff0c;放在o…

FJSP:烟花算法(FWA)求解柔性作业车间调度问题(FJSP),提供MATLAB代码

一、烟花算法介绍 参考文献&#xff1a; Tan, Y. and Y. Zhu. Fireworks Algorithm for Optimization. in Advances in Swarm Intelligence. 2010. Berlin, Heidelberg: Springer Berlin Heidelberg. 二、烟花算法求解FJSP 2.1FJSP模型介绍 柔性作业车间调度问题(Flexible …

在VMware虚拟机上安装win10 跳过 通过microsoft登录

在VMware虚拟机上安装win10 跳过 “通过microsoft登录” 配置虚拟机&#xff0c;将网卡断开&#xff0c; 具体操作&#xff1a; 虚拟机/设置/硬件/网络适配器/设备状态&#xff0c;取消已连接和启动时连接的两个对号&#xff0c; 再把虚拟机重启&#xff0c;然后就可以跳过这个…

Type-C转音频(C/3.5mm接口USB2.0数据传输)带PD充电低成本解决方案

LDR6500&#xff1a;领先市场的USB-C DRP接口USB PD通信芯片 产品介绍 LDR6500&#xff0c;由乐得瑞科技精心研发&#xff0c;是一款针对USB Type-C标准中Bridge设备而优化的USB-C DRP&#xff08;Dual Role Port&#xff0c;双角色端口&#xff09;接口USB PD&#xff08;Po…

【原创】springboot+mysql农业园区管理系统设计与实现

个人主页&#xff1a;程序猿小小杨 个人简介&#xff1a;从事开发多年&#xff0c;Java、Php、Python、前端开发均有涉猎 博客内容&#xff1a;Java项目实战、项目演示、技术分享 文末有作者名片&#xff0c;希望和大家一起共同进步&#xff0c;你只管努力&#xff0c;剩下的交…

Redis进阶知识个人汇总

持久化 三种方式实现它的持久化&#xff1a; RDB持久化 全称Redis数据备份文件&#xff0c;又称Redis数据快照 这种就是将Redis内存中所有数据记录到磁盘中&#xff0c;当实例出故障后&#xff0c;从磁盘中读快照文件进行恢复数据。 一般使用bgsave指令实现 复制主线程得到一…