C++中的memset用法詳解

memset簡介

memset是一個初始化函數,作用是將某一塊內存中的全部設置為指定的值。

void *memset(void *s, int c, size_t n); 
  • s指向要填充的內存塊。
  • c是要被設置的值。
  • n是要被設置該值的字符數。
  • 返回類型是一個指向存儲區s的指針。

需要說明的幾個地方

一、不能任意賦值

memset函數是按照字節對內存塊進行初始化,所以不能用它將int數組出初始化為0和-1之外的其他值(除非該值高字節和低字節相同)。
其實c的實際范圍應該在0~255,因為memset函數隻能取c的後八位給所輸入范圍的每個字節。也就是說無論c多大隻有後八位二進制是有效的

=================================================================================================
對於int a[4];
memset(a, -1, sizeof(a)) 與 memset(a, 511, sizeof(a)) 所賦值的結果一樣都為-1:
因為 -1 的二進制碼為(11111111 11111111 11111111 11111111);511 的二進制碼為(00000000 00000000 00000001 11111111);
後八位均為(11111111),所以數組中的每個字節都被賦值為(11111111)。
註意int占四個字節,例如a[0]的四個字節都被賦值為(11111111),那麼a[0](11111111 11111111 11111111 11111111),即a[0] = -1。

二、註意所要賦值的數組的元素類型

先來看兩個例子:
例一:對char類型的數組a初始化,設置元素全為’1’

int main(){
    char a[4];
    memset(a,'1',4);
    for(int i=0; i<4; i++){
        cout<<a[i]<<" ";
    }
    return 0;
}

在這裡插入圖片描述

例二:對int類型的數組a初始化,設置元素值全為1

int main(){
    int a[4];
    memset(a,1,sizeof(a));
    for(int i=0; i<4; i++){
        cout<<a[i]<<" ";
    }
    return 0;
}

在這裡插入圖片描述

1、首先要說明的第一點

 對於第二個程序,數組a是整型的,一般int所占內存空間為4個字節,所以在使用memset賦值時,下面的語句是錯誤的

int a[4];
memset(a,1,4);

由於memset函數是以字節為單位進行賦值的,所以上述代碼是為數組a的前4個字節進行賦值,那麼所得到的執行結果就隻能是:

在這裡插入圖片描述

正確的memset語句應為:

memset(a,1,16); //int所占內存為4字節的情況
memset(a,1,sizeof(a));

至於為什麼不是預期得到的1,將在下面的第二點進行說明。

當然,不同的機器上int的大小可能不同,所以最好用sizeof()函數

2、為什麼第一個程序可以正確賦值1而第二個不可以?

這就又回到瞭剛剛說的第一個問題,memset函數中隻能取c的後八位賦給每個字節

  • 第一個程序中,數組a是字符型的,字符型占據的內存大小就是1Byte,而memset函數也是以字節為單位進行賦值的,所以輸出正確。
  • 第二個程序中,數組a是整型的,整型占據的內存大小為4Byte,而memset函數還是按照字節為單位進行賦值,將1(00000001)賦給每一個字節。那麼對於a[0]來說,其值為(00000001 00000001 00000001 00000001),即十進制的16843009。

關於所要賦值的字符數的寫法

先來看一個示例:

#include<bits/stdc++.h>
using namespace std;

void fun1(int a[]){
    memset(a,-1,sizeof(a)); 
}

int main(){
    int a[6];
    fun1(a);
    for(int i=0; i<6; i++){
        cout<<a[i]<<" ";
    }
    return 0;
}

當數組作為參數傳遞時,其傳遞的實際上是一個指針,這個指針指向數組的首地址,如果用sizeof(a)函數得到的隻是指針的長度,而不是數組的長度。

解決方案:
在函數中加入數組長度參數,在傳遞前先獲取數組長度,然後將數組長度作為參數傳遞進去。

#include<bits/stdc++.h>
using namespace std;

void fun1(int a[], int len){
    memset(a,-1,len); 
}

int main(){
    int a[6];
    int len = sizeof(a);
    fun1(a,len);
    for(int i=0; i<6; i++){
        cout<<a[i]<<" ";
    }
    return 0;
}

具體用法實例

初始化數組

char str[100];
memset(str,0,100);

清空結構體類型的變量

typedef struct Stu{
	char name[20];
	int cno;
}Stu;
Stu stu1; 
memset(&stu1, 0 ,sizeof(Stu));

Stu stu2[10]; //數組
memset(stu2, 0, sizeof(Stu)*10);

此外,如果結構體中有數組的話還是需要對數組單獨進行初始化處理的。

總結

  memset函數在初始化處理時非常方便,但也有其局限性,比如要註意初始化數值,要註意字節數等等。當然,直接選擇用for循環或while循環來進行初始化也是可以的,隻不過memset更快捷一些。

到此這篇關於C++中的memset用法詳解的文章就介紹到這瞭,更多相關memset的用法內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: