文章目录

  • 一、业务逻辑需求
  • 二、完整代码实现

一、业务逻辑需求


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

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

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

    // I . 查找子串 keyp = 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 . 查找子串 keyp = 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;
}

执行结果 :

【C 语言】字符串模型 ( 键值对模型 )相关推荐

  1. C语言字符串相关一级指针内存模型

    C语言字符串相关一级指针内存模型 通过实例探索一级指针内存模型 通过实例探索一级指针内存模型 void main() {char buf[20]= "aaaa";char buf2 ...

  2. android 获取url中的参数,验证邮箱格式,截取字符串中键值对的值,String的字节长度,去空格,替换字符

    String ss="hello"; byte[] buff=ss.getBytes(); int f=buff.length; System.out.println(f); 字节 ...

  3. 五大存储模型关系模型 键值存储 文档存储 列式存储 图形数据库

    也可以认为是五大数据库存储模型. 数据库市场需要细分,行式数据库不再满足所有的需求,而有很多需求需要通过本内存数据库和列式数据库解决,列式数据库在数据分析.海量存储.BI这三个领域有自己独到. 1. ...

  4. Android Okhttp3的使用(很全面,包含Post提交字符串、键值对、表单、上传文件、无参请求和Get有参无参请求,还有自动添加token)

    Okhttp简介 okhttp是现代化应用程序的网络通信的方式.它用来帮助程序交换数据和媒体信息,使用okhttp可以让你的程序加载物料(翻译为服务器数据更合适在)更加高效.更节省网络带宽. okht ...

  5. Java FastJson解析json字符串 提取键值

    ps : 没啥难点! 总结的两个要点错误请纠正 不想看的直接复制代码运行看结果就行 简单例子为例 (例子在下面 习惯总结写上面 没有好的排版经验) 简单例子没啥要点了 照着套就行 - 之前用的 jav ...

  6. cockroachdb mysql_CockroachDB学习笔记——[译]CockroachDB中的SQL:映射表中数据到键值存储...

    CockroachDB学习笔记--[译]CockroachDB中的SQL:映射表中数据到键值存储 原文标题:SQL in CockroachDB: Mapping Table Data to Key- ...

  7. [转]实现键值对存储(长文)

    实现键值对存储(0):目录 本文由 伯乐在线 - 熊铎 翻译.未经许可,禁止转载! 英文出处:Emmanuel Goossaert (CodeCapsule.com).欢迎加入翻译组. 2014年7月 ...

  8. php 生成动态键值 数组_你的PHP项目遇到性能问题了吗?看完这篇性能分析恍然大悟...

    你的项目中遇到性能问题了吗?遇到性能问题你是如何解决的呢?你的解决方式是否正确呢?下面就跟大家一起分享php项目的性能问题. PHP语言级性能分析 php在什么情况下会遇到性能问题呢? 在讨论性能问题 ...

  9. 细说PHP笔记03(第7章)--数组与数据结构,数组定义,数组遍历,数组内部指针遍历,键值操作函数,统计数组函数,回调函数处理数组元素,数组排序,拆分、合并、分解、结合数组,数组实现堆栈,随机选取元素

    1.数组 索引数组:索引值是整数 关联数组:索引值是字符串 2.数组的定义 1.以直接赋值的方式声明 $数组名[下标]=资料内容 或 $数组名[关联字符串(键值)]=资料内容 <?php $va ...

最新文章

  1. 为什么css一开始需要设定margin和padding为 0px?
  2. python123动物重量排序_python基本常识
  3. 选化学可否报计算机专业,选课选物化生报什么专业前景好
  4. 当汇错款时该怎么办?
  5. VBS的疑惑,它们不考虑效率吗?
  6. java反射可以获得什么属性_java反射获得属性的值
  7. 高中生计算机专业的研究论文,高中计算机个性化教学模式研究的论文
  8. win10系统用户:如何获得超级管理员权限(vue开发之Win10踩坑)
  9. 西瓜书+实战+吴恩达机器学习(八)监督学习之朴素贝叶斯 Naive Bayes
  10. 8.数据结构 --- 动态存储管理
  11. JAVA多线程(转)
  12. win10电脑360调用不到JAVA,win7/win10系统360浏览器打不开原因及解决方法
  13. 按指定次数重复显示 Excel 单元格
  14. DC-1详解(绝对的详细!)
  15. android系统方法裁剪图片 华为手机显示为圆
  16. HBulider 连接手机,在手机端展示
  17. IT 技术学习资料分享
  18. 第十三周 任务三
  19. Ventoy 制作U盘启动盘 使用教程
  20. 【编译原理】 三地址语句的具体实现

热门文章

  1. 错误:不允许有匹配 [xX][mM][lL] 的处理指令目标。
  2. 移动网站用backbone还是angular?
  3. 【转】线性代数的几何意义
  4. 不能创建对象: 'CDONTS.Newmail'”
  5. 职业生涯中的八大“定位法则”(转)
  6. 《实战突击:PHP项目开发案例整合(第2版)(含DVD光盘1张)》
  7. 线性回归——lasso回归和岭回归(ridge regression)
  8. bootstrap validator 出现Maximum call stack size exceeded
  9. Python 错误和异常小结
  10. C# Excel数据有效性