class xxxx {
int cnt;
void fun();
}
	
	然后我想
attachInterrupt(PIN11, fun, CHANGE);
然后在
void fun()
{
   cnt++;
}
	
	实现类似这样效果
发现只能绕路实现,直接实现不了的
 
					
				
	程序写好了
 
	
#define LW_PWM 3
#define LW_IN1 2
#define LW_IN2 4
#define LW_ECA A1
#define LW_ECB A0
#define RW_PWM 5
#define RW_IN1 7
#define RW_IN2 6
#define RW_ECA A3
#define RW_ECB A2
#define CW 0
#define CCW 1
#include "balance_car.hpp"
void setup() {
  Serial.begin(9600);
  
  BL_CAR.setup_motor(balance_car::LW, LW_PWM, LW_IN1, LW_IN2, LW_ECA, LW_ECB);
  BL_CAR.setup_motor(balance_car::RW, RW_PWM, RW_IN1, RW_IN2, RW_ECA, RW_ECB);
}
void loop() {
  delay(2000);
  BL_CAR.show_pos();
  BL_CAR.run(80, CW, 80, CW);
  delay(2000);
  BL_CAR.show_pos();
  BL_CAR.run(80, CCW, 80, CCW);
  delay(2000);  
  BL_CAR.show_pos();
}
	调换了一下6,7口,发现接线不对
程序功能为了测试连线和编码器
	左右轮顺时针转一会,显示编码器值
再逆时针转一会,显示编码器值
	
经测试,连线没问题,编码器正常工作
 
					
				
	http://playground.arduino.cc/Main/RotaryEncoders
主要理论如下图:
	
	
代码参考网站的例子:
/* read a rotary encoder with interrupts
   Encoder hooked up with common to GROUND,
   encoder0PinA to pin 2, encoder0PinB to pin 4 (or pin 3 see below)
   it doesn't matter which encoder pin you use for A or B  
   uses Arduino pullups on A & B channel outputs
   turning on the pullups saves having to hook up resistors 
   to the A & B channel outputs 
*/ 
#define encoder0PinA  2
#define encoder0PinB  4
volatile unsigned int encoder0Pos = 0;
void setup() { 
  pinMode(encoder0PinA, INPUT); 
  digitalWrite(encoder0PinA, HIGH);       // turn on pullup resistor
  pinMode(encoder0PinB, INPUT); 
  digitalWrite(encoder0PinB, HIGH);       // turn on pullup resistor
  attachInterrupt(0, doEncoder, CHANGE);  // encoder pin on interrupt 0 - pin 2
  Serial.begin (9600);
  Serial.println("start");                // a personal quirk
} 
void loop(){
// do some stuff here - the joy of interrupts is that they take care of themselves
}
void doEncoder() {
  /* If pinA and pinB are both high or both low, it is spinning
   * forward. If they're different, it's going backward.
   *
   * For more information on speeding up this process, see
   * [Reference/PortManipulation], specifically the PIND register.
   */
  if (digitalRead(encoder0PinA) == digitalRead(encoder0PinB)) {
    encoder0Pos++;
  } else {
    encoder0Pos--;
  }
  Serial.println (encoder0Pos, DEC);
}
	偷偷告诉你,例子有bug,你能发现不?
 
					
				读取CurieIMU的加速度计和陀螺仪数据
	
参考API:
https://www.arduino.cc/en/Reference/CurieIMUreadMotionSensor
	
Description
Reads the raw values of the motion sensor (accelerometer + gyro).
SyntaxCurieImu.readMotionSensor(int ax, int ay, int az, int gx, int gy, int gz)
Parameters
		ax: a variable in which the accelerometer's value along x will be stored.
ay: a variable in which the accelerometer's value along y will be stored.
az: a variable in which the accelerometer's value along z will be stored.
gx: a variable in which the gyro's value along x will be stored.
gy: a variable in which the gyro's value along y will be stored.
gz: a variable in which the gyro's value along z will be stored.
	
以上数据为原始数据。
		
	
转换方法:
		To convert the raw value into mg use the following formula: 
float g = (gRaw/32768.0)*getAccelerometerRange() 
where gRaw is either ax, ay or az.
	
来自:https://www.arduino.cc/en/Reference/CurieIMUreadAccelerometer
		
	
		To convert any of the raw values in angular velocity (°/s) use the following formula: 
float av = ( avRaw/32768.9)*getGyroRange() 
where avRaw is either gx, gy or gz.
	
来自:https://www.arduino.cc/en/Reference/CurieIMUreadGyro
		
	
有关自动校正:
		https://www.arduino.cc/en/Reference/CurieIMUautoCalibrateAccelerometerOffset
https://www.arduino.cc/en/Reference/CurieIMUreadGyro
	
 
					
				float conv_g(int gRaw) {
  // 转换成标准重力加速度
  return (gRaw/32768.0)*CurieIMU.getAccelerometerRange() ;
}
float conv_av (int avRaw) {
  return (avRaw/32768.0)*CurieIMU.getAccelerometerRange() ;
}
void balance_car::showIMU()
{
   readIMU();
   char buf[100];
   sprintf(buf, "IMU data: %d, %d, %d, %d, %d, %d", _ax, _ay, _az, _gx, _gy, _gz);
   Serial.println(buf);
   sprintf(buf, "IMU data: %f, %f, %f, %f, %f, %f", conv_g(_ax), conv_g(_ay), conv_g(_az), conv_av(_gx), conv_av(_gy), conv_av(_gz));
   Serial.println(buf);
}
	
部分测试代码如上:
	
小车直立放置后,输出结果如下:
	LW:0    RW:0
LW:0    RW:0
IMU data: 4, -25, 16301, -15, -4, -10
IMU data: 0.000244, -0.001526, 0.994934, -0.000916, -0.000244, -0.000610
LW:0    RW:0
LW:0    RW:0
LW:0    RW:0
IMU data: -4, 17, 16296, -4, -7, 7
IMU data: -0.000244, 0.001038, 0.994629, -0.000244, -0.000427, 0.000427
LW:0    RW:0
LW:0    RW:0
LW:0    RW:0
IMU data: 34, -8, 16347, -6, 0, 0
IMU data: 0.002075, -0.000488, 0.997742, -0.000366, 0.000000, 0.000000
 
					
				模仿别人写了个角度计算的函数
	
float balance_car::Angle_Calcu(void)         
{
    readIMU();
    float angleA= atan(conv_g(_ay)/ conv_g(_az))* (180)/ 3.1415926;    
    return angleA;                                                                                                   
}  
	然后loop中输出角度
	
这个角度基本上准确
	用串口绘图软件看,虽然角度基本正确,但是运动的时候,图像有锯齿
网上说要滤波,回头研究一下咋滤波呢
| 有奖活动 | |
|---|---|
| 硬核工程师专属补给计划——填盲盒 | |
| “我踩过的那些坑”主题活动——第002期 | |
| 【EEPW电子工程师创研计划】技术变现通道已开启~ | |
| 发原创文章 【每月瓜分千元赏金 凭实力攒钱买好礼~】 | |
| 【EEPW在线】E起听工程师的声音! | |
| 高校联络员开始招募啦!有惊喜!! | |
| 【工程师专属福利】每天30秒,积分轻松拿!EEPW宠粉打卡计划启动! | |
| 送您一块开发板,2025年“我要开发板活动”又开始了! | |