itoa函数的递归实现(二级指针实现)
问题提出
《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函数的递归实现(二级指针实现)的更多相关文章
- C语言一级指针与二级指针
指针的概念 指针就是地址, 利用这个地址可以找到指定的数据 指针就是地址, 那么在使用的时候, 常常会简单的说 指针变量为指针 指针变量就是存储地址的变量 int *p1;// 申请了一个变量, 即在 ...
- 真正明白C语言二级指针(转载)
指针是C语言的灵魂,我想对于一级指针大家应该都很熟悉,也经常用到:比如说对于字符串的处理,函数参数的“值,结果传递”等,对于二级指针或者多级指针,我想理解起来也是比较容易的,比如二级指针就是指向指针的 ...
- C 真正理解二级指针
本文转载自CSDN博主liaoxinmeng,做数据结构时遇到指针方面的问题,想了许久,因此我觉得很有必要复习一下二级指针及其使用 正文如下: 指针是C语言的灵魂,我想对于一级指针大家应该都很熟悉,也 ...
- 真正明白c语言二级指针
指针是C语言的灵魂,我想对于一级指针大家应该都很熟悉,也经常用到:比如说对于字符串的处理,函数参数的“值,结果传递”等,对于二级指针或者多级指针,我想理解起来也是比较容易的,比如二级指针就是指向指针的 ...
- 对线程等待函数pthread_join二级指针参数分析
分析之前先搞明白,这个二级指针其实在函数内部是承接了上个线程的返回值. 看man手册,发现返回值是个普通指针.人家用二级指针来承接,可能准备干大事.这个可以自己搜索一下.原因嘛,二级指针是保存了这个地 ...
- C#中调用C++的dll的参数为指针类型的导出函数(包括二级指针的情况)
严格来说这篇文章算不上C++范围的,不过还是挂了点边,还是在自己的blog中记录一下吧. C++中使用指针是家常便饭了,也非常的好用,这也是我之所以喜欢C++的原因之一.但是在C#中就强调托管的概念了 ...
- C-指针,二级指针,二维数组作为函数参数使用,C语言链表(详解)
一级指针 int *p; //表示定义一个int型(4字节)的指针p &p //表示p自身的地址位置 p ...
- (C++)函数参数传递中的一级指针和二级指针
主要内容: 1.一级指针和二级指针 2.函数指针传递的例子 3.什么时候需要传递二级指针? 4.二级指针在链表中的使用 1.一级指针和二级指针 一级指针:即我们一般说的指针,就是内存地址: 二级指针: ...
- FreeRTOS信号量的封装函数参数是二级指针
1. 先看正确的封装方式,问题所在,为什么要用2级指针 void cissys_lockcreate(void** mutex) { //创建信号量,应该是互斥锁 *mutex = ((Semapho ...
随机推荐
- [AngularJS] Reusable directive, require from parent controller
Glorious Directives for Our Navigation NoteWrangler navigation has now been broken into two parts: t ...
- cocos2d-x读取xml(适用于cocos2d-x 2.0以上版本号)
为了能在cocos2d-x的文本标签中显示中文,一个是转换文件编码格式,还有一种就是读取utf-8格式的xml文件.我选择了后者,其原因大家可以去搜索一下cocos2d-x显示中文,希望可以你给答案. ...
- [C++知识点]2015.4.18
1.“static_cast/强制转换”后普通函数.虚函数的表现 A基类 B子类 实例aPtr:将B实例强制抓换得到的A实例 实例bPtr:将A实例强制转换得到的B实例 总结: (1) aPtr因为是 ...
- jquery循环遍历radio单选按钮,并设置选中状态
背景:自己在做项目过程中遇到的问题,现在记录一下. 需求:在ajax获取后台数据的之后,需要根据获取的数据对页面中的radio单选按钮进行选中状态设置 因为自身js功底欠佳,所以耽误了点时间,现在把方 ...
- C语言局部变量和全局变量问题汇总
1.局部变量能否和全局变量重名? 答:能,局部会屏蔽全局.要用全局变量,需要使用"::" 局部变量可以与全局变量同名,在函数内引用这个变量时,会用到同名的局部变量,而不会用到全局变 ...
- (译文)12个简单(但强大)的JavaScript技巧(一)
原文连接: 12 Simple (Yet Powerful) JavaScript Tips 我将会介绍和解析12个简单但是强大的JavaScript技巧. 这些技巧所有的JavaScript程序员都 ...
- 汉诺塔的问题:4个柱子,如果塔的个数变位a,b,c,d四个,现要将n个圆盘从a全部移到d,移动规则不变
四柱汉诺塔问题的求解程序.解题思路:如a,b,c,d四柱. 要把a柱第n个盘移到目标柱子(d柱),先把上层 分两为两部份,上半部份移到b柱,下半部分移到c柱,再把第n盘移到 目标柱子,然后,c柱盘子再 ...
- Service的一些使用
service服务一般主要是作为后台服务使用的,前台服务一般结合通知一起. service一般主要用作长期后台服务的,而且和Activity结合性不那么紧密, 一般如果需要频繁的更新UI主要是用Act ...
- android显示手机电量
package com.basillee.asus.demo; import android.app.Notification; import android.content.BroadcastRec ...
- windows下go开发环境部署 (sublime+gosublime+geocode)
1.下载git.go和sublime免安装版 go下载地址:http://pan.baidu.com/s/1hq1mrDM#path=%252Fgo%252F1.5.2sublime下载地址:http ...