转自:http://blog.csdn.net/lickylin/article/details/7856779 感谢作者
最近在做远程升级的内容,通过实践才真正体会到不同拷贝函数的作用
char*strcpy(char *dest, const char *src);
其对字符串进行操作,完成从源字符串到目的字符串的拷贝,当源字符串的大小大于目的字符串的最大存储空间后,执行该操作会出现段错误。
int sprintf(char*str, const char *format, ...)
函数操作的源对象不限于字符串:源对象可以是字符串、也可以是任意基本类型的数据。主要是实现将其他数据类型转换为字符串
void *memcpy(void*dest, const void *src, size_t n)
实现内存的拷贝,实现将一块内存拷贝到另一块内存。该函数对源对象与目的对象没有类型现在,只是对内存的拷贝
但是在软件升级中,当接收到网络传送的升级内容后,进行数据拷贝时,最好使用memcpy来进行数据的拷贝。因为strcpy、sprintf进行拷贝时,当检查到源字符串中有’\0’即ascii码为00)时,即认为数据结束符,就会停止拷贝
- #include<stdio.h>
- #include<stdlib.h>
- #include<string.h>
- int main()
- {
- char str1[48];
- char str2[48];
- char str3[48];
- char test[48];
- int count = 0;
- FILE *fp1 = NULL;
- FILE *fp2 = NULL;
- FILE *fp3 = NULL;
- FILE *fp = NULL;
- memset(str1, 0, sizeof(str1));
- memset(str2, 0, sizeof(str2));
- memset(str3, 0, sizeof(str3));
- fp = fopen("/usr/local/CTEST/memtest.bin", "r");
- if(fp == NULL)
- {
- printf("open source file error \n");
- return -1;
- }
- fp1 = fopen("/usr/local/CTEST/memtest1.bin", "w");
- if(fp1 == NULL)
- {
- printf("open file1 for strcpy error\n");
- return -1;
- }
- fp2 = fopen("/usr/local/CTEST/memtest2.bin", "w");
- if(fp2 == NULL)
- {
- printf("open file2 for strcpy error\n");
- return -1;
- }
- fp3 = fopen("/usr/local/CTEST/memtest3.bin ", "w");
- if(fp3 == NULL)
- {
- printf("open file3 for strcpy error\n");
- return -1;
- }
- fread(test, 1, 48, fp);
- fclose(fp);
- printf("test is %s\n", test);
- strcpy(str1, test);
- fwrite(str1, 1, 48, fp1);
- sprintf(str2, "%s", test);
- fwrite(str2, 1, 48, fp2);
- memcpy(str3, test, 48);
- fwrite(str3, 1, 48, fp3);
- fclose(fp1);
- fclose(fp2);
- fclose(fp3);
- return 0;
- }
上述源文件memtest.bin中是一个升级文件的部分内容,升级文件中会有许多ascii码为00的数据,
上面是一段程序,首先打开该文件读取一部分内容(此处读48个字节),并存放在test中,然后使用strcpy、sprintf、memcpy分别拷贝到
str1、str2、str3中,并写入到文件memtest1.bin、memtest2.bin、memtest3.bin中,最后将这3个文件分别与memtest.bin对比
上图是memtest.bin与memtest1.bin的对比,可发现不相同,说明strcpy在拷贝的过程中,遇到ascii码为00时,就认为数据结束导致的。
上图是memtest.bin与memtest2.bin的对比,可发现不相同,说明sprintf在拷贝的过程中,遇到ascii码为00时,就认为数据结束导致的。
上图是memtest.bin与memtest3.bin的对比,可发现是相同相同,因为memcpy是对内存的拷贝,所以不会出现上面的问题
通过对比可以发现,经过sprintf、strcpy拷贝的数据遇到00值后就停止复制了,导致写入的文件与源文件的内容不符,如果使用这两个函数
进行拷贝,那升级以后就会导致系统无法启动,而经过memcpy拷贝后的文件memtest3.bin与源文件是相同的,所以在拷贝软件升级内容时尽量
使用memcpy,而不要使用strcpy、sprintf,否则可能会导致系统无法启动