清华大学C++语言程序设计(第六单元随堂笔记指针)

本文主要是介绍清华大学C++语言程序设计(第六单元随堂笔记指针),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

指针

指针是c++从c语言中继承过来的极其重要的数据类型,它提供了一种极为直接的地址操作手段。

指针变量的声明

指针也是一种数据类型,具有指针类型的变量称为指针变量,指针变量是用来储存内存单元的地址的。

声明指针的语法是:

数据类型 *标识符

”*“表示声明的是一个指针类型的变量。数据类型可以是任意类型的,指的是指针所指向的对象的类型。

如:

int * ptr;

”*“指针运算符

”&“取地址符 ”&”出现在变量声明的左边时表示的是引用。 “&”放在等号右边或者是在执行语句中作为一元运算符出现时。表示取对象的地址。

可以声明指向常量的指针,但是不能通过指针来改变所指对象的值,但是指针本身可以改变,可以指向另外的对象。

int a;
const int *pi=&a;
int b;
pi=&b;   //pi的值本身可以改变
*pi=1;  //不能通过更改pi来更改pi所指值的对象

指针类型的常量

int *const p2=&a;
p2=&b;   //错误,p2是指针常量但不能改变

指向const常量的指针

#include<iostream>
using namespace std;
int main() {const int a = 2;int *ptr = &a;   //报错int类型的指针不能兼容const int数据}

###上边两个例子好好对比一下,可以看出来这两个区别

指针的运算

指针是一种数据类型。与其他数据类型相似,指针变量也可以参加部分运算,包括算数运算,关系运算,和赋值运算。

指针可以进行加减运算,但是运算的规则是比较特殊的。p1+n1它表示的是指针p1当前所指位置的前n1个数的地址。它可以写作:p1[n1]同理也有减法。

空指针的指定

int *p;
p=0;    //将p设为空指针,不指向任何地址
或者
p=NULL;

#如果不便于用一个有效的地址给一个指针变量赋初值,那么用当前0作为它的初值,从而避免指向一写极其神奇的值

用指针处理数组

#include<iostream>
using namespace std;
int main() {int a[10] = { 1,2,3,4,5,6,7,8,9,10 };for (int *i = a; i < (a + 10); i++) {cout << *i << endl;}
}

指针数组

语法形式:

数据类型 *数组名[下标表达式];

下列语句:

int *pa[3]; //说实在的这个等于一个二维数组

**注意:**由于指针数组的每一个元素都是一个指针,必须先赋值后引用,因此,声明数组之后,对指针元素赋值是必不可少。

例:6-8 利用指针数组输出单位元素

#include<iostream>
using namespace std;
int main() {int line[] = { 1,2,3,4,5 };int line1[] = { 1,2,3,4,5 };int line2[] = { 1,2,3,4,5 };int line3[] = { 1,2,3,4,5 };int *array[4] = { line, line1,line2,line3 };for (auto e : array) {for (int *i = e; i < (e+5); i++) {cout << *i << " ";}cout << endl;}
}

在这里插入图片描述

用指针作为函数参数

例:分离整数小数

#include<iostream>
using namespace std;
void splitfloat(float x, float &intpart, float &fracpart) {intpart = static_cast<int>(x);fracpart = x - intpart;
}
void splitfloat2(float x, float *intpart, float *fracpart) {*intpart = static_cast<int>(x);*fracpart = x - *intpart;
}
int main() {float intpart;float fracpart;splitfloat(2.8, intpart, fracpart);cout << "整数部分为:" << intpart << ",小数部分为:" << fracpart << endl;float *intpart1=NULL;float *fracpart1=NULL;splitfloat2(3.2, intpart1, fracpart1);cout << "整数部分为:" << *intpart1 << ",小数部分为:" << *fracpart1;

#这两个函数可以对比一下区分一下这个区别

指针型函数

指针型函数的一般定义形式是:

数据类型 *函数名(参数表){

函数体;

}

这样可以返回一个指针函数

指向函数的指针

在程序执行的过程中,不仅数据需要占据内存空间,执行程序的代码也被调入内存并且占据一定的空间。没以恶搞函数都有一个函数名(除了lamba函数除外),实际这个函数名就表示代码在内存中的起始地址。

声明一个函数指针时,也需要说明函数的返回类型,第一个圆括号中的内存指明一个函数指针的名称,形参数列表,其一般语法如下:

数据类型 (*函数指针名)(形参表)

给复杂的函数起别名:

###*typedef int(fuction1)(int,int);

fuction1 fuc1;####

例:指向函数的指针的应用

#include<iostream>
using namespace std;
typedef int(*fuction1)(int,int);
int sum(int x, int y) {return x + y;}
int dif(int x, int y) {return x - y;
}
int main() {fuction1 fuc1;int(*func2)(int, int);fuc1 = sum;int x=1, y=2;cout << "第一个func函数" << fuc1(x, y)<< endl;fuc1 = dif;cout << "第二个func函数" << fuc1(x, y)<< endl;func2 = sum;cout << "func2函数输出结果" <<func2(x,y) <<endl;
}

在这里插入图片描述

对象指针

一般语法格式:

类名 *对象指针名;

例:

point &p1;
point p2;
pi=&p2;

就像通过对象名访问对象的成员一样,使用对象指针一样可以方便的访问对象成员,语法形式为:

对象指针->成员名

例:利用指针去访问成员

#include<iostream>
using namespace std;
class Point {
public:Point(float x1, float y1) {x = x1;y = y1;}float getx() {return x;}float gety() {return y;}void change(int x1, int y1) { x = x1, y = y1; };
private:float x;float y;
};
int main() {Point *p1;Point p2(1,2);p1 = &p2;p1->change(3, 4);cout << "点更改后的值(" << p2.getx() << "," << p2.gety() << ")";
}

在这里插入图片描述

前向引用的另一个解决方法:

class Fred;
class BARNEY{Fred *x;   //错误,fred的定义不完善
}
class Fred{BARNEY y;
}改后class Fred;
class BARNEY{Fred *x;   
}
class Fred{BARNEY y;
}
//这个如果换成一个指针的话,就可以解决了。

this指针

this指针是一个隐含于每一个类的非静态成员函数中的特殊指针(包括构造函数和析构函数)它用于指向被成岩函数操作的对象。

而类的成员函数调用,私有函数成员的过程可以看成一个this->私有函数成员,而成员函数可以看成多了一个参数,如:点类中 getx()可以看作 getx(this->x)

this指针明确指出了成员函数当前操作的数据所属的对象。实际可以看成this指针为类的隐藏参数。

##当局部函数和类成员函数重名是可以通过this调用来唯一标识

Point(int x,int y){this->x=x;this->y=y;
}

指向类的非静态成员的指针

这个和指向对象的指针一样,不在细细阐述来个例子吧。

#include<iostream>
using namespace std;
class Point {
public:Point(float x1, float y1) {x = x1;y = y1;}float getx() {return x;   }float gety() {return y;}void change(int x1, int y1) { x = x1, y = y1; };static void show() { //静态函数成员很标准的一个特征是不能出现私有的x和ycout <<"这是一个静态成员函数";}private:float x;float y;
};int main() {void (*funcptr)() = Point::show;float (Point::*funcptr)()  = &Point::getx;//这两种成员函数的指针需要好好考虑一番}

动态分配内存

简而言之就是堆对象的操作。

动态分配内存的两个十分重要的运算符:new, delete

运算符new的功能是动态分配内存,或者称为动态创建堆对象,语法格式为:
new 数据类型 (初始化参数);

该语句在程序运行的过程中申请分配用于存放只当类型数据的内存空间,并且根据初始化参数列表中给出的值进行初始化。如果申请内存成功,new运算便会返回一个指向新分配内存首地址的类型的指针,可以通过这个指针对堆对象进行访问;

如:

int *point;
point=new int[2];  //申请一个int内存,并把初值设为2
point=new int;   //分配内存空间但是不希望其有初值
point=new int();  //用零对该对象初始化

#注意:#在用new建立一个类的对象时,如果该类存在在用户定义的默认构造函数,则“new T”和“new T()”的效果是相同的都会调用这个默认构造函数。但,若调用用户未定义的默认构造函数,使用“new T”创建对象时,会调用系统默认生成的隐含构造函数,若用“new T()”系统会额外未基本数据类型和指针类型成员用0赋值,而且这一过程是递归的。也就是说,如果该对象的某个成员对象也没有用户定义的默认构造函数,那么对该成员对象的基本数据类型和指针类型的成员,同样也会被以0赋初值。#

运算符delete用来删除由new建立的对象,释放指针所指向的内存空间。格式为:

delete 指针名;

如果删除的是对象,则该对象的析构函数会被调用。对于new创建的对象,只能使用delete进行一次删除操作,反之会导致运行出错。

#注意:new对象分配的内存,必须用delete删除加以释放,否则可能会导致动态分配的内存无法回收,使程序占的内存越来越大(是不是可以利用这一点创建一个占内存的病毒?)

使用new也可以创建数组类型的对象,这时候需要给出数组的结构说明,用new运算创建一维数组的语法形式:
new 类型名 [数组长度];

#这个后面也可以加小括号但是括号不能加任何参数,加小括号意味这全部以0赋值#

用new创建的数组,用delete删除时在指针前面加“[]",格式如下:

delete [] 指针名

深复制和浅复制

简单而说深复制和浅层复制就是由指针指向而产生的问题,如果用指针在类里面,会出现复制的类的指针仍然指向原先的地址,但是,实际上,应该指向的是一个新分配的地址,这就是浅复制和深复制的区别。

解决这样的问题的方法,就是在构造函数中,要为指针类型的成员,分配专门的空间。以这条规则构建的复制,称作为深复制!

Test(const Test& C){a=C.a;str=new char[strlen(C.str)+1];   //给str分配一个新的内存空间strcpy(str,C.str);               //把str给c.str}~Test(){delete []str;                  //把str对象删除}

string类和vector类

这两个类看开发者文档即可,开发者文档才是yyds

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

一个关于vector很好的博客

这篇关于清华大学C++语言程序设计(第六单元随堂笔记指针)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/382947

相关文章

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C语言进阶(预处理命令详解)

《C语言进阶(预处理命令详解)》文章讲解了宏定义规范、头文件包含方式及条件编译应用,强调带参宏需加括号避免计算错误,头文件应声明函数原型以便主函数调用,条件编译通过宏定义控制代码编译,适用于测试与模块... 目录1.宏定义1.1不带参宏1.2带参宏2.头文件的包含2.1头文件中的内容2.2工程结构3.条件编

Go语言并发之通知退出机制的实现

《Go语言并发之通知退出机制的实现》本文主要介绍了Go语言并发之通知退出机制的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、通知退出机制1.1 进程/main函数退出1.2 通过channel退出1.3 通过cont

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

Go语言编译环境设置教程

《Go语言编译环境设置教程》Go语言支持高并发(goroutine)、自动垃圾回收,编译为跨平台二进制文件,云原生兼容且社区活跃,开发便捷,内置测试与vet工具辅助检测错误,依赖模块化管理,提升开发效... 目录Go语言优势下载 Go  配置编译环境配置 GOPROXYIDE 设置(VS Code)一些基本

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态