将非线性模拟值减少为数字输出

时间:2019-01-03 23:09:13

标签: c adc digital

我制作了一个键盘,可以给我一个模拟值。现在,我想根据电阻来检测按下了哪个键。但是,这些值是非线性的。

我使用很多“ if / else-es”来做到这一点。有更好的方法吗?

if(analogValue < 159){
  keyPress(KEY_A_ACUTE);
}else if(analogValue < 400){
  keyPress(KEY_I_ACUTE);
}else if(analogValue < 537){
  keyPress(KEY_O_ACUTE);
}else if(analogValue < 624){
  keyPress(KEY_U_ACUTE);
}else if(analogValue < 685){
  keyPress(KEY_N_TILDE);
}else if(analogValue < 800){
  keyPress(KEY_E_ACUTE);
}

1 个答案:

答案 0 :(得分:2)

您可以创建结构类型的数组并使用循环:

struct KeyValue
{
    int    key_name;
    int    max_value;  // Strictly less than threshold
};

static const struct KeyValue keys[] =
{
    { KEY_A_ACUTE, 159 },
    { KEY_I_ACUTE, 400 },
    { KEY_O_ACUTE, 537 },
    { KEY_U_ACUTE, 624 },
    { KEY_N_TILDE, 685 },
    { KEY_E_ACUTE, 800 },
};
enum { NUM_KEYS = sizeof(keys) / sizeof(key[0]) };

int analague_to_key_name(int analogue_value)
{
    for (int i = 0; i < NUM_KEYS; i++)
    {
         if (analogue_value < keys[i].max_value)
             return keys[i].key_name;
    }
    return KEY_UNKNOWN;
}

唯一的陷阱是处理超出范围的值-我选择返回一个特殊的KEY_UNKNOWN值,但是您可以使用任何适当的替代策略。当然,数据必须按max_value的升序排列。

该技术的一个优点是它不受表添加的影响,并且可以处理阈值之间的任何任意差异。如果表足够大(也许有几十个条目),则可以切换到对该值进行专门的二进制搜索,从而进行较少的比较。二进制搜索是专用的,因为您需要接受一个范围。您可能需要使用min_valuemax_value创建结构。