• C语言有关内存的几个疑问,以memset以及memcpy为例


    memset简介
    memset是一个初始化函数,作用是将某一块内存中的全部设置为指定的值。

    void *memset(void *s, int c, size_t n); 
    
    
    • 1
    • 2
    1. s指向要填充的内存块。
    2. c是要被设置的值。
    3. n是要被设置该值的字符数。
    4. 返回类型是一个指向存储区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。

    二、注意所要赋值的数组的元素类型

    不同类型的内存赋值,看两个例子:

    int main(){
        char a[4];
        memset(a,'1',4);
        for(int i=0; i<4; i++){
            cout<<a[i]<<" ";
        }
        return 0;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    输出为 1 1 1 1

    int main(){
        int a[4];
        memset(a,1,sizeof(a));
        for(int i=0; i<4; i++){
            cout<<a[i]<<" ";
        }
        return 0;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    输出为(00000001 00000001 00000001 00000001)十进制的16843009,
    参考https://blog.csdn.net/weixin_44162361/article/details/115790452
    对于汇编语言来说,不论是memset(a,‘1’,4);或者memset(a,1,sizeof(a));1在汇编中的存储方式都是31h,因此对于不同的数据来说,每个内存中保存二进制数值是相同,都是31h,只能通过变量类型来区分是整数1还是字符1,疑问在于,它是如何区分的?

    例如将上述程序做修改,将第二个memset语句修改为第一个语句,即:

    int main() {
        int a[4];
        memset(a, '1', 4);
        for (int i = 0; i < 4; i++) {
            cout << a[i] << " ";
        }
        return 0;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    对于汇编来说,依然表示为
    int a[4];
    memset(a, ‘1’, 4);
    005324B2 push 4
    005324B4 push 31h
    005324B6 lea eax,[a]
    005324B9 push eax
    005324BA call _memset (0531181h)
    005324BF add esp,0Ch

    char a[4];
        memset(a, '1', 4);
    
    • 1
    • 2

    相同,
    char类型需要用ascii表,整数类型不需要,’1‘对应的ascii为31h,所以会显示31h;

  • 相关阅读:
    新版本Android Studio logcat日志过滤提示
    Adaptive AUTOSAR CM模块介绍(三)
    JavaSE——多线程中lock和synchronized的区别
    fMRI时变功能连接的数据和模型考虑
    最大连续子数组
    技术分享 | App测试时常用的adb命令你都掌握了哪些呢?
    fastposter v2.8.4 发布 电商海报生成器
    关于学生使用flowus的一些事
    React router6--路由表多级路由嵌套的默认子路由问题
    ATA-2168高压放大器用途有哪些方面
  • 原文地址:https://blog.csdn.net/zhangxiaio1/article/details/125414736