我有一个Arduino项目,我从网络服务器读取数据.
我有一个EthernetClient,它在回调函数中逐个字符地读取数据.
我的工作代码看起来像(只有相关部分):
void setup() {
Serial.begin(9600);
...
}
void loop() {
char* processedData = processData(callback); // this is in a external lib
}
boolean callback(char* buffer, int& i) {
...
if (Client.available()) {
char c = client.read();
buffer[i++] = c;
Serial.print(c);
}
...
}
这没有任何问题(读取和处理数据),但当我删除Serial.begin(9600);和Serial.print(c);它停止工作,我不知道为什么?唯一改变的是char c没有打印.可能是什么问题呢?
最佳答案 当看似无关的代码被改变时,回调函数改变其行为的一个常见原因是优化器相关的错误.
许多嵌入式编译器无法理解将在程序中调用回调函数(或中断服务例程).他们看不到对该函数的显式调用,然后假设它从未被调用过.
当编译器做出这样的假设时,它将优化由回调函数改变的变量,因为它无法在初始化点和访问点之间看到变量被程序改变.
// Bad practice example:
int x;
void main (void)
{
x=5;
...
if(x == 0) /* this whole if statement will get optimized away,
the compiler assumes that x has never been changed. */
{
do_stuff();
}
}
void callback (void)
{
x = 0;
}
当这个bug出现时,它几乎不可能找到,它可能会导致任何奇怪的症状.
解决方案是始终将main()和中断/回调/线程之间共享的所有文件范围(“全局”)变量声明为volatile.这使得编译器无法做出不正确的优化器假设.
(请注意,volatile关键字不能用于实现同步,也不能保证任何内存障碍.这个问题与这些问题没有丝毫关系!)