初始化函数 void * memset ( )

形式:void * memset(void * s, int ch, size_t n);

  • memset() 函数常用于非常量的内存空间初始化。

本质: 用来对一段内存空间全部设置为某个字符,一般用在对定义的字符串进行初始化。
方式: 将s中当前位置后面的n个字节 (typedef unsigned int size_t )用 ch 替换并返回 s,ch只有最低的字节起作用。
优点:
1. 对较大的结构体或数组清零的最快方法 。
2. memset可以方便的清空一个结构类型的变量或数组。

#include <iostream>
#include <cstring>
using namespace std;
int main() {
    char a[5];
    memset(a,'1',5;
    for(int i=0;i<5;i++)
        cout<<a[i]<<"";
    system("pause");
    return 0;
}

而,如下程序想把数组中的元素值设置成1,却是不可行的

#include <iostream>
#include <cstring>
#include <windows.h>
using namespace std;
int main()
{
    int a[5];
    memset(a,1,20; //也等价于memset(a,1,sizeof(a));.
    for(int i=0;i<5;i++)
        cout<<a[i]<<"";
    system("pause");
    return 0;
}

问题是:第一个程序为什么可以,而第二个不行?

答:

  • 因为第一个程序的数组a是字符型的,字符型占据内存大小是1Byte,而memset函数也是以字节为单位进行赋值的,所以你输出没有问题。
  • 而第二个程序a是整型的,使用 memset还是按字节赋值,这样赋值完以后,每个数组元素的值实际上是0x01010101即十进制的16843009。
  • 如果用memset(a,1,20),就是对a指向的内n存的20个字节进行赋值,每个都用数1去填充,转为二进制后,1就是00000001,占一个字节。一个int元素是4字节,合一起是0000 0001,0000 0001,0000 0001,0000 0001,转化成十六进制就是0x01010101,就等于16843009,就完成了对一个int元素的赋值了。

所以用memset对非字符型数组赋初值,除了令其等于1或0,其他是不可取的 !!!

另例:

#include<iostream>
#include<cstring>
using namespace std;
int main()
{
	const char *s = "GoldenGlobalView";
	memset(s, 'G', 6);                
	/*这里没有问题,可以编译运行,单步运行到这里会提示内存访问冲突,
	肯定会访问冲突,s指向的是不可写空间。*/
	printf("%s", s);
	getchar();
	return 0;
}

以上例子出现内存访问冲突应该是因为s被当做常量放入程序存储空间,menset( ) 只能用于对非常量的内存空间初始化。
如果修改为 char s[]="Golden Global View";则没有问题了。

  • 对于结构体
    • 有一个结构体Some x,可以这样清零: memset(&x,0,sizeof(Some));
    • 如果是一个结构体的数组Some x[10],可以这样:memset(x,0,sizeof(Some)*10);
发布了10 篇原创文章 · 获赞 0 · 访问量 365

猜你喜欢

转载自blog.csdn.net/Cabbage_W/article/details/104619027