问题提出

《C Programming Language》书中在递归这一节预留了两个使用递归实现的函数,其中itoa函数是用来将一个整数转换为一个字符串。书中已有使用循环实现的版本,但是直接得到的是反序的结果,需要最后调用reverse函数。而递归版本则可以避免这个问题。 
首先使用原接口void itoa(int n, char s[])进行实现,发现递归调用的时候总是错误,输出的结果只能得到整数n的最高位和最低位的数字。代码如下:

 void itoa(int n, char* s)
{
if (n < )
{
*s = '-';
s++;
}
if (abs(n) / )
itoa(abs(n) / , s++);
*s = abs(n) % + '';
}

基本思想:使用字符指针s进行移动,递归调用的过程中进行指针的自增。但是使用gcc编译后运行结果并不正确。只能输出整数n的最高位和最低位的两个数字。 
错误原因:经过仔细调试,发现字符指针参数每次递归调用时,传入的s++这个变量时进行了复制,也就是在后续的递归调用中对指针s的自增操作不会反映到第一次调用的指针s上去,因此,第一次调用这个函数并最终调用*s = abs(n) % 10 + '0';这一语句时,指针s指向的是字符数组的第二个位置。就将整数n的最低位放在第二个位置,而整数n的最高位一直进行递归调用并将参数指针变量复制(一直指向第一个位置),也就放在了第一个位置。最后返回的结果就只有两个数字。

已有的方案

针对上述遇到的问题,在网上搜索了这个问题。使用最多的就是使用局部静态变量保存字符数组的下标,参考http://blog.csdn.net/long_xing/article/details/2212450。但是,另外一篇博客中的实现存在逻辑错误,提前给字符数组赋“\0”,参见http://blog.csdn.net/roma823/article/details/6546719

自己实现

经过一段时间的思考,最终想到的办法就是二级字符指针,因为使用字符指针函数参数时,递归调用传入的字符指针使用了复制,这应该是编译器对字符指针(相当于字符串),在进行传递参数时使用复制指针(也就是复制字符串)这种实现,与普通意义上的指针参数调用时直接操作源指针的思想有所区别。因此在使用一个二级字符指针,每次传递的是指向字符串的同一个指针,这样就不会出现递归调用过程中对指针的自增操作对上一级调用函数无效的情况,代码如下:

 void itoa(int num, char** s)
{
if (num < )
{
**s = '-';
(*s)++;
}
if (abs(num / ) > )
{
itoa(abs(num / ), s);
(*s)++;
}
**s = abs(num % ) + '';
}

上述版本使用了abs库函数,可以转换最大负整数。测试如下:

 int main()
{
char s[], *ps = s;
itoa(, &ps); //测试正整数
*(++ps) = '\0';
printf("%s\n", s); ps = s;
itoa(-, &ps); //测试最大的负整数(32 bits)
*(++ps) = '\0';
printf("%s\n", s); return ;
}

itoa函数的递归实现(二级指针实现)的更多相关文章

  1. C语言一级指针与二级指针

    指针的概念 指针就是地址, 利用这个地址可以找到指定的数据 指针就是地址, 那么在使用的时候, 常常会简单的说 指针变量为指针 指针变量就是存储地址的变量 int *p1;// 申请了一个变量, 即在 ...

  2. 真正明白C语言二级指针(转载)

    指针是C语言的灵魂,我想对于一级指针大家应该都很熟悉,也经常用到:比如说对于字符串的处理,函数参数的“值,结果传递”等,对于二级指针或者多级指针,我想理解起来也是比较容易的,比如二级指针就是指向指针的 ...

  3. C 真正理解二级指针

    本文转载自CSDN博主liaoxinmeng,做数据结构时遇到指针方面的问题,想了许久,因此我觉得很有必要复习一下二级指针及其使用 正文如下: 指针是C语言的灵魂,我想对于一级指针大家应该都很熟悉,也 ...

  4. 真正明白c语言二级指针

    指针是C语言的灵魂,我想对于一级指针大家应该都很熟悉,也经常用到:比如说对于字符串的处理,函数参数的“值,结果传递”等,对于二级指针或者多级指针,我想理解起来也是比较容易的,比如二级指针就是指向指针的 ...

  5. 对线程等待函数pthread_join二级指针参数分析

    分析之前先搞明白,这个二级指针其实在函数内部是承接了上个线程的返回值. 看man手册,发现返回值是个普通指针.人家用二级指针来承接,可能准备干大事.这个可以自己搜索一下.原因嘛,二级指针是保存了这个地 ...

  6. C#中调用C++的dll的参数为指针类型的导出函数(包括二级指针的情况)

    严格来说这篇文章算不上C++范围的,不过还是挂了点边,还是在自己的blog中记录一下吧. C++中使用指针是家常便饭了,也非常的好用,这也是我之所以喜欢C++的原因之一.但是在C#中就强调托管的概念了 ...

  7. C-指针,二级指针,二维数组作为函数参数使用,C语言链表(详解)

    一级指针 int *p;            //表示定义一个int型(4字节)的指针p &p                 //表示p自身的地址位置 p                  ...

  8. (C++)函数参数传递中的一级指针和二级指针

    主要内容: 1.一级指针和二级指针 2.函数指针传递的例子 3.什么时候需要传递二级指针? 4.二级指针在链表中的使用 1.一级指针和二级指针 一级指针:即我们一般说的指针,就是内存地址: 二级指针: ...

  9. FreeRTOS信号量的封装函数参数是二级指针

    1. 先看正确的封装方式,问题所在,为什么要用2级指针 void cissys_lockcreate(void** mutex) { //创建信号量,应该是互斥锁 *mutex = ((Semapho ...

随机推荐

  1. 开源PaaS产品介绍

    简介 本文主要介绍一下PaaS领域两个著名的开源软件: Cloud Foundry和OpenShift. [广告]如果你喜欢本博客,请点此查看本博客所有文章:http://www.cnblogs.co ...

  2. 解决content is not allowed in prolog问题

    将xml文档用notepad++以UTF-8无BOM格式编码保存便可以了

  3. Git链接到自己的Github(2)进阶使用

    接着上一篇的,从github clone下代码. 1.先查看当前开发分支 $ cat .git/HEAD ref: refs/heads/master 这里的master是默认分支. 2.查看当前状态 ...

  4. PHP中$_POST,$_GET,$_REQUEST,$_FILES全局变量的全局指什么

    我一直担心,同一个表单,同时提交2次会发生什么事?在服务器端表单变量会不会彼此覆盖呢?也就是说假如我们在PHP中用$_REQUEST["name"]访问某个表单变量,会不会因为别人 ...

  5. Android_Menu_contextMenu

    xml文件: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns ...

  6. Forwarding a Range of Ports in VirtualBox

    STAN SCHWERTLY MAY 9, 2012 ARTICLES 3 COMMENTS Doesn't allow forwarding a range of ports through the ...

  7. a链接中关于this的使用

    a连接点击事件用 this 时,要用 onclick='click(this)',href='javascript:void()' a连接无法使用,要看看是不是自动变成ie7或者更低

  8. /lib /usr/lib /usr/local/lib 的区别

    /lib是内核级的,/usr/lib是系统级的,/usr/local/lib是用户级的. /lib/ — 包含许多被 /bin/ 和 /sbin/ 中的程序使用的库文件.目录 /usr/lib/ 中含 ...

  9. 【转】Linux网络相关查询脚本

    1. 查看TCP连接状态 netstat -nat |awk '{print $6}'|sort|uniq -c|sort -rn netstat -n | awk '/^tcp/ {++S[$NF] ...

  10. GitHub 中国区前100 名技术专家

    [本文是在一片新闻上摘录的,原地址为:http://mt.sohu.com/20160407/n443539407.shtml] 本文根据Github公开API,抓取了地址显示China的用户,根据粉 ...