普通数组的memset函数用法详细解读

普通数组才能用memset,vector不行

  • LeetCode上用memset初始化普通数组示例
    一般开数组最好开全局普通数组,即可以开大很大,又方便memset赋值
  • vector是个容器,vector千万不要用memset初始化,会出很大的问题
    memset会破坏vector的内部的结构,导致出现内存泄露的问题

最常用的memset赋值

头文件 <cstring>
memset(f, 0, sizeof(f));

一维数组 f[5]可以,两维数组 f[6][3]也可以memset。通过memset后每一个元素都是0

  • 0x3f(正无穷)
  • -0x3f(负无穷)
  • 0
  • -1

memset(f, 0x3f, sizeof(f)); 每个元素大小1,061,109,567
memset(f, -0x3f, sizeof(f)); 每个元素大小-1,044,266,559
够大,一般这么大你不会用到,而且相加不容易溢出


memset函数说明

头文件:
C:<memory.h>/<string.h>
C++:< cstring >
话说刚开始使用memset的时候一直以为memset是对每一个int赋值的,心里想有了memset还要for循环对数组进行初始化干嘛。
但其实memset这个函数的作用是将数字以单个字节逐个拷贝的方式放到指定的内存中去

memset(dp,0,sizeof(dp));

int类型的变量一般占用4个字节,对每一个字节赋值0的话就变成了“00000000 00000000 000000000 00000000” (即10进制数中的0)

memset(dp,-1,sizeof(dp));

赋值为-1的话,放的是 “11111111 11111111 11111111 11111111 ”(十进制的-1)
这样你可能以为如果你赋值1的话会让整个dp数组里的每一个int变成1,其实不然。

memset(dp,1,sizeof(dp));

以上代码执行后,dp数组的内容为 00000001 00000001 00000001 00000001 转化为十进制后不为1

我们在很多程序中都会看到这样的代码,

memset(a,127,sizeof(a));

127是什么特别的数字呢?通过基础的进制转换可以得知127的二进制表示是01111111,那么在dp数组里放的内容就是“01111111 01111111 01111111 01111111”,(10进制的2139062143),这样就实现了将数组里的全部元素初始化为一个很大的数的目的了,在最短路径问题以及其他很多算法中都是需要用到的。值得注意的是,int类型的范围为2^31-1,大约是2147483647的样子(如果我没有记错的话),所以初始化int类型的数组也可以使用127这个数值。

memset(a,128,sizeof(a)); 或memset(a,0x80,sizeof(a));

如果是128呢?因为128的二进制是10000000,那么放的内容就是10000000 10000000 10000000 10000000,经过计算可得这个数是-2139062144。这样就可以将数组初始化为一个很小的数了

memset(a,0x7F,sizeof(a));

它将arr中的值全部赋为2139062143,这是用memset对int赋值所能达到的最大值

memset(a, 0x3f, sizeof(a));

  • 0x3f3f3f3f的十进制是1061109567,也就是10^9级别的 (和0x7fffffff一个数量级),而一般场合下的数据都是小于10^9的,所以它可以作为无穷大使用而不致出现数据大于无穷大的情形。

  • 另一方面,由于一般的数据都不会大于10^9,所以当我们把无穷大加上一个数据时,它并不会溢出(这就满足了“无穷大加一个有穷的数依然是无穷大”),事实上0x3f3f3f3f+0x3f3f3f3f=2122219134,这非常大但却没有超过32-bit int的表示范围,所以0x3f3f3f3f还满足了我们“无穷大加无穷大还是无穷大”的需求。
    最后,0x3f3f3f3f还能给我们带来一个意想不到的额外好处:如果我们想要将某个数组清零,我们通常会使用memset(a,0,sizeof(a))这样的代码来实现(方便而高效),但是当我们想将某个数组全部赋值为无穷大时(例如解决图论问题时邻接矩阵的初始化),就不能使用memset函数而得自己写循环了(写这些不重要的代码真的很痛苦),我们知道这是因为memset是按字节操作的,它能够对数组清零是因为0的每个字节都是0,现在好了,如果我们将无穷大设为0x3f3f3f3f,那么奇迹就发生了,0x3f3f3f3f的每个字节都是0x3f!所以要把一段内存全部置为无穷大,我们只需要memset(a,0x3f,sizeof(a))。
    所以在通常的场合下,0x3f3f3f3f真的是一个非常棒的选择。

总结起来就是三句话
1 够大,一般这么大你不会用到
2 够大但不容易溢出
3 方便数组赋值

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

捡起一束光

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值