2013-04-29 58 views
2

有人告诉我,(特别是在an answer to C++ Standard Library on Arduino,并在堆栈溢出问题C++ string and Arduino String. How to combine them?))的Arduino的编译器没有实现new操作。不过,我已经为Arduino(在Arduino IDE中)编写了一个使用它的程序,它完美地工作。运营商新的Arduino的

void setup() { 
    Serial.begin(9600); 
} 

void loop() { 
    char* array; 
    char c; 
    unsigned arraySize; 

    Serial.write("Enter a 1 digit number.\n"); 

    do { 
     c = Serial.read(); 
    } while(c < '0' or c > '9'); 
    arraySize = c-'0'; 

    Serial.write("You wrote "); 
    Serial.write(c); 
    Serial.write(".\n"); 
    Serial.write("Now enter "); 
    Serial.write(c); 
    Serial.write(" lower-case letters.\n"); 

    array = new char[arraySize]; 

    for (unsigned i = 0; i < arraySize;) { 
     array[i] = Serial.read(); 
     if (array[i] >= 'a' and array[i] <= 'z') 
      i++; 
    } 

    Serial.write("You entered: "); 

    for (unsigned i = 0; i < arraySize; i++) { 
     Serial.write(array[i]); 
     Serial.write(" "); 
    } 
    Serial.write("\n"); 
} 

下面是一个示例输出展示其功能:

Enter a 1 digit number. 
You wrote 5. 
Now enter 5 lower-case letters. 
You entered: h e l l o 
Enter a 1 digit number. 
You wrote 9. 
Now enter 9 lower-case letters. 
You entered: w a s s u p m a n 
Enter a 1 digit number. 
You wrote 9. 
Now enter 9 lower-case letters. 
You entered: h o w y a d o i n 
Enter a 1 digit number. 
You wrote 4. 
Now enter 4 lower-case letters. 
You entered: c o o l 
Enter a 1 digit number. 
You wrote 7. 
Now enter 7 lower-case letters. 
You entered: i t w o r k s 
Enter a 1 digit number. 

那么,为什么我不断听到这个?这些人是错的,还是我误解了他们的意思?

+2

正如他们在军队中所说的,如果地图与地形不匹配,请相信地形。 – 2013-04-29 12:25:27

回答

10

newdelete被定义为Arduino的发行版的一部分: /usr/share/arduino/hardware/arduino/cores/arduino/new.h

/* Header to define new/delete operators as they aren't provided by avr-gcc by default 
    Taken from http://www.avrfreaks.net/index.php?name=PNphpBB2&file=viewtopic&t=59453 
*/ 
#ifndef NEW_H 
#define NEW_H 

#include <stdlib.h> 

void * operator new(size_t size); 
void operator delete(void * ptr); 

__extension__ typedef int __guard __attribute__((mode (__DI__))); 

extern "C" int __cxa_guard_acquire(__guard *); 
extern "C" void __cxa_guard_release (__guard *); 
extern "C" void __cxa_guard_abort (__guard *); 

extern "C" void __cxa_pure_virtual(void); 

#endif 

new.h包括在Printable.h所以你得到它在你的Arduino基本具备。尽管这些运算符未在AVR Libc中定义。我对这些设计选择的解释是:Libc人们认为这是一个糟糕的主意,而Arduino人员都是关于易用性的:如果你想要newdelete,请拥有它们。

1

仅仅因为平台没有预先定义operator new()并不意味着无法定义它。在你的代码中,大概有人会为分配函数写一个定义,所以一切都很好。

(我假设你熟悉new表达和分配功能,这是容易混淆的所谓operator new()之间的差额。)


这里有一个婴儿例子定义,应该得到一个简短的程序去:

char buf[1024]; 
char * cur = buf; 

void * operator new(std::size_t n) 
{ 
    char * res = cur; 
    std::size_t inc = (n + 15)/16 * 16; 

    if (std::distance(cur, buf + sizeof(buf)) < inc) 
     throw std::bad_alloc(); 

    cur += inc; 

    return res; 
} 

void operator delete(void * p) noexcept 
{ 
} 

这显然会非常快地耗尽内存。

+0

但我没有定义它。而且我没有使用任何特殊的库。因此,如果所有形式的'operator new'都具有定义,那么与Arduino平台上的'new'是否“实现”不同?当他们说Arduino不支持'new'或'delete'时,这些人是什么意思?显然他们工作,那么问题是什么? – anthropomorphic 2013-04-29 08:53:26

+0

我喜欢这被称为“宝贝的例子。“ – 2016-05-29 06:04:23

+0

这个例子有两个问题:1:试图在嵌入式系统上使用异常,2:假设Arduino有一个'std'命名空间(它不会) – Pharap 2017-08-20 07:23:28

9

正如你已经证明,newdelete做Arduino的原则工作。问题是你的Arduino有2K的RAM用于全局变量,堆栈和堆合并,所以你为它编写的任何实质程序都必须高度关注内存使用情况。

因此,对于像这样的嵌入式系统,通常最好使用静态分配(或者在某些情况下可能是按类动态分配),而不是通用动态分配:对于简单的程序,您不需要需要动态分配,对于复杂的程序,你不能负担得起它。