【C 语言】字符串模型 ( 键值对模型 )

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 【C 语言】字符串模型 ( 键值对模型 )

文章目录

一、业务逻辑需求

二、完整代码实现





一、业务逻辑需求


在 C 中实现 键值对 字符串 的 读取 , 解析 , 保存 操作 ;


键值对字符串样式 "key = value" , = 两边有若干不等的空格 ;

根据 key 获取 value ;


首先 , 查找 键 字符串 , 查找后 , 辅助指针变量移动到 键 字符串后面的位置 ;


 

// I . 查找子串 key
    p = key_value;
    // 注意此处返回值是 key 在 key_value 字符串中首次出现的地址
    // 如果继续向后遍历,  跳过 key 的字符个数即可
    p = strstr(p, key);
    // 辅助指针变量 , 达到下一次检索条件
    // 上面的 strstr 函数返回的是 key 在 key_value 字符串中首次出现的地址
    // 这里跳过 key 的字符个数 , 从 key 后的第一个字符开始遍历
    p = p + strlen(key);


然后 , 查找 = 字符 , 与上面的操作基本相同 ;


 

// II . 查找 = 字符
    // strstr 函数返回的是 = 在 p 字符串中首次出现的地址
    p = strstr(p, "=");
    // 没有查找到子串
    if(p == NULL)
    {
        return -1;
    }
    // 辅助指针变量 , 越过 = , 继续向后执行
    p = p + strlen("=");


最后 , 将 = 字符后的内容中的空格去除 ; 下面的方法是参考 【C 语言】字符串模型 ( 两头堵模型 | 将 两头堵模型 抽象成业务模块函数 | 形参返回值 | 函数返回值 | 形参指针判空 | 形参返回值操作 ) 博客中的方法修改而来的 ;


int trim_space(char *str_all, char *str_no_space)
{
    // 验证指针合法性 , 指针为空直接返回报错
    if(str_all == NULL || str_no_space == NULL)
    {
        printf("error : str_all == NULL || count == NULL");
        return -1;
    }
    // 局部临时指针变量 接收 函数形参
    char *str = str_all;
    char *str_no_space_tmp = str_no_space;
    // 两个字符串索引 , i 是指向头部 , j 指向尾部
    int i = 0, j = strlen(str) - 1;
    // 保存非空字符串长度 , 局部临时变 , 计算结果
    int count_tmp = 0;
    // 循环条件是 i 指针指向的 位置 为空 则继续循环
    // 遇到第一个不为空的字符 , 便停止循环
    // 停止循环时的 i 指向从左侧开始第一个不为空的字符
    while(isspace(str[i]) && str[i] != '\0')
    {
        i++;
    }
    // 循环条件是 j 指针指向的 位置 为空 则继续循环
    // 遇到第一个不为空的字符 , 便停止循环
    // 停止循环时的 j 指向从右侧开始第一个不为空的字符
    while(isspace(str[j]) && str[j] != '\0')
    {
        j--;
    }
    // 计算结果
    count_tmp = j - i + 1;
    // 将去除空格的字符串拷贝到 str_no_trim_tmp 指针指向的空间中
    strncpy(str_no_space_tmp, str + i, count_tmp);
    return 0;
}





二、完整代码实现


完整代码示例 :


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int trim_space(char *str_all, char *str_no_space)
{
    // 验证指针合法性 , 指针为空直接返回报错
    if(str_all == NULL || str_no_space == NULL)
    {
        printf("error : str_all == NULL || count == NULL");
        return -1;
    }
    // 局部临时指针变量 接收 函数形参
    char *str = str_all;
    char *str_no_space_tmp = str_no_space;
    // 两个字符串索引 , i 是指向头部 , j 指向尾部
    int i = 0, j = strlen(str) - 1;
    // 保存非空字符串长度 , 局部临时变 , 计算结果
    int count_tmp = 0;
    // 循环条件是 i 指针指向的 位置 为空 则继续循环
    // 遇到第一个不为空的字符 , 便停止循环
    // 停止循环时的 i 指向从左侧开始第一个不为空的字符
    while(isspace(str[i]) && str[i] != '\0')
    {
        i++;
    }
    // 循环条件是 j 指针指向的 位置 为空 则继续循环
    // 遇到第一个不为空的字符 , 便停止循环
    // 停止循环时的 j 指向从右侧开始第一个不为空的字符
    while(isspace(str[j]) && str[j] != '\0')
    {
        j--;
    }
    // 计算结果
    count_tmp = j - i + 1;
    // 将去除空格的字符串拷贝到 str_no_trim_tmp 指针指向的空间中
    strncpy(str_no_space_tmp, str + i, count_tmp);
    return 0;
}
/*
 * 根据 key 获取 value
 * char *key_value : 键值对字符串 "name =   Tom"
 * char *key : 键 "name"
 * char *value : 值 "Tom"
 * int *value_len : 值 的字符个数 , 4 , 包括结尾的 \0 字符
 */
int get_value(char *key_value, char *key, char *value, int *value_len)
{
    // 辅助指针变量 , 接收查找子串的返回值 , 同时指向当前处理的字符串
    char *p = NULL;
    // 各种函数执行返回值
    int ret = 0;
    // I . 查找子串 key
    p = key_value;
    // 注意此处返回值是 key 在 key_value 字符串中首次出现的地址
    // 如果继续向后遍历,  跳过 key 的字符个数即可
    p = strstr(p, key);
    // 没有查找到子串
    if(p == NULL)
    {
        return -1;
    }
    // 辅助指针变量 , 达到下一次检索条件
    // 上面的 strstr 函数返回的是 key 在 key_value 字符串中首次出现的地址
    // 这里跳过 key 的字符个数 , 从 key 后的第一个字符开始遍历
    p = p + strlen(key);
    // II . 查找 = 字符
    // strstr 函数返回的是 = 在 p 字符串中首次出现的地址
    p = strstr(p, "=");
    // 没有查找到子串
    if(p == NULL)
    {
        return -1;
    }
    // 辅助指针变量 , 越过 = , 继续向后执行
    p = p + strlen("=");
    // III . 将 = 字符后面的空格去除
    ret = trim_space(p, value);
    if(ret != 0)
    {
        printf("error : trim_space %d \n", ret);
    }
    return 0;
}
int main()
{
    // 要解析的键值对字符串
    char *key_value = "name =   Tom  ";
    // 键
    char *key = "name";
    // 存放解析后的 值
    char value[1024];
    // 存放解析后的 值 的字符个数
    char value_len = 0;
    // 接收 get_value 方法的返回值
    int ret = 0;
    // 获取 key_value 键值对字符串中的 key 对应的 value 值
    ret = get_value(key_value, key, value, &value_len);
    // 执行失败后的处理结果
    if(ret != 0)
    {
        printf("error : get_value failed %d\n", ret);
        return ret;
    }
    // 执行成功, 打印 value 值
    printf("value = %s\n", value);
    // 命令行不要退出
    system("pause");
    return ret;
}



执行结果 :


image.png

目录
相关文章
|
1月前
|
JavaScript 前端开发 Java
模板字符串和普通字符串的性能差异大吗?
总体而言,模板字符串和普通字符串的性能差异并非在所有场景下都非常显著,但在一些复杂的、对性能要求较高的场景中,模板字符串可能会展现出一定的优势。不过,在实际开发中,性能并非是选择使用哪种字符串的唯一考量因素,代码的可读性、可维护性以及开发效率等同样重要。
|
6月前
|
存储 算法 Java
Java数据结构与算法:用于高效地存储和检索字符串数据集
Java数据结构与算法:用于高效地存储和检索字符串数据集
|
7月前
|
C++
【C++基础】C++中的字符串
【C++基础】C++中的字符串
24 0
|
7月前
|
自然语言处理 语音技术
语言大模型和文本大模型的区别
【2月更文挑战第16天】语言大模型和文本大模型的区别
171 2
语言大模型和文本大模型的区别
|
7月前
|
C++
C++语言学习数组和字符串应用案例
【4月更文挑战第8天】该文展示了C++中数组和字符串的应用案例。数组示例定义了一个整数数组并访问、修改其元素,计算了元素之和。字符串示例中,定义了一个字符串并遍历、修改字符,进行了字符串拼接、查找子字符串及替换操作。
48 3
|
7月前
|
存储 安全 索引
Day5 长篇:字符串和常用数据结构
Day5 长篇:字符串和常用数据结构
29 0
|
7月前
|
存储 算法 搜索推荐
在C++语言中数组算法
在C++语言中数组算法
53 0
|
索引 Python
python之集合、序列、字典类型
python之集合、序列、字典类型
185 0
python之集合、序列、字典类型
下一篇
DataWorks