-1

我有使用的课程std::stack

class NotificationService{
    public:
        void addPendingNotification(uint8_t *uuid);
        uint8_t* getNextPendingNotification();
        void popPending();
    private:
        std::stack<uint8_t*> pendingNotification;
};

void NotificationService::addPendingNotification(uint8_t *uuid) {
    pendingNotification.push(uuid);
    Serial.print("Insert to stack: ");
    Serial.print(uuid[0]);
    Serial.print(uuid[1]);
    Serial.print(uuid[2]);
    Serial.println(uuid[3]);
}

uint8_t *NotificationService::getNextPendingNotification() {
    if (pendingNotification.size() > 0) {
        uint8_t *uuid = pendingNotification.top();
        Serial.println(*uuid);
        pendingNotification.pop();
        return uuid;
    } else {
        return NULL;
    }
};

void NotificationService::popPending(){
    while (!pendingNotification.empty())
    {
        uint8_t *uuid = pendingNotification.top();
        Serial.print(uuid[0]);
        Serial.print(uuid[1]);
        Serial.print(uuid[2]);
        Serial.println(uuid[3]);
        pendingNotification.pop();
    }
}

我在主代码中添加到堆栈(BLE 通知回调):

static void NotificationSourceNotifyCallback(
    BLERemoteCharacteristic *pNotificationSourceCharacteristic,
    uint8_t *pData,
    size_t length,
    bool isNotify)
{
    if (pData[0] == 0)
    {
        uint8_t messageId[4] = {pData[4], pData[5], pData[6], pData[7]};
        switch (pData[2])
        {
            //Incoming Call
        case 1:
        {
            notificationService->addPendingNotification(messageId);
        }
/** code **/
}

一切正常,直到我想从堆栈中弹出项目,然后每个项目都具有相同的值(最后插入的元素)。

串行打印日志:

Insert to stack: 8000
Insert to stack: 32000
Insert to stack: 19000
Insert to stack: 44000
Insert to stack: 4000
Pop whole stack:
4000
4000
4000
4000
4000

所以我尝试在在线编译器中编写类似的代码:

http://cpp.sh/7hv4

它工作正常。

我究竟做错了什么?

4

2 回答 2

3
std::stack<uint8_t*> pendingNotification;

你有一堆指针。为此,您必须有一堆不同的对象供堆栈保存指针,并且只要您打算使用指针堆栈,这些对象就必须保持有效。您的代码不会这样做。

除非您有充分的理由,否则不要创建指针堆栈。相反,创建数据值堆栈。

于 2018-07-30T06:08:52.467 回答
0
notificationService->addPendingNotification(messageId);

将指向局部变量(messageId数组)的指针压入堆栈。

这个局部变量的作用域在稍后的某个时间点结束(只要封闭if块结束)。但是堆栈中的指针仍然存在。此时,取消引用此指针会调用未定义的行为,因此当您将指针从堆栈中弹出后,您得到的结果是未定义的行为。

在您的特定情况下,编译器似乎为所有实例重新使用了相同的内存位置messageId,这意味着在推送所有项目后,该内存位置仍然保存最后一个推送项目的值。但是要重新迭代:这是未定义的行为,您不应该依赖它。

相反,要么将值的副本推送到堆栈上,要么将堆栈上的指针推送到将在指针(堆栈)的生命周期内保持分配的内存。

于 2018-07-30T06:40:06.390 回答