C字符串反转算法

一个很简洁的算法:

void Reverse(char s[])

{

    for(int i = 0, j = strlen(s) - 1; i < j; ++i, --j) {

        char c = s[i];

        s[i] = s[j];

        s[j] = c;

    }

}

#关于a, b交换其它算法:

    a ^= b;

    b ^= a;

    a ^= b;

一个五种解法的版本:

转自:http://www.cnblogs.com/Mainz/articles/1164602.html

这是网络流传的Microsoft的面试题目之一:“编写反转字符串的程序,要求优化速度、优化空间”。因为最近一直很多关注算法方面的实践和研究,因此对这个问题进行了一些思考,给出了5种实现方法(有两种解法相关性比较大)。

解法一:第一次看到这题目,想到最简单、最直觉的解法就是:遍历字符串,将第一个字符和最后一个交换,第二个和倒数第二个交换,依次循环,即可,于是有了第一个解法:

字符串反转算法 - lzyzwbn - lzyzwbn的博客char* strrev1(const char* str)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客       int len = strlen(str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* tmp = new char[len + 1];

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        strcpy(tmp,str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       for (int i = 0; i < len/2; ++i)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客       字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客          char c = tmp[i];

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客           tmp[i] = tmp[len – i - 1];

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客           tmp[len – i - 1] = c;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        }

字符串反转算法 - lzyzwbn - lzyzwbn的博客       return tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客}

这里是通过数组的下标方式访问字符串的字符,实际上用指针直接操作即可。解法二正是基于此,实现代码为:

字符串反转算法 - lzyzwbn - lzyzwbn的博客char* strrev2(const char* str)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* tmp = new char[strlen(str) + 1];

字符串反转算法 - lzyzwbn - lzyzwbn的博客        strcpy(tmp,str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* ret = tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* p = tmp + strlen(str) - 1;

字符串反转算法 - lzyzwbn - lzyzwbn的博客       while (p > tmp)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客       字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客              char t = *tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *tmp = *p;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *p = t;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              --p;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              ++tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        }

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       return ret;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客}

显然上面的两个解法中没有考虑时间和空间的优化,一个典型的优化策略就是两个字符交换的算法优化,我们可以完全不使用任何外部变量即完成两个字符(或者整数)的交换,这也是一个很经典的面试题目。特别是一些嵌入式硬件相关编程中经常要考虑寄存器的使用,因此经常有不使用任何第三个寄存器即完成两个寄存器数据的交换的题目。一般有两个解法,对应这里的解法三和解法四。

接上文

解法三的实现代码为:

字符串反转算法 - lzyzwbn - lzyzwbn的博客char* strrev3(const char* str)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* tmp = new char[strlen(str) + 1];

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        strcpy(tmp,str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* ret = tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* p = tmp + strlen(str) - 1;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       while (p > tmp)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客       字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *p ^= *tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *tmp ^= *p;             

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *p ^= *tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              --p;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              ++tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        }

字符串反转算法 - lzyzwbn - lzyzwbn的博客       return ret;

字符串反转算法 - lzyzwbn - lzyzwbn的博客}

 

解法四的实现代码为:

字符串反转算法 - lzyzwbn - lzyzwbn的博客char* strrev4(const char* str)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* tmp = new char[strlen(str) + 1];

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        strcpy(tmp,str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* ret = tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* p = tmp + strlen(str) - 1;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       while (p > tmp)

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客       字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *p = *p + *tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *tmp = *p - *tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              *p = *p - *tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客               --p;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              ++tmp;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        }

字符串反转算法 - lzyzwbn - lzyzwbn的博客       return ret;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客}

 

实际上我们还可以通过递归的思想来解决这个问题,思想很简单:每次交换首尾两个字符,中间部分则又变为和原来字符串同样的问题,因此可以通过递归的思想来解决这个问题,对应解法五的实现代码为:

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客char* strrev5(/**//*const */char* str,int len)

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       if (len <= 1)

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客              return str;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char t = *str;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       *str = *(str + len -1);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       *(str + len -1) = t;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       return (strrev5(str + 1,len - 2) - 1);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客}

 

以下给出一个测试程序:

字符串反转算法 - lzyzwbn - lzyzwbn的博客nt main(int argc,char* argv[])

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客字符串反转算法 - lzyzwbn - lzyzwbn的博客{

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* str = "hello";

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        P(str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* str2 = strrev1(str);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        P(str2);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* str3 = strrev2(str2);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        P(str3);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* str4 = strrev3(str3);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        P(str4);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* str5 = strrev4(str4);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        P(str5);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       char* str6 = strrev5(str5,strlen(str5));

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客        P(str6);

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客      

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客       return 0;

字符串反转算法 - lzyzwbn - lzyzwbn的博客

字符串反转算法 - lzyzwbn - lzyzwbn的博客}

你就可以看到字符串"hello"和"olleh"交替输出了。

  说明:1)这里解法中没有认真考虑输入字符串的合法性和特殊长度(如NULL、一个字符等)字符串的处理;2)前4个算法不改变输入字符串的值,解法五修改了输入字符串。

你可能感兴趣的:(数据结构与算法C)