/ 利用关键字 static 修饰的变量 叫做静态变量 并且存储在静态区

// 特点如果你不给初值的话  默认 0

// 特点只初始化一次

// main函数 上面  是全局静态变量

//  main函数 里面  叫局部静态变量  作用域是大括号内

void function(){

    int a = 5;

    int b = 6;

    printf("%d,%d",a,b);

}


int main(int argc, const char * argv[]) {

    //  内存划分为5个区域

    //  栈区 堆区 静态区 常量区 代码区

    //  内存地址 由高到低

    //  学习目标 按顺序记忆

#pragma mark  -- 代码区


    // 电脑的存储 都是以二进制数据进行存储的

    // 咱们写得代码 会被系统转化成二进制数据 存储代码区

    

    

    

#pragma mark  -- 常量区 

    // 常量区 存储的常量特点

    // h 更改成 w

//    char *str = "zhuang";

    // 常量 是程序运行期间 不能改变量

//    str = "haha"; // 指针的赋值 就是重指向

//    char str[] = "zhuang";

//    str[1]='w';

    // 数组名字  地址 数组首元素的地址

    // 数组首元素的常量地址

    // 是把常量字符串  拷贝到栈区

    

    

    

#pragma mark  -- 静态区

    

    // 静态区: 存储两种变量

    // 1. 静态变量

    // 2. 全局变量

    // 3. 静态区变量 保存周期:直到程序运行结束 静态变量才会被释放(伴你到永生)


    

#pragma mark  -- 栈区

    

//    int num1 = 5;

//    int num2 = 10;

//    int num3 = 15;

//    printf("%p\n",&num1);

//    printf("%p\n",&num2);

//    printf("%p\n",&num3);

    

    // 栈区有多大?

    // 大概7M-8M

    

    

//    char str[8187 * 1024] = {0};

    // 出栈 入栈的规则

    // 先入栈的 后出栈  先入栈的  在栈低

    // 入栈  可以叫 压栈

    // 出栈  的顺序  从栈顶开始 出栈

//    int num1 =5;

//    int num2 = 10;

//    int num3 = 15;

//    function();

    // 之所以栈区容量不是特大  又不出现奔溃的现象 是因为 栈区 频繁进行 出栈 和入栈 只要一次性 不把栈空间堆满  不会轻易出现崩溃

    // 定义变量的时候  切记 给初值 避免出现莫名其妙的问题

    

    

#pragma  mark  -- 堆区

    // 堆区 是程序员开辟空间 是程序员释放空间

    // 手动开辟空间 手动释放空间

    // 堆区空间 大概就是咱们内存全部空间

    

    

    // 开辟空间函数

    // void * 表示无类型指针 可以转换成任何类型的指针

    // void *malloc(<#size_t#>)

    // size 开辟空间大小 开辟多少字节空间

    // 给整型指针p 指向的位置  开辟了4个字节的 堆内存空间

//    int *p = malloc(4);

//    *p = 10;

//    printf("%d \n",*p);

//    

    

    

//    char *str = malloc(8);

//    strcpy(str, "wang"); // 正确的

//    strcpy(str, "wangwangwang");// 错误的, 开辟多少空间  就要用多少空间 不要超过开辟的空间

    

    

//    printf("%s",str);

//    int *p1 = malloc(4);

//    *p1 = 5;

//    short *p2 = malloc(4); //  开辟多少空间  最好你就用多少

//    *p2 = 10;

//    *(p2 + 1) = 15;

//    

    

    

    

    // 释放空间函数

    // free(<#void *#>)

    // 1 .标记删除  不会抹去该空间存储的数据  只会把这块空间标记为可用

    // 2. 把指针置为空

    // 开辟空间

//    int *p = malloc(4);

//    // 使用空间

//    *p = 10;

//    printf("%d\n",*p);

//    // 释放空间

//    free(p);

//    p =NULL;


//    int *p = malloc(sizeof(int) * 5);

//    for (int i = 0; i < 5; i++) {

//        *p = i + 1;

//        p++;

//    }

//    //  指针地址发生变化  释放了本不属于你开辟的区域 这时候程序崩溃

//    

//    

//    // 修改  把指针移回去

//    for (int i = 0; i < 5; i++) {

//        p--;

//    }

//    free(p);

//    p = NULL;

//    

//**********************************************************************

//    char str[] ={"dj5s5f"};

//    // 计算出有多少个数字

//    int count = 0;

//    for (int i  = 0; i < strlen(str); i++) {

//        if (str[i] >= '0' && str[i] <= '9') {

//            count++;

//        }

//    }

//    char *strValue = malloc(count + 1);

//    int index = 0;

//    for (int i = 0; i < strlen(str); i++) {

//        if (str[i] >= '0' && str[i] <= '9' ) {

//            strValue[index] = str[i];

//            index++;

//        }

//    }

//    strValue[index] = '\0';

//    printf("%s",strValue);

//    free(strValue);

//    strValue = NULL;

//    

//*******************************************************************

 

    

//    char *words[3] = {0};

//    char str[] = {0};

//    for (int i = 0; i < 3; i++) {

//        printf("请输入一个字符串");

//        scanf("%s",str);

//        printf("%lu",strlen(str));

//        words[i] = malloc(strlen(str) + 1);

//        // 保存字符串

//        strcpy(words[i], str);

//        

//    }

//    

//    

//    

//    for (int i = 0; i < 3; i++) {

//        printf("%s",words[i]);

//        free(words[i]);

//        words[i] = NULL;

//    }

//

// ************************************************************

    

    

    

    

    

    

    

    

    

    

    

    // 其他分配堆内存函数

    // 在堆内存当中 开辟 n * size 个空间  并且把开辟的内存空间清零

    // 因为有一个清零的过程  所有效率偏低

    // calloc(int n , size);

    

    

    

    // 重新分配函数

    // realloc(原来分配的地址, 新空间大小)

    // 情况1: 如果原来分配的地址  可以扩充空间 那么就在原地址扩充

    // 情况2: 如果原来分配的地址  不能扩充了  那么系统会重新分配一个空间  并且把原来地址存储的数据拷贝到新空间里 然后系统自动释放原地址的空间

//    

//    int *p_old = malloc(10);

//    int *p_new =realloc(p_old, 15);

//    printf("%p %p\n",p_old,p_new);

//    

//    

    

    

    

   // 内存操作函数

   // 把开辟的空间  多少字节  重置成C

   // memset(开辟的空间的指针, int c, 字节数)

   // 用途: 把开辟好的空间  清零

//    int *p = malloc(sizeof(int) * 4);

//    for (int i = 0; i < 4; i++) {

//        p[i]=i + 5;

//        

//    }

//    // 清零

//    memset(p, 0, sizeof(int) * 4);

//    for (int i = 0; i < 4; i++) {

//        printf("%d",p[i]);

//    }

    

    

    // 内存拷贝函数

    // 从来源  拷贝 目的地  多个字节

    // memcmp(目的地, 来源, 字节数)

    

    

    

    

    

    // 内存比较函数

    // 两个地址  按字节  进行比较  返回第一个不同的差值

    // memcmp(指针1, 指针2, 字节数)