我使用itoa()
函数将int
转换为string
,但它给出了一个错误:
undefined reference to `itoa'
collect2: ld returned 1 exit status
是什么原因?还有其他方法可以执行此转换吗?
使用snprintf
,它比itoa
更便携。
itoa is not part of standard C, nor is it part of standard C++;但是,许多编译器和相关库都支持它。
sprintf
的例子
char* buffer = ... allocate a buffer ...
int value = 4564;
sprintf(buffer, "%d", value);
snprintf
的例子
char buffer[10];
int value = 234452;
snprintf(buffer, 10, "%d", value);
这两个函数类似于fprintf
,但输出被写入数组而不是流。 sprintf
和snprintf
之间的区别在于snprintf
通过写入可以存储在buffer
中的最大字符数来保证没有缓冲区溢出。
像Edwin建议的那样,使用snprintf:
#include <stdio.h>
int main(int argc, const char *argv[])
{
int n = 1234;
char buf[10];
snprintf(buf, 10, "%d", n);
printf("%s\n", buf);
return 0;
}
如果您真的想使用itoa,则需要包含标准库头。
#include <stdlib.h>
我也相信,如果你在Windows上(使用MSVC),那么itoa
实际上是_itoa
。
见http://msdn.microsoft.com/en-us/library/yakksftt(v=VS.100).aspx
然后,再次,因为你从collect2
收到消息,你很可能在* nix上运行GCC。
看这个例子
#include <stdlib.h> // for itoa() call
#include <stdio.h>
int main() {
int num = 145;
char buf[5];
// convert 123 to string [buf]
itoa(num, buf, 10);
// print our string
printf("%s\n", buf);
return 0;
}
看到这个link有其他例子。
在继续之前,我必须警告你,itoa
不是ANSI函数 - 它不是标准的C函数。您应该使用sprintf
将int
转换为字符串。
itoa
有三个论点。
char *
变量,程序可能会崩溃,所以你应该传入一个正常大小的char数组,它会正常工作。该函数返回一个指向其第二个参数的指针 - 它存储了转换后的字符串。
itoa
是一个非常有用的功能,有一些编译器支持 - 很遗憾它不是所有人的支持,不像atoi
。
如果你仍然想使用itoa
,那么你应该如何使用它。否则,你有另一个选择使用sprintf
(只要你想要基数8,10或16输出):
char str[5];
printf("15 in binary is %s\n", itoa(15, str, 2));
更好地使用sprintf(),
char stringNum[20];
int num=100;
sprintf(stringNum,"%d",num);
使用snprintf
- 它是每个编译器都可以使用的标准配置。通过使用NULL, 0
参数调用它来查询所需的大小。最后为null分配一个字符。
int length = snprintf( NULL, 0, "%d", x );
char* str = malloc( length + 1 );
snprintf( str, length + 1, "%d", x );
...
free(str);
通常snprintf()
是要走的路:
char str[16]; // could be less but i'm too lazy to look for the actual the max length of an integer
snprintf(str, sizeof(str), "%d", your_integer);
您可以使用此功能制作自己的itoa
:
void my_utoa(int dataIn, char* bffr, int radix){
int temp_dataIn;
temp_dataIn = dataIn;
int stringLen=1;
while ((int)temp_dataIn/radix != 0){
temp_dataIn = (int)temp_dataIn/radix;
stringLen++;
}
//printf("stringLen = %d\n", stringLen);
temp_dataIn = dataIn;
do{
*(bffr+stringLen-1) = (temp_dataIn%radix)+'0';
temp_dataIn = (int) temp_dataIn / radix;
}while(stringLen--);}
这是一个例子:
char buffer[33];
int main(){
my_utoa(54321, buffer, 10);
printf(buffer);
printf("\n");
my_utoa(13579, buffer, 10);
printf(buffer);
printf("\n");
}
char string[something];
sprintf(string, "%d", 42);
void itos(int value, char* str, size_t size) {
snprintf(str, size, "%d", value);
}
..与call by reference合作。像这样使用它:
int someIntToParse;
char resultingString[length(someIntToParse)];
itos(someIntToParse, resultingString, length(someIntToParse));
现在resultingString
将持有你的C-String'。
类似于Ahmad Sirojuddin的实现,但语义略有不同。从安全角度来看,只要函数写入字符串缓冲区,该函数就应该“知道”缓冲区的大小并拒绝写入它的末尾。我猜它是你找不到itoa的部分原因。
此外,以下实现避免了两次执行模块/ devide操作。
char *u32todec( uint32_t value,
char *buf,
int size)
{
if(size > 1){
int i=size-1, offset, bytes;
buf[i--]='\0';
do{
buf[i--]=(value % 10)+'0';
value = value/10;
}while((value > 0) && (i>=0));
offset=i+1;
if(offset > 0){
bytes=size-i-1;
for(i=0;i<bytes;i++)
buf[i]=buf[i+offset];
}
return buf;
}else
return NULL;
}
以下代码都测试上面的代码并演示其正确性:
int main(void)
{
uint64_t acc;
uint32_t inc;
char buf[16];
size_t bufsize;
for(acc=0, inc=7; acc<0x100000000; acc+=inc){
printf("%u: ", (uint32_t)acc);
for(bufsize=17; bufsize>0; bufsize/=2){
if(NULL != u32todec((uint32_t)acc, buf, bufsize))
printf("%s ", buf);
}
printf("\n");
if(acc/inc > 9)
inc*=7;
}
return 0;
}