从BLE收到的数据无法从十六进制转换为字符串(北欧)

时间:2017-11-05 04:39:37

标签: c string hex bluetooth-lowenergy nrf51

在ble_app_template的on_ble_evt(ble_evt_t *)函数中,我添加了一个'BLE_GATTS_EVT_WRITE'的情况。它包含以下代码:

    case BLE_GATTS_EVT_WRITE:
    { 
        ble_gatts_evt_write_t * p_evt_write = &p_ble_evt->evt.gatts_evt.params.write;
        SEGGER_RTT_printf(0, "Length = %d \n", p_evt_write->len);
        int n;
        char* buf2 = "";    
        char* buf3 = "";    
        count += p_evt_write->len;
        for(n=0; n<p_evt_write->len; n++)
        {
                   SEGGER_RTT_printf(0, "Received[%d] : %X \n", n, p_evt_write->data[n]);
                   SEGGER_RTT_printf(0, "Received[%d] : %d \n", n, p_evt_write->data[n]);
                   if(n>0){                                                     
                       sprintf(buf2, "%s", ((char*)p_evt_write->data[n]));    
                       strcat(buf3, buf2);
                   }                                                    
                   else{                                                
                       sprintf(buf2, "%s", ((char*)p_evt_write->data[n]));   
                       strcpy(buf3, buf2);
                   }
                   SEGGER_RTT_printf(0, "buf2 string: %s \n", buf2);
                   SEGGER_RTT_printf(0, "buf2 hex: %X \n", buf2[0]);
                   SEGGER_RTT_printf(0, "buf3 in string: %s \n", buf3);
                   SEGGER_RTT_printf(0, "count: %d \n", count);
        }
   }

我收到存储在p_evt_write-&gt; data [n]中的BLE的十六进制值。我想将所有这些接收到的十六进制值连接成一个字符串,并将其存储在'buf2'中。

但是我在sprintf行收到错误。当我在sprintf中输入“%X”时,值不会转换为字符串,因为buf2 / buf3字符串不会打印任何内容。

的当前代码
sprintf(buf2, "%s", ((char*)p_evt_write->data[n]));

返回错误'error:从不同大小的整数强制转换为指针[-Werror = int-to-pointer-cast]'。如果我删除了(char *)类型转换,则返回错误'error:format'%s'需要类型为'char *''的参数。

我看到了p_evt_write-&gt; data [n],其中数据被声明为

uint8_t                     data[1];

我目前正在使用Nordic的nRF51DK和Eclipse IDE以及SEGGER RTT JLink进行调试。所有打印都打印到SEGGER RTT。 (printf不起作用,使用SEGGER_RTT_printf代替打印)

如何将所有十六进制值成功连接在一起形成一个字符串?谢谢。

2 个答案:

答案 0 :(得分:1)

要附加到缓冲区,您可以使用以下内容:

size_t total_len = 0;
uint8_t buf = NULL;

while (receive()) {
    size_t len = p_evt_write->len;
    buf = realloc(buf, total_len + len);
    memcpy(buf + total_len, p_evt_write->data, len);
    total_len += len;
}

要将uint8_t数组转换为以NUL结尾的字符串,请分配足够的空间,复制字节并附加NUL。

char* s = malloc(total_len + 1);
memcpy(s, buf, total_len);
s[total_len] = 0;

当然,如果buf的任何元素是NUL(0),您将看起来有一个截断的字符串。没有解决这个问题(没有摆脱NULs)。

请注意,->data不包含&#34;十六进制值&#34;。十六进制是数字的文本表示。你没有十六进制;你有数字。这太棒了,因为这意味着不需要转换。您只需复制字节并添加NUL。

请注意,您获得的具体错误是因为您将uint8_t(一个数字)转换为char*(指针)。这是没有意义的。您可能会选择(char*)(&(p_evt_write->data[n])),但这不会起作用,因为%s并不只是指向一个指向一个字符的指针,而是指向一个以该字符结尾的序列的第一个字符的指针一个NUL角色。

答案 1 :(得分:0)

假设您想要获取所收到数据的十六进制表示字符串,请尝试这样的事情......

    ble_gatts_evt_write_t * p_evt_write = &p_ble_evt->evt.gatts_evt.params.write;
    char* buf2 = malloc((p_evt_write->len * 2) + 1);
    memset(buf2, 0, (p_evt_write->len * 2) + 1);
    for(int n=0; n<p_evt_write->len; n++)
    {
        char buf3[10];
        sprintf(buf3, "%02X", (unsigned int)p_evt_write->data[n]);
        strcat(buf2, buf3);
    }
    SEGGER_RTT_printf(0, "buf2 string: %s \n", buf2);
    free(buf2);