char *key = array[i].key; char *value = array[i].value; // 如果值是字母或数字,则直接拼接 if(isalpha(value[0]) && isalnum(value[1]) && strcmp(value, "true")!=0 && strcmp(value, "false")!=0) { sprintf(&query_list[len], "%s=%s&", key, value); } else { // 否则需要将值加上双引号再拼接 sprintf(&query_list[len], "\"%s\""="%s"&", key, value); }上面的代码有什么问题
时间: 2024-03-20 18:41:44 浏览: 17
上面的代码中,双引号位置不正确,会导致语法错误。正确的代码应该是:
```c
char *key = array[i].key;
char *value = array[i].value;
// 如果值是字母或数字,则直接拼接
if (isalpha(value[0]) && isalnum(value[1]) && strcmp(value, "true") != 0 && strcmp(value, "false") != 0) {
sprintf(&query_list[len], "%s=%s&", key, value);
} else { // 否则需要将值加上双引号再拼接
sprintf(&query_list[len], "%s=\"%s\"&", key, value);
}
```
在拼接时,需要注意双引号的位置。正确的位置是在 `%s` 前后添加,而等号 `=` 不需要放在双引号内。同时,由于 `%s` 和 `&` 都是字符串,需要将它们用双引号括起来。
相关问题
#include<stdio.h> #include<string.h> #include<stdlib.h> #include<ctype.h> #include<openssl/hmac.h> char *signature_calculate(char *json_obj, char *key){ unsigned char *key_byte = (unsigned char *)key; char *sorted_json = to_url_query(json_obj); unsigned char *dataddd = (unsigned char *)sorted_json; unsigned char *signature = HMAC(EVP_sha256(), key_byte, strlen(key), dataddd, strlen(dataddd), NULL, NULL); char hex_signature = malloc(2 * EVP_MAX_MD_SIZE + 1); for(int i=0; i<EVP_MAX_MD_SIZE; i++) { sprintf(&hex_signature[i2], "%02x", signature[i]); } return hex_signature; } typedef struct { char key[256]; char value[256]; } KeyValue; int compare(const void a, const void b) { return strcmp(((KeyValue)a)->key, ((KeyValue)b)->key); } char *sort_dict(KeyValue *array, int size) { // 对KeyValue数组按ASCII码升序排序 qsort(array, size, sizeof(KeyValue), compare); char *query_list = malloc(size * 256); int len=0; for(int i=0; i<size; i++) { if(strlen(array[i].value)==0){ // 如果值为空或者空字符串则不拼接 continue; } char *key = array[i].key; char *value = array[i].value; if(isalpha(value[0]) && isalnum(value[1]) && strcmp(value, "true")!=0 && strcmp(value, "false")!=0) { sprintf(&query_list[len], "%s=%s&", key, value); } else { sprintf(&query_list[len], "%s="%s"&", key, value); } len = strlen(query_list); } if(len>0) { query_list[len-1] = 0; } return query_list; } char *to_url_query(char *json, char *prefix){ // 将json字符串转换为URL键值对形式的字符串 int len = strlen(json); KeyValue *array = malloc(len * sizeof(KeyValue)); int i=0; int j=0; int level=0; char *key; // 处理嵌套字典的键名 while(i<len){ if(json[i]=='{' || json[i]=='['){ level++; i++; } else if(json[i]=='}' || json[i]==']'){ level--; i++; } else if(json[i]==','){ array[j].value[i-array[j].key] = 0; i++; j++; } else if(json[i]==':'){ key = array[j].key; array[j].value[0] = 0; i++; } else if(json[i]=='"' && level%2==0){ i++; int k=0; while(json[i]!='"') { array[j].value[k] = json[i]; i++; k++; } array[j].value[k] = 0; i++; } else if(json[i]!=',' && json[i]!=':' && json[i]!=' '){ array[j].key[i-j] = json[i]; i++; } else { i++; } } array[j].value[i-array[j].key] = 0; j++; char *sorted_json = sort_dict(array, j); char *query_list = malloc(strlen(sorted_json)+1); if(strlen(prefix)>0){ sprintf(query_list, "%s.%s", prefix, sorted_json); } else { strcpy(query_list, sorted_json); } free(array); free(sorted_json); return query_list; } 请对上面的代码添加注释
```
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<ctype.h>
#include<openssl/hmac.h>
// 计算签名函数,接收一个JSON字符串和一个密钥字符串,返回一个签名字符串
char *signature_calculate(char *json_obj, char *key){
// 将密钥转换为unsigned char类型的指针
unsigned char *key_byte = (unsigned char *)key;
// 将JSON字符串转换为URL键值对形式的字符串
char *sorted_json = to_url_query(json_obj);
// 将URL键值对形式的字符串转换为unsigned char类型的指针
unsigned char *dataddd = (unsigned char *)sorted_json;
// 使用OpenSSL库的HMAC函数计算签名
unsigned char *signature = HMAC(EVP_sha256(), key_byte, strlen(key), dataddd, strlen(dataddd), NULL, NULL);
// 将签名字符串转换为十六进制形式的字符串并返回
char hex_signature = malloc(2 * EVP_MAX_MD_SIZE + 1);
for(int i=0; i<EVP_MAX_MD_SIZE; i++) {
sprintf(&hex_signature[i2], "%02x", signature[i]);
}
return hex_signature;
}
// 定义一个键值对结构体类型,用于存储键值对
typedef struct {
char key[256];
char value[256];
} KeyValue;
// 对KeyValue数组按ASCII码升序排序的比较函数
int compare(const void a, const void b) {
return strcmp(((KeyValue)a)->key, ((KeyValue)b)->key);
}
// 将KeyValue数组按ASCII码升序排序并拼接成URL键值对形式的字符串
char *sort_dict(KeyValue *array, int size) {
// 对KeyValue数组按ASCII码升序排序
qsort(array, size, sizeof(KeyValue), compare);
// 初始化一个字符串,用于存储拼接后的URL键值对形式的字符串
char *query_list = malloc(size * 256);
int len=0;
for(int i=0; i<size; i++) {
// 如果值为空或者空字符串则不拼接
if(strlen(array[i].value)==0){
continue;
}
char *key = array[i].key;
char *value = array[i].value;
// 如果值是字母或数字,则直接拼接
if(isalpha(value[0]) && isalnum(value[1]) && strcmp(value, "true")!=0 && strcmp(value, "false")!=0) {
sprintf(&query_list[len], "%s=%s&", key, value);
} else { // 否则需要将值加上双引号再拼接
sprintf(&query_list[len], "%s="%s"&", key, value);
}
len = strlen(query_list);
}
// 去掉最后一个&符号
if(len>0) {
query_list[len-1] = 0;
}
return query_list;
}
// 将JSON字符串转换为URL键值对形式的字符串
char *to_url_query(char *json, char *prefix){
// 计算JSON字符串的长度
int len = strlen(json);
// 初始化一个KeyValue数组,用于存储键值对
KeyValue *array = malloc(len * sizeof(KeyValue));
int i=0;
int j=0;
int level=0;
char *key;
// 处理嵌套字典的键名
while(i<len){
if(json[i]=='{' || json[i]=='['){ // 如果遇到{或[,则进入下一层级
level++;
i++;
} else if(json[i]=='}' || json[i]==']'){ // 如果遇到}或],则退出上一层级
level--;
i++;
} else if(json[i]==','){ // 如果遇到逗号,则处理下一个键值对
array[j].value[i-array[j].key] = 0;
i++;
j++;
} else if(json[i]==':'){ // 如果遇到冒号,则表示下一个字符串是值,保存键名
key = array[j].key;
array[j].value[0] = 0;
i++;
} else if(json[i]=='"' && level%2==0){ // 如果遇到双引号,则表示下一个字符串是键名或值
i++;
int k=0;
while(json[i]!='"') { // 保存双引号之间的字符串
array[j].value[k] = json[i];
i++;
k++;
}
array[j].value[k] = 0;
i++;
} else if(json[i]!=',' && json[i]!=':' && json[i]!=' '){ // 如果不是逗号、冒号或空格,则表示当前字符是键名的一部分
array[j].key[i-j] = json[i];
i++;
} else { // 否则忽略当前字符
i++;
}
}
// 保存最后一个键值对的值
array[j].value[i-array[j].key] = 0;
j++;
// 将KeyValue数组按ASCII码升序排序并拼接成URL键值对形式的字符串
char *sorted_json = sort_dict(array, j);
// 初始化一个字符串,用于存储拼接后的URL键值对形式的字符串
char *query_list = malloc(strlen(sorted_json)+1);
if(strlen(prefix)>0){ // 如果有前缀,则在拼接后的字符串前加上前缀
sprintf(query_list, "%s.%s", prefix, sorted_json);
} else { // 否则直接返回拼接后的字符串
strcpy(query_list, sorted_json);
}
// 释放内存
free(array);
free(sorted_json);
return query_list;
}
```
解释这段代码 typedef struct items_val_t { int items_number; char **items_value; } items_val_t; union union_val { items_val_t array; /*!< array类型 */ char *string; /*!< string类型 */ double number; /*!< number类型 */ int bv; /*!< boolean类型 */ }; typedef struct desc_kv_t { const char *key; /*!< key */ union union_val rval; /*!< request value 通过json请求的值 */ union union_val dval; /*!< 默认值 */ int (*validate)(desc_kv_t *desc_kv); /*!< 检验请求参数的合法性 */ } desc_kv_t;
这段代码定义了三个结构体类型:`items_val_t`、`union_val` 和 `desc_kv_t`。
`items_val_t` 结构体类型定义了一个变量,包括整数变量 `items_number` 和字符串指针数组 `items_value`。
`union_val` 结构体类型定义了一个联合体变量,其中可以储存四种不同的类型:数组类型,字符串类型,浮点数类型,布尔类型。
最后,`desc_kv_t` 结构体类型定义了一个变量,包括字符串常量 `key`,两个联合体变量:请求值 `rval` 和默认值 `dval`,以及一个函数指针 `validate`,用于检验请求参数的合法性。
相关推荐
![pdf](https://img-home.csdnimg.cn/images/20210720083512.png)
![docx](https://img-home.csdnimg.cn/images/20210720083331.png)
![pdf](https://img-home.csdnimg.cn/images/20210720083512.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![](https://csdnimg.cn/download_wenku/file_type_ask_c1.png)
![doc](https://img-home.csdnimg.cn/images/20210720083327.png)