清华大学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

相关文章

Go语言中make和new的区别及说明

《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

Go语言中nil判断的注意事项(最新推荐)

《Go语言中nil判断的注意事项(最新推荐)》本文给大家介绍Go语言中nil判断的注意事项,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1.接口变量的特殊行为2.nil的合法类型3.nil值的实用行为4.自定义类型与nil5.反射判断nil6.函数返回的

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C