smartknob状态机
358 2024-09-07 08:53
用状态机优化下面代码
if (reading >= lower - (upper - lower) && reading < upper + (upper - lower) * 2)
{
long value = CLAMP(reading, lower, upper);
press_value_unit = 1. * (value - lower) / (upper - lower);
static bool pressed;
static bool accumulatable;
static uint8_t press_count;
static uint8_t release_count;
if (!pressed && accumulatable && press_value_unit > 0.75)
{
accumulatable = false;
// snprintf(buf_, sizeof(buf_), "accumulatable: %d", accumulatable);
// log(buf_);
}
else if (!pressed && !accumulatable && press_value_unit > 0.75)
{
pressed = true;
accumulatable = true;
motor_task_.playHaptic(true);
// snprintf(buf_, sizeof(buf_), "pressed: %d \t accumulatable: %d", pressed, accumulatable);
// log(buf_);
}
else if (pressed && accumulatable && press_value_unit > 0.75)
{
press_count++;
// snprintf(buf_, sizeof(buf_), "press_count: %d", press_count);
// log(buf_);
}
else if (pressed && accumulatable && press_value_unit < 0.25)
{
accumulatable = false;
}
else if (pressed && !accumulatable && press_value_unit < 0.25)
{
release_count++;
// snprintf(buf_, sizeof(buf_), "release_count: %d", release_count);
// log(buf_);
if (press_count > 30 && release_count > 2)
{
changeConfigToZero();
press_count = 0;
// snprintf(buf_, sizeof(buf_), "配置0");
// log(buf_);
motor_task_.playHaptic(false);
pressed = false;
}
else if (press_count > 2 && release_count > 2)
{
changeConfig(true);
press_count = 0;
// snprintf(buf_, sizeof(buf_), "下一个配置");
// log(buf_);
motor_task_.playHaptic(false);
pressed = false;
}
}
else
{
release_count = 0;
}
}
为了优化这段代码,我们可以使用一个状态机来管理不同的状态和事件。状态机可以帮助我们更清晰地管理状态转换,并且使代码更加模块化和易于维护。下面是一个简化的状态机实现的示例:
首先,我们定义状态和事件:
```c
enum State {
STATE_IDLE,
STATE_PRESSED,
STATE_ACCUMULATING,
STATE_RELEASED
};
enum Event {
EVENT_PRESS,
EVENT_ACCUMULATE,
EVENT_RELEASE,
EVENT_IDLE
};
```
然后,我们定义一个结构体来保存状态机的状态:
```c
struct StateMachine {
State state;
bool accumulatable;
uint8_t press_count;
uint8_t release_count;
};
```
接下来,我们定义状态转换函数:
```c
void handleEvent(StateMachine *sm, Event event, double press_value_unit, long lower, long upper) {
long value = CLAMP(reading, lower, upper);
double press_value_unit = 1. * (value - lower) / (upper - lower);
switch (sm->state) {
case STATE_IDLE:
if (press_value_unit > 0.75) {
sm->state = STATE_PRESSED;
sm->accumulatable = true;
// log("Pressed");
}
break;
case STATE_PRESSED:
if (press_value_unit > 0.75) {
if (sm->accumulatable) {
sm->state = STATE_ACCUMULATING;
sm->press_count++;
// log("Accumulating");
}
} else if (press_value_unit < 0.25) {
sm->state = STATE_RELEASED;
sm->release_count++;
// log("Released");
}
break;
case STATE_ACCUMULATING:
if (press_value_unit > 0.75) {
sm->press_count++;
// log("Press count: ", sm->press_count);
} else if (press_value_unit < 0.25) {
sm->state = STATE_RELEASED;
sm->accumulatable = false;
// log("Released");
}
break;
case STATE_RELEASED:
if (press_value_unit < 0.25) {
if (sm->press_count > 30 && sm->release_count > 2) {
changeConfigToZero();
sm->press_count = 0;
// log("Config 0");
motor_task_.playHaptic(false);
} else if (sm->press_count > 2 && sm->release_count > 2) {
changeConfig(true);
sm->press_count = 0;
// log("Next config");
motor_task_.playHaptic(false);
}
sm->state = STATE_IDLE;
} else {
sm->release_count = 0;
}
break;
}
}
```
最后,我们在主循环中使用状态机:
```c
StateMachine sm = {STATE_IDLE, true, 0, 0};
while (true) {
// 假设 reading 是从传感器读取的值
double reading = getSensorReading();
if (reading >= lower - (upper - lower) && reading < upper + (upper - lower) * 2) {
handleEvent(&sm, EVENT_PRESS, reading, lower, upper);
} else {
handleEvent(&sm, EVENT_IDLE, 0, lower, upper);
}
// 其他循环逻辑...
}
```
这个状态机的实现将原始的 if-else 逻辑分解为不同的状态和事件处理函数,使得代码更加清晰和易于理解。每个状态和事件都有明确的定义,这有助于未来的维护和扩展。
全部评论