1.memset

原型:   extern void *memset(void *buffer, int c, int count);

功能:   把buffer所指内存区域的前count个字节设置成字符 c。注意,memset始终按字节赋值,           

     说明: 返回指向buffer的指针。用来对一段内存空间全部设置为某个字符, memset可以方便的清空一个结构类型的变量或数组。   

     使用

          char a[10];int b[10];
memset(a, '\0', sizeof(a));
memset(b,0,sizeof(int)*10);
struct _test{ char s[10];int x;int y; };
struct _test st,st1[10];
memset(&st,0,sizeof(struct _test)); //很方便的清空结构体
memset(st,0,sizeof(struct _test)*10);

2.memcpy

原型:extern void *memcpy(void*dest,void*src,unsignedintcount); 
          

     功能: 由src所指内存区域复制count个字节到dest所指内存区域。始终按字节拷贝,不判断目标是否有足够的大小。             

     说明: src和dest所指内存区域不能重叠,函数返回指向dest的指针.可以拿它拷贝任何数据类型的对象。

使用

char a[10],b[5];
memcpy(b, a, sizeof(b)); /将a拷贝给b,*注意如果用sizeof(a),会造成b的内存地址溢出*/

3.strcpy

原型: extern char *strcpy(char *dest,char *src);

功能: 把src所指由NULL结束的字符串复制到dest所指的数组中。 注意,结束符号本身已经拷贝。              

       说明: src和dest所指内存区域不可以重叠且dest必须有足够的空间来容纳 src的字符串.返回指向dest的指针。                    

       使用

char a[100],b[50];
strcpy(a,b);//要注意a中的字符串长度(第一个‘\0’之前)是否超过50位,如超过,则会造成b的内存地址溢出。

定义

char   *strcpy(char   *strDest,const   char   *strSrc)
{
assert((strDest!=NULL)&&(strSrc !=NULL)) //判断指针是否合法,即分配内存,指向某块确定区域
char *address strDest; //记住目标地址的起始值
while((*strDest++=*strSrc++)!='\0') //先拷贝,后判断,这样就不用在拷贝完了后,再加一句
NULL; // *strDest = '\0'; -->即加一个结束符.因为字符串结束已拷贝了.
return address; //返回目标首地址的值。
}

可以看到,并不判断目标是否有足够大小。错误的使用如下:

int main(){
char s[]="123456789";
char d[]="1234"
strcpy(d,s);
cout<<d<<s<<endl;
}

输出为1234567896789。由于s,d的内存连续。故拷贝后结果为123456789/06789/0。此时访问了不可预知的地址。

strcpy和mempy的代码如下:

char * strcpy(char * dest, const char * src) // 实现src到dest的复制
{
  if ((src == NULL) || (dest == NULL)) //判断参数src和dest的有效性
  {       return NULL;
  }
  char *strdest = dest; //保存目标字符串的首地址
  while ((*strDest++ = *strSrc++)!='\0'); //把src字符串的内容复制到dest下
  return strdest;
}
void *memcpy(void *memTo, const void *memFrom, size_t size)
{
  if((memTo == NULL) || (memFrom == NULL)) //memTo和memFrom必须有效
return NULL;
  char *tempFrom = (char *)memFrom; //保存memFrom首地址
  char *tempTo = (char *)memTo; //保存memTo首地址
  while(size -- > 0) //循环size次,复制memFrom的值到memTo中
  *tempTo++ = *tempFrom++ ;
  return memTo;
}

4.三者区别 

memset   主要应用是初始化某个内存空间。用来对一段内存空间全部设置为某个字符,一般用于在对定义的字符串初始化为'\0'                

           memcpy   是用于copy源空间的数据到目的空间中。是用来做内存拷贝,可以用来拷贝任何数据类型的对象,可以指定拷贝的数据长度。

           strcpy   用于字符串copy,遇到‘\0’,赋值完结束符即结束。但是注意,'\0'已经被拷贝。

例子:

	char str1[]="123456789";
char str2[]="123456789"; memcpy(str1,str1+2,strlen(str1+2));
strcpy(str2,str2+2);

str1="345678989";

str2="3456789";

memset,memcpy,strcpy的使用与区别的更多相关文章

  1. C的memset,memcpy,strcpy 的区别 及memset memcpy memmove源码

    extern void *memcpy(void *dest,void *src,unsigned int count);#include <string.h>   功能:由src所指内存 ...

  2. memset,memcpy,strcpy

    http://www.cppblog.com/junfeng568/archive/2006/03/11/4022.html

  3. memcpy、memmove、memset及strcpy函数实现和理解

    memcpy.memmove.memset及strcpy函数实现和理解 关于memcpy memcpy是C和C++ 中的内存拷贝函数,在C中所需的头文件是#include<string.h> ...

  4. 逆向 string.h 函数库 memset、strcpy、strcmp 函数

    memset 函数 函数原型:void *memset(void *str, int c, size_t n) 主要功能:复制字符 c(一个无符号字符)到参数 str 所指向的字符串的前 n 个字符 ...

  5. C++:memset ,memcpy 和strcpy 的根本区别!

    #include <stdio.h> #include <stdlib.h> #include <string.h> #include <assert.h&g ...

  6. memset memcpy函数

    memset 函数 1.其头文件为: #include<memory> 或者#include<string> 2.原型    看清是对每个字节,不是其类型 void *mems ...

  7. memset,memcpy,memcmp用法

    void* memset(void *s, int ch, size_t n); 将s所指向的某一块内存中的前n个字节的内容全部设置为ch指定的ASCII值. 例如:memset(lpMyStruct ...

  8. strcpy, mencpy, memmove三者区别

    首先来看strcpy,目的是实现字符串的复制,这里需要注意几个点: 1.判断指针的有效性 2.将复制后的指针地址返回,为了支持链式操作 3.不要忘记将字符串最后一个自负'\0'复制给dest 4.注意 ...

  9. memset memcpy

    memset与memcpy的用法: void *memset(void *s,int c,size_t n)总的作用:将已开辟内存空间 s 的首 n 个字节的值设为值 cmemset可以方便的清空一个 ...

随机推荐

  1. [Kafka] - Kafka内核理解:消息存储机制

    一个Topic分为多个Partition来进行数据管理,一个Partition中的数据是有序.不可变的,使用偏移量(offset)唯一标识一条数据,是一个long类型的数据 Partition接收到p ...

  2. 智课雅思词汇---十五、前缀co-com-con-col-cor-是什么意思

    智课雅思词汇---十五.前缀co-com-con-col-cor-是什么意思 一.总结 一句话总结:前缀:co- 表示"共同", 通常放在元音词根前 1.前缀co-com-con- ...

  3. Win7使用之查端口,杀进程

    对 Win7 的 cmd 使用不熟练,这里记录下经常用到的命令吧! 查看所有端口占用情况:netstat -ano 查看指定端口占用情况:netstat -ano | findstr "80 ...

  4. 深入剖析Redis主从复制

    [http://sofar.blog.51cto.com/353572/1413024/]   [Redis 主从复制的内部协议和机制]   一.主从概述 Redis 支持 Master-Slave( ...

  5. Windows7 如何关闭系统更新

    我们点击开始菜单,找到控制面板这个选项,如图:  然后进入操作中心,如图:  然后选择如图所示的选项,如图:  然后选择更改设置选项,如图:  然后我们选择从不检查更新并点击确定按钮,如图:

  6. java多线程学习一

    声明:本篇博客是本人为了自己学习保存的心得,其内容主要是从大神——五月的仓颉的博客中学习而来,在此多谢大神五月的仓颉的分享,敬礼! 第一章:进程和线程的概念 进程:进程是操作系统中作为分配资源的基本单 ...

  7. vue自定义全局组件(自定义插件)

    有时候我们在做开发的时候,就想自己写一个插件然后就可以使用自己的插件,那种成就感很强.博主最近研究element-ui和axios的时候,发现他们是自定义组件,但是唯一有一点不同的是,在用elemen ...

  8. JavaScript如何处理解析JSON数据详解

    JSON (JavaScript Object Notation)一种简单的数据格式,比xml更轻巧. JSON 是 JavaScript 原生格式,这意味着在 JavaScript 中处理 JSON ...

  9. GEF入门实例_总结_04_Eclipse插件启动流程分析

    一.前言 本文承接上一节:GEF入门实例_总结_03_显示菜单和工具栏 注意到app目录下的6个类文件. 这6个文件对RCP应用程序而言非常重要,可能我们现在对这几个文件的理解还是云里雾里,这一节我们 ...

  10. 条款39:明智的使用private继承

    首先看一下private继承的法则:class之间的继承关系如果是private的话,那么编译器不会将一个derived对象自动当作为一个base class对象. 从base class继承而来的所 ...