Removing first word using pointers from char array(C++)

huangapple go评论97阅读模式
英文:

Removing first word using pointers from char array(C++)

问题

I need to write a program to remove the first word from a char array (char []) using only pointers. If there are multiple blank spaces in front, I have to ignore them and remove the first word.

These are the loops I used:

  1. #include <iostream>
  2. #include <cmath>
  3. int main()
  4. {
  5. char sentence[1000];
  6. std::cout << "Input sentence: ";
  7. std::cin.getline(sentence, 1000);
  8. char *p = sentence;
  9. while (*p == ' ' && *p++ == ' ')
  10. {
  11. p++;
  12. }
  13. while (*p != ' ')
  14. {
  15. p++;
  16. }
  17. std::cout << "Sentence without first word: ";
  18. while (*p != 0)
  19. {
  20. std::cout << *p;
  21. p++;
  22. }
  23. std::cout << std::endl;
  24. return 0;
  25. }

*p is a pointer to a character array (char *p = char int[1000]).

When I output the char array later to see if the program is working, it doesn't work properly if the array was an empty array or a one-word array.

Why does *p (pointer to character) print out random signs (chars) instead of nothing when the array is blank?

Example:

  1. Input: word
  2. Output: #U

Full program instead of just loops for editing the array.

英文:

I need to write a program to remove the first word from a char array (char []) using only pointers. If there are multiple blank spaces in front, I have to ignore them and remove the first word.

These are the loops I used:

  1. #include &lt;iostream&gt;
  2. #include &lt;cmath&gt;
  3. int main()
  4. {
  5. char sentence[1000];
  6. std::cout&lt;&lt;&quot;Input sentence: &quot;;
  7. std::cin.getline(sentence,1000);
  8. char *p=sentence;
  9. while(*p==&#39; &#39; &amp;&amp; *p++==&#39; &#39;)
  10. {
  11. p++;
  12. }
  13. while(*p!=&#39; &#39;)
  14. {
  15. p++;
  16. }
  17. std::cout&lt;&lt;&quot;Sentence without first word: &quot;;
  18. while(*p!=0)
  19. {
  20. std::cout&lt;&lt;*p;
  21. p++;
  22. }
  23. std::cout&lt;&lt;std::endl;
  24. return 0;
  25. }

*p is a pointer to char[] (char *p=char int[1000])

When I output the char array later to see if the program is working, it doesn't work properly if the array was an empty array or a one-word array.

Why does *p (pointer to character) print out random signs (chars) instead of nothing, when the array is blank?

Example:

  1. Input: word
  2. Output: #U

Full program instead of just loops for editing the array.

答案1

得分: 2

Your 1st and 2nd while loops don't stop if the null terminator '\0' is reached. So, when the array is empty or contains only 1 word, you are advancing p past the null terminator into an uninitialized section of the array, and possibly even beyond the end of the array into surrounding memory.

Also, you should not be doing p++ inside the 1st loop's condition like you are doing. Your 1st loop is advancing p by two characters per iteration when *p is a space character. It should be advancing p by only one character per iteration.

Try this instead:

  1. #include <iostream>
  2. int main()
  3. {
  4. char sentence[1000] = {};
  5. std::cout << "Input sentence: ";
  6. std::cin.getline(sentence, 1000);
  7. char *p = sentence;
  8. // skip leading whitespace before 1st word
  9. while (*p <= ' ' && *p != '
    #include <iostream>
  10. int main()
  11. {
  12.     char sentence[1000] = {};
  13.     std::cout << "Input sentence: ";
  14.     std::cin.getline(sentence, 1000);
  15.     char *p = sentence;
  16.     // skip leading whitespace before 1st word
  17.     while (*p <= ' ' && *p != '\0') {
  18.         ++p;
  19.     }
  20.     // skip 1st word
  21.     while (*p > ' ') {
  22.         ++p;
  23.     }
  24.     // skip whitespace between 1st and 2nd words
  25.     while (*p <= ' ' && *p != '\0') {
  26.         ++p;
  27.     }
  28.     std::cout << "Sentence without first word: ";
  29.     while (*p != '\0') {
  30.         std::cout << *p;
  31.         ++p;
  32.     }
  33.     // or simply:
  34.     // std::cout << p;
  35.     std::cout << std::endl;
  36.     return 0;
  37. }
  38. ') {
  39. ++p;
  40. }
  41. // skip 1st word
  42. while (*p > ' ') {
  43. ++p;
  44. }
  45. // skip whitespace between 1st and 2nd words
  46. while (*p <= ' ' && *p != '
    #include <iostream>
  47. int main()
  48. {
  49.     char sentence[1000] = {};
  50.     std::cout << "Input sentence: ";
  51.     std::cin.getline(sentence, 1000);
  52.     char *p = sentence;
  53.     // skip leading whitespace before 1st word
  54.     while (*p <= ' ' && *p != '\0') {
  55.         ++p;
  56.     }
  57.     // skip 1st word
  58.     while (*p > ' ') {
  59.         ++p;
  60.     }
  61.     // skip whitespace between 1st and 2nd words
  62.     while (*p <= ' ' && *p != '\0') {
  63.         ++p;
  64.     }
  65.     std::cout << "Sentence without first word: ";
  66.     while (*p != '\0') {
  67.         std::cout << *p;
  68.         ++p;
  69.     }
  70.     // or simply:
  71.     // std::cout << p;
  72.     std::cout << std::endl;
  73.     return 0;
  74. }
  75. ') {
  76. ++p;
  77. }
  78. std::cout << "Sentence without first word: ";
  79. while (*p != '
    #include <iostream>
  80. int main()
  81. {
  82.     char sentence[1000] = {};
  83.     std::cout << "Input sentence: ";
  84.     std::cin.getline(sentence, 1000);
  85.     char *p = sentence;
  86.     // skip leading whitespace before 1st word
  87.     while (*p <= ' ' && *p != '\0') {
  88.         ++p;
  89.     }
  90.     // skip 1st word
  91.     while (*p > ' ') {
  92.         ++p;
  93.     }
  94.     // skip whitespace between 1st and 2nd words
  95.     while (*p <= ' ' && *p != '\0') {
  96.         ++p;
  97.     }
  98.     std::cout << "Sentence without first word: ";
  99.     while (*p != '\0') {
  100.         std::cout << *p;
  101.         ++p;
  102.     }
  103.     // or simply:
  104.     // std::cout << p;
  105.     std::cout << std::endl;
  106.     return 0;
  107. }
  108. ') {
  109. std::cout << *p;
  110. ++p;
  111. }
  112. // or simply:
  113. // std::cout << p;
  114. std::cout << std::endl;
  115. return 0;
  116. }
英文:

Your 1st and 2nd while loops don't stop if the null terminator &#39;\0&#39; is reached. So, when the array is empty or contains only 1 word, you are advancing p past the null terminator into an uninitialized section of the array, and possibly even beyond the end of the array into surrounding memory.

Also, you should not be doing p++ inside the 1st loop's condition like you are doing. Your 1st loop is advancing p by two characters per iteration when *p is a space character. It should be advancing p by only one character per iteration.

Try this instead:

  1. #include &lt;iostream&gt;
  2. int main()
  3. {
  4. char sentence[1000] = {};
  5. std::cout &lt;&lt; &quot;Input sentence: &quot;;
  6. std::cin.getline(sentence, 1000);
  7. char *p = sentence;
  8. // skip leading whitespace before 1st word
  9. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;
    #include &lt;iostream&gt;
  10. int main()
  11. {
  12. char sentence[1000] = {};
  13. std::cout &lt;&lt; &quot;Input sentence: &quot;;
  14. std::cin.getline(sentence, 1000);
  15. char *p = sentence;
  16. // skip leading whitespace before 1st word
  17. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;\0&#39;) {
  18. ++p;
  19. }
  20. // skip 1st word
  21. while (*p &gt; &#39; &#39;) {
  22. ++p;
  23. }
  24. // skip whitespace between 1st and 2nd words
  25. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;\0&#39;) {
  26. ++p;
  27. }
  28. std::cout &lt;&lt; &quot;Sentence without first word: &quot;;
  29. while (*p != &#39;\0&#39;) {
  30. std::cout &lt;&lt; *p;
  31. ++p;
  32. }
  33. // or simply:
  34. // std::cout &lt;&lt; p;
  35. std::cout &lt;&lt; std::endl;
  36. return 0;
  37. }
  38. &#39;) {
  39. ++p;
  40. }
  41. // skip 1st word
  42. while (*p &gt; &#39; &#39;) {
  43. ++p;
  44. }
  45. // skip whitespace between 1st and 2nd words
  46. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;
    #include &lt;iostream&gt;
  47. int main()
  48. {
  49. char sentence[1000] = {};
  50. std::cout &lt;&lt; &quot;Input sentence: &quot;;
  51. std::cin.getline(sentence, 1000);
  52. char *p = sentence;
  53. // skip leading whitespace before 1st word
  54. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;\0&#39;) {
  55. ++p;
  56. }
  57. // skip 1st word
  58. while (*p &gt; &#39; &#39;) {
  59. ++p;
  60. }
  61. // skip whitespace between 1st and 2nd words
  62. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;\0&#39;) {
  63. ++p;
  64. }
  65. std::cout &lt;&lt; &quot;Sentence without first word: &quot;;
  66. while (*p != &#39;\0&#39;) {
  67. std::cout &lt;&lt; *p;
  68. ++p;
  69. }
  70. // or simply:
  71. // std::cout &lt;&lt; p;
  72. std::cout &lt;&lt; std::endl;
  73. return 0;
  74. }
  75. &#39;) {
  76. ++p;
  77. }
  78. std::cout &lt;&lt; &quot;Sentence without first word: &quot;;
  79. while (*p != &#39;
    #include &lt;iostream&gt;
  80. int main()
  81. {
  82. char sentence[1000] = {};
  83. std::cout &lt;&lt; &quot;Input sentence: &quot;;
  84. std::cin.getline(sentence, 1000);
  85. char *p = sentence;
  86. // skip leading whitespace before 1st word
  87. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;\0&#39;) {
  88. ++p;
  89. }
  90. // skip 1st word
  91. while (*p &gt; &#39; &#39;) {
  92. ++p;
  93. }
  94. // skip whitespace between 1st and 2nd words
  95. while (*p &lt;= &#39; &#39; &amp;&amp; *p != &#39;\0&#39;) {
  96. ++p;
  97. }
  98. std::cout &lt;&lt; &quot;Sentence without first word: &quot;;
  99. while (*p != &#39;\0&#39;) {
  100. std::cout &lt;&lt; *p;
  101. ++p;
  102. }
  103. // or simply:
  104. // std::cout &lt;&lt; p;
  105. std::cout &lt;&lt; std::endl;
  106. return 0;
  107. }
  108. &#39;) {
  109. std::cout &lt;&lt; *p;
  110. ++p;
  111. }
  112. // or simply:
  113. // std::cout &lt;&lt; p;
  114. std::cout &lt;&lt; std::endl;
  115. return 0;
  116. }

答案2

得分: 1

  1. while (*p != ' ')

字符串的末尾由一个 '\0' 字节标记。这个字节不是空格。因此,这个while循环会继续前进,寻找一个不存在的空格,导致未定义的行为。这是计算机编程黄金法则的一个例子:「你的计算机总是按照你告诉它做的事情来做,而不是你想让它做的事情。」你告诉你的计算机,在看到 '\0' 之前不要停止while循环。所以你的计算机会执行这个任务。你需要在这里检查空格和 '\0'

  1. <details>
  2. <summary>英文:</summary>

while(*p!=' ')

  1. The end of a character string is marked by a &#39;`
    The end of a character string is marked by a &#39;`\0&#39;` byte.
  2. Which is not a space.
  3. So, this while loop will happily plow forward, looking for a nonexistent space, resulting in undefined behavior.
  4. This is an example of the Golden Rule Of Computer Programming in force: &quot;Your Computer Always Does Exactly What You Tell It To Do Instead Of What You Want It To Do&quot;. You told your computer to not stop the while loop until it sees a `&#39; &#39;`. So that&#39;s what your computer does. The end of the string will not stop your computer, from this mission.
  5. You need to check for both the space and a `&#39;\0&#39;`, here.
  6. </details>
  7. # 答案3
  8. **得分**: 1
  9. 这是一个关于C语言的问题,而不是C++问题,重要的是要记住,“传统”的C字符串以`'\0'`字节结尾。未能检查这一点会导致各种内存访问错误。
  10. 这是一个快速、粗糙、不太规范的实现,它跳过了输入的每一行的第一个单词,同时正确地检查C字符串的结束字节。是否应该跳过第一个单词之后的空格(而不仅仅是第一个单词之前的空格),这很容易调整。
  11. ```c++
  12. #include <iostream>
  13. #include <string>
  14. namespace {
  15. const char* skip_first_word_c_style(const char* c_str) {
  16. while (*c_str && *c_str == ' ') ++c_str;  // spaces before first word
  17. while (*c_str && *c_str != ' ') ++c_str;  // first word
  18. while (*c_str && *c_str == ' ') ++c_str;  // spaces after first word
  19. return c_str;
  20. }
  21. }  // namespace
  22. int main() {
  23. std::string line;
  24. while (std::getline(std::cin, line)) {
  25. std::cout << skip_first_word_c_style(line.c_str()) << std::endl;
  26. }
  27. }
  28. &#39;` byte.
  29. Which is not a space.
  30. So, this while loop will happily plow forward, looking for a nonexistent space, resulting in undefined behavior.
  31. This is an example of the Golden Rule Of Computer Programming in force: &quot;Your Computer Always Does Exactly What You Tell It To Do Instead Of What You Want It To Do&quot;. You told your computer to not stop the while loop until it sees a `&#39; &#39;`. So that&#39;s what your computer does. The end of the string will not stop your computer, from this mission.
  32. You need to check for both the space and a `&#39;
    The end of a character string is marked by a &#39;`\0&#39;` byte.
  33. Which is not a space.
  34. So, this while loop will happily plow forward, looking for a nonexistent space, resulting in undefined behavior.
  35. This is an example of the Golden Rule Of Computer Programming in force: &quot;Your Computer Always Does Exactly What You Tell It To Do Instead Of What You Want It To Do&quot;. You told your computer to not stop the while loop until it sees a `&#39; &#39;`. So that&#39;s what your computer does. The end of the string will not stop your computer, from this mission.
  36. You need to check for both the space and a `&#39;\0&#39;`, here.
  37. </details>
  38. # 答案3
  39. **得分**: 1
  40. 这是一个关于C语言的问题,而不是C++问题,重要的是要记住,“传统”的C字符串以`'\0'`字节结尾。未能检查这一点会导致各种内存访问错误。
  41. 这是一个快速、粗糙、不太规范的实现,它跳过了输入的每一行的第一个单词,同时正确地检查C字符串的结束字节。是否应该跳过第一个单词之后的空格(而不仅仅是第一个单词之前的空格),这很容易调整。
  42. ```c++
  43. #include <iostream>
  44. #include <string>
  45. namespace {
  46. const char* skip_first_word_c_style(const char* c_str) {
  47. while (*c_str && *c_str == ' ') ++c_str;  // spaces before first word
  48. while (*c_str && *c_str != ' ') ++c_str;  // first word
  49. while (*c_str && *c_str == ' ') ++c_str;  // spaces after first word
  50. return c_str;
  51. }
  52. }  // namespace
  53. int main() {
  54. std::string line;
  55. while (std::getline(std::cin, line)) {
  56. std::cout << skip_first_word_c_style(line.c_str()) << std::endl;
  57. }
  58. }
  59. &#39;`, here.
  60. </details>
  61. # 答案3
  62. **得分**: 1
  63. 这是一个关于C语言的问题,而不是C++问题,重要的是要记住,“传统”的C字符串以`'
    The end of a character string is marked by a &#39;`\0&#39;` byte.
  64. Which is not a space.
  65. So, this while loop will happily plow forward, looking for a nonexistent space, resulting in undefined behavior.
  66. This is an example of the Golden Rule Of Computer Programming in force: &quot;Your Computer Always Does Exactly What You Tell It To Do Instead Of What You Want It To Do&quot;. You told your computer to not stop the while loop until it sees a `&#39; &#39;`. So that&#39;s what your computer does. The end of the string will not stop your computer, from this mission.
  67. You need to check for both the space and a `&#39;\0&#39;`, here.
  68. </details>
  69. # 答案3
  70. **得分**: 1
  71. 这是一个关于C语言的问题,而不是C++问题,重要的是要记住,“传统”的C字符串以`'\0'`字节结尾。未能检查这一点会导致各种内存访问错误。
  72. 这是一个快速、粗糙、不太规范的实现,它跳过了输入的每一行的第一个单词,同时正确地检查C字符串的结束字节。是否应该跳过第一个单词之后的空格(而不仅仅是第一个单词之前的空格),这很容易调整。
  73. ```c++
  74. #include <iostream>
  75. #include <string>
  76. namespace {
  77. const char* skip_first_word_c_style(const char* c_str) {
  78. while (*c_str && *c_str == ' ') ++c_str;  // spaces before first word
  79. while (*c_str && *c_str != ' ') ++c_str;  // first word
  80. while (*c_str && *c_str == ' ') ++c_str;  // spaces after first word
  81. return c_str;
  82. }
  83. }  // namespace
  84. int main() {
  85. std::string line;
  86. while (std::getline(std::cin, line)) {
  87. std::cout << skip_first_word_c_style(line.c_str()) << std::endl;
  88. }
  89. }
  90. '`字节结尾。未能检查这一点会导致各种内存访问错误。
  91. 这是一个快速、粗糙、不太规范的实现,它跳过了输入的每一行的第一个单词,同时正确地检查C字符串的结束字节。是否应该跳过第一个单词之后的空格(而不仅仅是第一个单词之前的空格),这很容易调整。
  92. ```c++
  93. #include <iostream>
  94. #include <string>
  95. namespace {
  96. const char* skip_first_word_c_style(const char* c_str) {
  97. while (*c_str && *c_str == ' ') ++c_str; // spaces before first word
  98. while (*c_str && *c_str != ' ') ++c_str; // first word
  99. while (*c_str && *c_str == ' ') ++c_str; // spaces after first word
  100. return c_str;
  101. }
  102. } // namespace
  103. int main() {
  104. std::string line;
  105. while (std::getline(std::cin, line)) {
  106. std::cout << skip_first_word_c_style(line.c_str()) << std::endl;
  107. }
  108. }

如果您需要进一步的信息或帮助,请告诉我。

英文:

Well, this is a C question, not a C++ question, and it is important to keep in mind that “traditional” C strings end with a &#39;\0&#39; byte. Failing to check for that leads to all sorts of memory access errors.

Here’s a quick and dirty and sloppy implementation that skips the first word of each line on its input while checking for C-string ending bytes properly. Whether the spaces after the first word (and not only before the first word) should be skipped, that’s easy to adjust.

  1. #include &lt;iostream&gt;
  2. #include &lt;string&gt;
  3. namespace {
  4. const char* skip_first_word_c_style(const char* c_str) {
  5. while (*c_str &amp;&amp; *c_str == &#39; &#39;) ++c_str; // spaces before first word
  6. while (*c_str &amp;&amp; *c_str != &#39; &#39;) ++c_str; // first word
  7. while (*c_str &amp;&amp; *c_str == &#39; &#39;) ++c_str; // spaces after first word
  8. return c_str;
  9. }
  10. } // namespace
  11. int main() {
  12. std::string line;
  13. while (std::getline(std::cin, line)) {
  14. std::cout &lt;&lt; skip_first_word_c_style(line.c_str()) &lt;&lt; std::endl;
  15. }
  16. }

huangapple
  • 本文由 发表于 2023年4月13日 20:54:42
  • 转载请务必保留本文链接:https://go.coder-hub.com/76005690.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定