常见的字符串面试题

      基本上求职者进行笔试时没有不考字符串的。字符串也是一种相对简单的数据结构,结合指针,容易多次引起面试官反复发问。我曾不止一次在笔试或面试时遇到下面几道试题。事实上,字符串也是一个考验程序员编程规范和编程习惯的重要考点。不能忽视这些细节,因为这些细节会体现你在操作系统、软件工程、边界内存处理等方面的知识掌控能力。

一、整数与字符串之间的相互转换
1.将字符串转换成整数

      例如有一字符串str[]=“1234”,最直观的转换方法就是依次将每一个字符转换成相应的数字,然后乘10累加,即((1*10+2)*10+3)*10+4。想要得到每一位相应的数字,只需用每个字符减去数字0的ASCII码,即2= str[1]-‘0’。
      如果字符串首字符标示的是一个负数,即str[0]==’-‘。那么此时应该设置一个符号判断标示位(用来判断以后得到的整数是否需要乘-1),并把要处理的字符数组下标设置为1。
      int str_to_int(char str[])
      {
        int i=0, isNeg=0,num=0;
        if(str[0]==’-‘)
          {isNeg=1;
           i=1;
          }
        while(str[i])
          {num=num*10+(str[i]-‘0’);
           i++;
          }
        if(isNeg)
          num*=-1;
        return num;
      }


2.将整数转换为字符串
      将整数转换为字符串相对要复杂些。通过取模,并结合商与余数的关系,可以有几种方法。例如,123除以100,商为1,余数为23,这样,首先可以把“1”保存到字符串里。然后23除以10,商为2,余数为3。这样就可以得到字符串“123”了。但是,怎样来确定整数的位数确实比较麻烦。有一种相对简单的方法。
      首先将123除10取模,得到数字3,此时商为12。再将12除10取模,得到数字2,商为1。这样,我们可以得到字符串“321”,接着进行一次逆序即可得到想要的字符串。
    同样,如果该整数小于0,我们需要人为地将其变成正数,再进行取模运算!!
       char int_to_str(int num,char str[])
       {
        int i=0,j=0,isneg=0;
        char temp[10];
        if(num<0)
          {
           num*=-1;
           isNeg=1;
          }
        do{
           temp[i]=(num%10)+’0′;
           num/=10;
           i++;
            }while(num)    //使用do-while结构保证当num为0时也执行循环。
        if(isNeg)
          temp[i++]=’-‘;
        while(i>0)
          str[j++]=temp[–i];
        str[j]=’/0′;      //字符串结束符不能忘掉。
       }

二、传递动态内存
What will happen after running the “Tests”?
    1. #include<iostream>
       void GetMemory(char *p, int num)
        {
          p=(char *)malloc(sizeof(char)*num);
        }
        int main()
        {
         char *str=NULL;
         GetMemory(str,100);
         strcpy(str,”hello”);
         return 0;
        }
      程序崩溃。毛病出在函数GetMemory中。该函数中的参数*p实际上是主函数中str的一个副本,编译器总是要为函数的每个参数做临时副本。在本程序段中,p申请了新的内存,只是把p所指向的内存地址改变了,但是str丝毫未变。因为函数GetMemory没有返回值,因此str并不指向p所申请的那段内存,所以函数GetMemory并不输出任何东西。事实上,每执行一次GetMemory就会申请一块内存,但是申请的内存却不能得到有效地释放,结果是内存一直被占用,最终造成内存泄露。
      修改方法有2种。(1)我们可以用函数的返回值来传递动态内存,把GetMemory函数的返回类型改为指针类型即可。
        char *GetMemory(char *p, int num)
        {
          p=(char *)malloc(sizeof(char)*num);
          return p;
        }
    (2)如果一定要用指针参数去申请内存,那么应该采用指向指针的指针,传str的地址给函数GetMemory。
        void GetMemory(char **p, int num)
        {
          *p=(char *)malloc(sizeof(char)*num);
        }
        int main()
        {
         char *str=NULL;
         GetMemory(&str,100);
         strcpy(str,”hello”);
         return 0;
        }

    2.#include <iostream>
      using namespace std;
      char *GetMemory(void)
      {
        char p[]=”hello”;
        return p;
      }
      int main()
      {
        char *str=NULL;
        str=GetMemory();
        cout<<str;
        return 0;
      }
       这段代码的错误就显而易见了。因为函数GetMemory返回的是一个指向“栈内存”的指针,当调用函数结束后,该指针的地址不是NULL,但是原来的内容已经被清楚,新内容不可知。所以,程序输出的可能是乱码,也有可能正常输出。

三、字符指针
What will happen after running the “Tests”?

     #include <iostream>
     using namespace std;
     int main()
      {
        char *str=”hello,every one!”;
        strcpy(str,”hello,world”);
        cout<<str;
      }
       这段代码的迷惑性是很强的。程序直接使用字符指针来初始化字符串。在C++中,用双引号括起来的字符串被规定为const类型!这样,意味着str也隐含地被设置为const类型,指向的是全局的const内存区。所以,程序无法通过strcpy试图修改str字符串的内容。                              

    原文作者:阿咪
    原文地址: https://blog.csdn.net/AmiRural/article/details/1491196
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞