当前位置: 首页 > news >正文

并非从0开始的c++ day6

并非从0开始的c++ day6

  • 二级指针练习-文件读写
  • 位运算
    • 位逻辑运算符
      • 按位取反 ~
      • 位于(AND):&
      • 位或(OR): |
      • 位异或: ^
    • 移位运算符
      • 左移<<
      • 右移>>
  • 多维数组
    • 一维数组
      • 数组名
      • 一维数组名传入到函数参数中
      • 数组指针
        • 1、先定义出数组的类型,在通过类型定义出数组指针
        • 2、先定义出数组指针的类型,通过类型创建数组指针变量
        • 3、直接定义数组指针变量
    • 二维数组
      • 二维数组的定义
      • 二维数组做函数参数

二级指针练习-文件读写

需求:在代码根目录下,新建一个txt文件,输入若干行字符,读取并一个个输出出来

//获取文件中有效的函数
int getFileLines(FILE * file)
{if (file == NULL){return -1;}char buf[1024];int lines = 0;while (fgets(buf, 1024, file) != NULL){lines++;//printf("%s\n", buf);}//将文件的光标 置首fseek(file, 0, SEEK_SET);//SEEK_SET//SEEK_CUR//SEEK_ENDreturn lines;
}void readFileData(FILE * file,char ** pArray,int len)
{if (file == NULL){return;}if (pArray == NULL){return;}if (len < 0){return;}char buf[1024] = { 0 };int count = 0;while (fgets(buf, 1024, file) != NULL){//计算开辟每个字符串的大小int currentLen = strlen(buf)+1;//字符串最后字符改为\0buf[strlen(buf) - 1] = '\0';//开辟堆区空间char * currentP = malloc(sizeof(char) * currentLen);//将文件中读取的内容,放入到开辟的空间中strcpy(currentP, buf);//将开辟空间的数据,放入到我们维护的数组中pArray[count++] = currentP;//清空bufmemset(buf, 0, 1024);}
}//打印数组
void showFileData(char** pArray, int len)
{for (int i = 0; i < len; i++){printf("%d行的数据为:%s\n", i + 1, pArray[i]);}
}//释放堆区空间
void freeSpace(char** pArray, int len)
{for (int i = 0; i < len; i++){if (pArray[i] != NULL){printf("%s被释放了\n", pArray[i]);free(pArray[i]);pArray[i] = NULL;}}free(pArray);pArray = NULL;
}void test01() 
{//打开文件FILE * file = fopen("./test.txt", "r");if (file == NULL){printf("文件打开失败\n");return;}//获取文件有效函数int len = getFileLines(file);printf("文件的有效行数为:%d\n", len);char** pArray = malloc(sizeof(char*) * len);//将文件中的数据放入到pArray的数组中readFileData(file, pArray, len);//打印数组showFileData(pArray,len);//释放堆区空间freeSpace(pArray,len);pArray = NULL;//关闭文件fclose(file);file = NULL;
}

FILE * file = fopen(“./test.txt”, “r”);
为读取文件的命令,r代表读

在getFileLines读取了file指针后,指针已经发生了偏移,若没有重置,直接再次调用,会报错
故这里使用memset重置,SEEK的三个可选项分别对应当前位置(cur)、开头(set)、末尾(end)

txt文件中内容自带换行,故在print的时候不需要手动加\n来换行
若不需要文件自带换行,想自己换的话,需要在读取的时候,将字符串最后一个字符改为\0即可

free的个数要对应malloc的个数,free要先free指针底下的内容,再释放指针自身,要不然底下的东西会变成无法访问的垃圾

位运算

位逻辑运算符

按位取反 ~

一元运算符~ 将每一1变为0,将每个0变为1

void test01()
{int num = 2;printf("~num = %d\n", ~num);// -3//0000 0010~ 1111 1101源码 ~1000 0011 补码 
}

这样得到的结果为-3,因为数据经过取反补码,就是符号位转变后加一

位于(AND):&

二进制运算符&通过对两个操作数逐位进行比较产生一个新值。对于每个位,只有两个操作数的对应位都是1是结果才为1。

同真为真,其余为假

void test02()
{int num = 123;if ((num & 1 )== 0){printf("偶数\n");}else{printf("奇数\n");}
}

位或(OR): |

同假为假,其余为真

void test03()
{int num1 = 5;int num2 = 3;printf("num1 | num2 = %d\n", num1 | num2);
}

可运用于将某个位变为1

位异或: ^

同样为真,不同为假

a:00101110
b:11000111
r:11101001
a^b =r a^r= b b^r = a

void test04() {int num1 = 5;int num2 = 9;//实现两个数字交换num1 = num1 ^ num2;num2 = num1 ^ num2;num1 = num1 ^ num2;printf("num1 = %d\n", num1);printf("num2 = %d\n", num2);}

移位运算符

左移<<

(10001010)<<2
(00101000)

<< n 代表乘以2^n

void test05()
{int num = 20;printf("%d\n", num <<= 3);
}

右移>>

>> n 代表如果number非负,则number除以2^n

void test06()
{int num = 20;printf("%d\n", num >>= 1);
}

多维数组

一维数组

  • 元素类型角度:数组是相同类型的变量的有序集合
  • 内存角度:连续的一大片内存空间

数组名


void test01()
{int arr[5] = { 1,2,3,4,5 };printf("sizeof arr = %d\n", sizeof(arr));printf("%d\n", &arr);printf("%d\n", &arr+1);int* p = arr;p = p + 3;printf("%d\n", p[-1]);
}

//一维数组的名 是不是指向第一个元素的指针
//对数组名 取地址 可得步长为20
//第一种 对数组名称 sizeof
//第二种 对数组名称取地址 步长为整个数组长度
//除了这两种情况以外,一维数组名 指向数组第一个元素的指针

//指针常量 int * const p 常量指针const int * p
//前者为指针常量,要指针指向的值才能修改,后者为常量指针,指针指向的值不能改,但指针本身能改

//数组名 要是指针常量
//arr = NULL;//指针的指向是不可以修改的
//arr[0] = 100;//指针指向的值是可以修改的

//如果将指针偏移过,则访问的时候可以为负数

一维数组名传入到函数参数中

void printArray(int  arr[],int len)//int arr[] 等价于 int *arr
{for (int i = 0; i < len; i++)printf("%d\n", arr[i]);//等同于*(arr+i)
}void test02()
{int arr[5] = { 1,2,3,4,5 };int len = sizeof(arr) / sizeof(int);printArray(arr, len);
}

数组指针

1、先定义出数组的类型,在通过类型定义出数组指针

void test01()
{int arr[5] = { 1,2,3,4,5 };typedef int(ARRAY_TYPE)[5];//ARRAY_TYPE是一个有5个int元素的数组的类型ARRAY_TYPE* arrP = &arr;//*arrP ==== arrfor (int i = 0; i < 5; i++){printf("%d\n", (*arrP)[i]);}
}

2、先定义出数组指针的类型,通过类型创建数组指针变量

void test02()
{int arr[5] = { 1,2,3,4,5 };typedef int(*ARRAY_TYPE)[5];ARRAY_TYPE arrp = &arr;//可以使用,但是比较麻烦printf("%d\n", *arrp[0]);
}

3、直接定义数组指针变量

void test03()
{int arr[5] = { 1,2,3,4,5 };//语法:数组元素类型(*数组指针变量名称)[元素个数]int(* p)[5] = &arr;//*p == arrfor (int i = 0; i < 5; i++){printf("%d\n", (*p)[i]);}
}

二维数组

二维数组的定义

void test01()
{int arr[3][3] ={{1,2,3},{2,3,4},{4,5,6}};int arr2[3][3] = { 1,2,3,4,5,6,7,8,9 };int arr3[][3] = { 1,2,3,4,5,6,7,8,9 };
}

三种定义方法,第一种可读性最强

二维数组做函数参数

void printArray(int (*pArray)[3],int len1,int len2)//(*pArray)[3] 也可以表述为pArr[3][3]更加易读
{for (int i = 0; i < len1; i++){for (int j = 0; j < len2; j++){printf("%d ", pArray[i][j]);printf("%d ", *(*(pArray+i)+j));}printf("\n");}
}void test02()
{int arr[3][3] ={{1,2,3},{4,5,6},{7,8,9}};printArray(arr, 3, 3);//sizeofprintf("sizeof arr= %d\n", sizeof(arr));//取地址printf("%d\n", &arr);printf("%d\n", &arr + 1);int(* p)[3][3] = &arr;
}

sizeof获取整个二维数组占用内存大小
取地址步长为整个二维数组长度
除了这两种情况,二维数组名指向第一行数组的数组指针

http://www.lryc.cn/news/16705.html

相关文章:

  • PMP考前冲刺2.22 | 2023新征程,一举拿证
  • RxJava的订阅过程
  • 【2.22】MySQL、Redis、动态规划
  • 2年手动测试,裸辞后找不到工作怎么办?
  • Leetcode6. N字形变换
  • 将Nginx 核心知识点扒了个底朝天(十)
  • GPU显卡环境配置安装
  • CIMCAI super unmanned intelligent gate container damage detect
  • web概念概述
  • 编译原理笔记(1)绪论
  • MySQL(八)
  • steam搬砖项目,小投入高回报,可放大操作,(内附教学资料)
  • 华为OD机试真题Python实现【最多提取子串数目】真题+解题思路+代码(20222023)
  • day32 多线程(上)
  • 【flink】 各种join类型对比
  • 常用正则表达式
  • PMP考试有没有什么技巧可以介绍一下么?
  • 2022-2023年营销报告(B站平台) | 5大行业势态、流量大盘全景洞察
  • Python的异常与工具包
  • 基于SSM的婴幼儿商城
  • 2023年新能源汽车行业研究报告
  • 手写Promise方法(直击Promise A+规范)
  • GooglePlay SSL Error Handler
  • OpenStack手动分布式部署Keystone【Queens版】
  • AAPT2
  • kafka学习
  • 坐拥两条黄金赛道,爱博医疗未来必是星辰大海!
  • DEV C++的使用入门程序做算术运算
  • 华为OD机试真题Python实现【商人买卖】真题+解题思路+代码(20222023)
  • 随想录二刷(数组二分法)leetcode 704 35 34 69 367