- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想了解指针、数组和字符串文字在 C++ 中的工作方式。
假设我们有下面这行代码:
const char* const letters[] = {"A+","A"};
如果我没理解错的话,这个声明将字母声明为指向常量字符的常量指针数组。根据我的理解,编译器实际上会将每个字符串文字转换为以空字符结尾的 char 数组,字母的每个元素实际上是指向该数组第一个元素的常量指针。
例如,letters[0]
实际上是指向“A+”的“A”的指针。然而
std::cout<< letters[0];
实际输出"A+"
到标准输出。怎么会这样?特别是因为 letters[0]
是常量指针吗?
我的第二个问题与上面的声明有关:如果字符串文字实际上是 const char 数组,那么为什么下面的代码行
const char* const letters[] = {{'A','+','\0'},{'A','\0'}};
抛出
error: braces around scalar initializer for type ‘const char* const’
const char* const letters[] = {{'A','+','\0'},{'A','\0'}};
^
谢谢!
最佳答案
标准规定,就您的程序而言,字符串文字表示为 const
的数组。静态存储持续时间的字符,尾随 '\0'
终结者。该标准未指定编译器如何实现此效果,仅指定您的程序可以以这种方式处理字符串文字。
因此,要么阻止修改字符串文字(例如,将字符串文字传递给期望 char *
的函数是可诊断的错误,代码将无法编译),要么 - 如果代码绕过类型系统修改任何字符在字符串文字中 - 涉及未定义的行为。
在您的示例中,letters[0]
类型为 const char *
, 并且其值等于字符串文字中第一个字符的地址 "A+"
.
std::cout
, 属于 std::ostream
类型, 有一个 operator<<()
接受 const char *
.此函数由语句 std::cout << letters[0]
调用并且该函数假定 const char *
指向以零结尾的数组 char
.它遍历该数组,单独输出每个字符,直到遇到结尾的 '\0'
。 (这不是输出)。
事实是,一个 const char *
表示指针指向 const char
,并不是不能更改指针(即 char * const
)。所以可以增加指针,但不能改变它指向的值。所以,如果我们这样做
const char *p = letters[0];
while (*p != '\0')
{
std::cout << *p;
++p;
}
循环遍历字符串文字 "A+"
的字符,分别打印每一个,并在到达 '\0'
时停止(上面产生相同的可观察输出 std::cout << letters[0]
)。
然而,在上面
*p = 'C';
不会编译,因为 p
的定义告诉编译器 *p
不能改变。但是,递增 p
仍然允许。
原因
const char* const letters [] = {{'A','+','\0'},{'A','\0'}};
不编译是因为数组初始化器不能用于初始化指针。例如;
const int *nums = {1,2,3}; // invalid
const * const int nums2 [] = {{1,2,3}, {4,5,6}}; // invalid
都是非法的。相反,需要定义数组,而不是指针。
const int nums[] = {1,2,3};
const int nums2[][3] = {{1,2,3}, {4,5,6}};
所有版本的 C 和 C++ 都禁止以这种方式初始化指针(或您的示例中的指针数组)。
从技术上讲,使用字符串文字来初始化指针的能力实际上是异常的,而不是禁止使用数组初始化指针。 C 引入字符串文字豁免的原因是历史性的(在 C 的早期,远在 K&R C 之前,字符串文字也不能用于初始化指针)。
关于c++ - 将字符串文字转换为 char 数组实际上在 C++ 中如何工作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57645930/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!