在C/C++中,字符串以'\0'结尾是一种强制要求,或者说,只有满足这个要求的

字符数组才能被称为字符串。否则,你所做的所有操作结果都是未定义的

C标准库string.h中所有关于字符串的函数都有一个特性,对于输入字符串,默认为是以'\0'结尾的

,否则就会出现未定义行为,比如strlen,实现就依赖了这一点:

int len = 0;

while(*s++)len++;对于输出字符串(一把是返回一个char*指针),也保证在末尾加上一个'\0'

本来,如果把字符串完全当成整体(像std::string那样)的话,是不用担心这个问题的,但是C/C++里面

字符串又是以字符数组的形式存在的,所以会出现这样的代码

char a[] = {'a', 'b', 'c', 'd'};

size_t len = strlen(a);

能通过编译,因为类型检查不会出错,但是运行时行为却很诡异,基本是错的,正确的定义方式

应该是这样的char a[] = {'a', 'b', 'c', 'd', '\0'};

这个坑也影响到了stl里面string的实现,比如string的data()方法和c_str()方法默认返回的char* 就一定是以'\0'结尾的

但可怕之处就在于string又可以拆开来用,看下面代码:

#include<iostream>
#include<string.h>
#include<string>
using namespace std;
int main() {
char a[] = { 'a', 'b', 'c', 'd','\0'};
string s(a);
cout << s.size() << endl;
cout << s.length() << endl;
s[4] = 'e';
cout << s << endl;
cout << s.size() << endl;
cout << s.c_str() << endl;
cout << s.data() << endl;
return 0;
}

有没发现开始的size和length的输出都符合C里面关于一个字符串的定义(排除那个'\0')

但是对于s[4] = 'e'这种会破坏字符串结构的行为,编译器竟无动于衷,导致后面s.c_str()输出就是乱码。

www.cplusplus.com里面关于string类的operator[](int)方法和at(int)方法的说明如下:

   char& operator[] (size_t pos);
const char& operator[] (size_t pos) const;
If pos is equal to the string length, the function returns a reference to the null character that follows the last character in the string (which should not be modified).
索引允许等于length,但是should not be modified! 也就是如果修改就是未定义行为
      char& at (size_t pos);
const char& at (size_t pos) const;
Get character in string
Returns a reference to the character at position pos in the string.

The function automatically checks whether pos is the valid position of a character in the string (i.e., whether pos is less than the string length), throwing an out_of_range exception if it is not.

这个函数倒是严格规定不许索引到length处!否则会有异常!

总结:字符串和字符数组不是一回事!而标准库里string类仍旧是以普通字符数组的形式来实现字符串的,

所以也留下了可能破坏字符串结构的隐患!

C语言string.h中所有不带n的字符串函数其实都假设了输入是合法的null terminated string,否则会造成未定义行为

比如 strcpy 应该改用 strncpy

      strcmp 应该改用strncmp

      strcat 应该改用strncat

      这里面的n全部都是包括'\0'在内的总字节数,简单解释就是,一般我们的字符串操作函数都是循环直到'\0'为止,现在

多了一个结束出口,就是到达n处为止!

另外像strlen 甚至C++里面char_traits::length对于不以'\0'结尾的字符串也是有隐患的,输入未定义行为!

C/C++ 字符串 null terminal的更多相关文章

  1. Java字符串null相加

    Java字符串null相加 最近和同事讨论了下面的一段代码: String a = null; a += a; System.out.println(a); 运行结果: nullnull 本着学习的态 ...

  2. Net Core 下 Newtonsoft.Json 转换字符串 null 替换成string.Empty

    原文:Net Core 下 Newtonsoft.Json 转换字符串 null 替换成string.Empty public class NullToEmptyStringResolver : De ...

  3. 空字符串‘’ null false 区别

    1.''空字符串 .null 和false都是以值为0来存储的   只是数据结构不一致而已 空字符串------字符串数据格式 null -----------null数据格式 false ----- ...

  4. Asp.Net Core 下 Newtonsoft.Json 转换字符串 null 替换成string.Empty

    public class NullToEmptyStringResolver : DefaultContractResolver { /// <summary> /// 创建属性 /// ...

  5. spring boot 返回json字符串 null值转空字符串

    @Configuration public class JacksonConfig { @Bean @Primary @ConditionalOnMissingBean(ObjectMapper.cl ...

  6. PHP递归方式把一个数组里面的null转换为空字符串”的方法

    在一些接口的调用中,直接查询数据库出来的字段可能为null字段,但是为了简便前端的判断,需要把null转换成空字符串'',这个时候就需要递归的方式进行.直接上代码如下: //递归方式把数组或字符串 n ...

  7. laravel之null替换空字符串中间件

    在laravel写接口的时候免不了数据库中保存null,可用诸如设置ORM的访问器或以下方法处理 $goods->name?$goods->name:''; 其实可以利用路由中间件,在需要 ...

  8. PHP 中空字符串介绍0、null、empty和false之间的关系

    0是数字,是empty,是false,不是null,值相当于空字符串,但类型不是字符串,去空格或强制转换为字符串型时不等于空字符串 ""的值相当于0,是empty,是空字符串,是f ...

  9. 空字符串(“”)和null和空格字符串(" ")的区别

    1.类型 null表示的是一个对象的值,而并不是一个字符串.例如声明一个对象的引用,String a = null ;""表示的是一个空字符串,也就是说它的长度为0,但它是一个字符 ...

随机推荐

  1. DispatcherServlet

    <servlet> <servlet-name>chapter2</servlet-name> <servlet-class>org.springfra ...

  2. Linux内核零碎知识

    UNIX系统:内核.shell外壳.文件系统.工具或应用程序. 操作系统功能:进程与处理机管理.存储管理.设备管理.作业管理.文件管理. 内存是磁盘的缓存,cache是内存的缓存. 可把内核看作是不断 ...

  3. 20150206读书笔记<深入理解计算机系统>

    ●第一章 C是系统级编程的首选.C++显示支持抽象,属于应用级程序设计语言. 简单例子: 一个典型系统的硬件组成: 存储器的层次结构: 注:存储器层次结构的设计思想是,该层存储器作为下一层存储器的高速 ...

  4. ThinkPHP框架如何修改X-Powered-By

    以前用ThinkPHP框架开发了一个小网站,前几天查询页面HTTP状态发现,里面有一项: X-Powered-By: ThinkPHP 2.0 这样虽然没什么,但感觉如果别有用心的人查询会知道你是用这 ...

  5. Java 基础知识点(必知必会其一)

    如何将字符串转换为数字? package Day_2; /** * @author Administrator * 功能: 如何将字符串转换为数字? */ public class StringToI ...

  6. OID,主键生成策略,PO VO DTO,get和load区别,脏检查,快照,java对象的三种状态

    主键生成策略 sequence 数据库端 native 数据库端 uuid  程序端 自动赋值 生成的是一个32位的16进制数  实体类需把ID改成String 类型 assigned  程序端 需手 ...

  7. 正则表达式 (C++)

    本主题讨论各正则表达式引擎的语法. 正则表达式语法 语法摘要 语义详细信息 匹配和搜索 格式标志 正则表达式语法   元素  元素可以是下列项之一: 一般字符,可匹配目标序列中的相同字符. 通配符“. ...

  8. java中的if-Switch选择结构

    字随笔走,笔随心走,随笔,随心.纯属个人学习分析总结,如有观者还请不啬领教. 1.if选择结构 什么是if结构:if选择结构是根据判断结果再做处理的一种语法结构. 起语法是: if(判断条件){ 操作 ...

  9. 创建PO/SO

    IF P_ZY EQ 'X'."直营订单 调拨单 PERFORM FRM_INIT_PO_HEADER. PERFORM FRM_INIT_PO_ITEM. PERFORM FRM_INIT ...

  10. Windows上模拟Linux环境

    有两种方法,一直是 MinGW http://jingyan.baidu.com/article/8cdccae985cf7c315413cd35.html 另外可以用 cygwin http://j ...