首先来个小例子。写strcpy时的一个小错误。代码如下:

void mystrcpy(char* ch1,char* ch2)
{
for (int i=0;ch2[i]!='\0';i++)
{
ch1[i]=ch2[i];
}
ch1[i+1]='\0';
}
最后那个i+1就不用了,直接用i。忘记i已经在循环里加过了。
 

顺便问一下,用这种方式为什么也能正常打印:
void mystrcpy(char* ch1,char* ch2)
{

int i=0;

while((ch1[i]=ch2[i])!='\0'){i++;}

}
这种方式没看到在那里加'\0'结束啊。看到的希望可以指教指教。

 

  1. 以字符串形式出现的,编译器都会为该字符串自动添加一个0作为结束符,如在代码中写 
  2.  
  3. "abc",那么编译器帮你存储的是"abc\0" 
  4.  
  5. "abc"是常量吗?答案是有时是,有时不是。 
  6.  
  7.   不是常量的情况:"abc"作为字符数组初始值的时候就不是,如 char str[] = "abc"; 
  8.  
  9.   因为定义的是一个字符数组,所以就相当于定义了一些空间来存放"abc",而又因为 
  10.  
  11.   字符数组就是把字符一个一个地存放的,所以编译器把这个语句解析为
  12.  
  13.     char str[3] = {'a','b','c'}; 
  14.  
  15.   又根据上面的总结1,所以char str[] = "abc";的最终结果是 
  16.  
  17.   char str[4] = {'a','b','c','\0'}; 
  18.  
  19.   还有,如果char str[] = "abc";是在函数内部写的话,那么这里 
  20.  
  21.   的"abc\0"因为不是常量,所以应该被放在栈上。 
  22.   
  23.   是常量的情况:  把"abc"赋给一个字符指针变量时,如 char* ptr = "abc"
  24.  
  25.   因为定义的是一个普通指针,并没有定义空间来存放"abc",所以编译器得帮我们 
  26.  
  27.   找地方来放"abc",显然,把这里的"abc"当成常量并把它放到程序的常量区是编译器 
  28.  
  29.   最合适的选择。所以尽管ptr的类型不是const char*,并且ptr[0] = 'x';也能编译 
  30.  
  31.   通过,但是执行ptr[0] = 'x';就会发生运行时异常,因为这个语句试图去修改程序 
  32.  
  33.   常量区中的东西。
  34.  
  35.   char* ptr = "abc";这种写法原来在c++标准中是不允许的, 
  36.  
  37.   但是因为这种写法在c中实在是太多了,为了兼容c,不允许也得允许。虽然允许,
  38.  
  39.   但是建议的写法应该是const char* ptr = "abc";这样如果后面写ptr[0] = 'x'的 
  40.  
  41.   话编译器就不会让它编译通过,也就避免了上面说的运行时异常。
  42.  
  43.   又扩展一下,如果char* ptr = "abc";写在函数体内,那么虽然这里的"abc\0"被 
  44.  
  45.   放在常量区中,但是ptr本身只是一个普通的指针变量,所以ptr是被放在栈上的,
  46.  
  47.   只不过是它所指向的东西被放在常量区罢了。