C++ 中指针和数组相关操作符的优先级
最编程
2024-10-01 20:07:57
...
概述
本文深入介绍了与指针和数组相关的运算符优先级,利用代码示例展示了当左结合和右结合运算符同时存在时的结合方式,同时也演示了如何使用()来强制人为指定结合顺序。
指针、数组相关的运算符优先级
下表展示了相关运算符的优先级,有4个级别,同级别内的运算符按照结合性依次调用。这4类也是所有运算符中优先级最高的4档,其它的运算符优先级都比它们低:
优先级 | 运算符 | 描述 | 结合性 |
---|---|---|---|
1 | :: | 作用域解析 | 左结合 |
2 | () [] . -> |
强制运算结合,函数形参列表 数组元素下标访问 类的成员访问 类指针的成员访问 |
右结合 |
3 | (int) * & |
强制转换 指针解引用 变量取地址 |
左结合 |
4 | .* ->* |
类的成员指针 类指针的成员指针 |
左结合 |
容易产生困惑的、需要仔细进行优先级判断的往往是一个左结合加一个右结合,例如:
*ptr[]
(int)a()
&class->data
obj->*fun()
请记住一个重要方法:当我们需要改变运算符的结合顺序(C++默认的优先级不是我们的意愿)时,可以通过添加()来人为强制指定优先顺序,因为()是除了::以外具有最高优先级的一类运算符。
简单例子:以[]和*为例探讨运算符结合规律
下面的p1, p2是数组,p3是指针:
int *p1[2]; // p1是一个数组,元素个数为2,每个元素为(int*)
int *(p2[2]); // 等价于*p2[2],p2是一个数组
int (*p3)[2]; // p3是一个指针,指向一个int数组,这个int数组的元素个数必须是2!
因此只要记住两点即可:
- **[]的优先级高于* **:即
*p1[2]
和*(p1[2])
等价。 - 这个优先级同时适用于定义语句(*为指针定义符)和执行语句(*为解引用符)中:
int *p1[2]; // 定义语句:先看[]:p1是一个数组,元素个数为2,每个元素为(int*)。等价于*(p1[2])
int (*p2)[2]; // 定义语句:先看*: p2是一个指针,指向一个int数组,这个int数组的元素个数必须是2!
cout << "*p1[0] = " << *p1[0] << endl; // 执行语句:先看[]:先取第0个元素,再解引用。等价于*(p1[0])
cout << "(*p2)[0] = " << (*p2)[0] << endl; // 执行语句:先看*:先解引用,再取第0个元素
完整示例:
#include <iostream>
using namespace std;
int main(){
// []的优先级高于*,因此下面的p1是数组,p2是指针:
int *p1[2]; // p1是一个数组,元素个数为2,每个元素为(int*)。等价于*(p1[2])
int (*p2)[2]; // p2是一个指针,指向一个int数组,这个int数组的元素个数必须是2!
int a = 1, b = 2;
int c[2] = {4,5};
p1[0] = &a;
p1[1] = &b;
p2 = &c;
cout << "*p1[0] = " << *p1[0] << endl;
cout << "*p1[1] = " << *p1[1] << endl;
cout << "*(p1[0]) = " << *(p1[0]) << endl; // 与上面两条等价
cout << "*(p1[1]) = " << *(p1[1]) << endl;
cout << "(*p2)[0] = " << (*p2)[0] << endl;
cout << "(*p2)[1] = " << (*p2)[1] << endl;
return 0;
}
输出:
*p1[0] = 1
*p1[1] = 2
*(p1[0]) = 1
*(p1[1]) = 2
(*p2)[0] = 4
(*p2)[1] = 5
复杂例子:探讨当左结合和右结合运算符同时存在时如何界定优先级
下面的例子比较复杂,需要耐心仔细阅读和体会。如果这个例子能搞清楚,那么相信你对运算符优先级的理解将会上升一个档次。
这个例子研究了当左结合和右结合运算符同时存在时的结合顺序,同时也演示了可以使用()强制人为指定结合顺序:
#include <iostream>
#include <string>
using namespace std;
class Student{
public:
Student(string name, int id):_name(name),_id(id){}
void printInfo(){
cout << "I am a student. My name is " << _name << ". My id is " << _id << endl;
}
void operator()(){
printInfo();
}
protected:
string _name;
int _id;
};
class Student2 : public Student{
public:
Student2(string name, int id):Student(name, id){}
void printInfo(){
cout << "I am a super Student!!! " << endl;
}
void operator()(){
cout << "I am Student2!!!" << endl;
}
};
struct StudentWrapper{
Student* _ps;
StudentWrapper(Student* ps):_ps(ps){}
Student* operator()(){return _ps;}
};
int main(){
// .和(), ->和()平级:从左向右
cout << "-----------------1------------------" << endl;
Student s1("Bob",101), s2("Jack", 102);
Student *ps1 = new Student("Eric",103);
s1.printInfo();
s2.printInfo();
ps1->printInfo();
// .高于*:先结合.
cout << "-----------------2------------------" << endl;
// 下面这条语句报错:先调用.printInfo(),再*,因此会报错
// *ps1.printInfo(); // error: request for member 'printInfo' in 'ps1'
(*ps1).printInfo();
// .和()高于*:先结合()和.(从右向左),最后结合*
cout << "-----------------3------------------" << endl;
StudentWrapper sw(ps1);
// 下面这条语句报错:先结合sw(),再结合.printInfo(),最后结合*,因此会报错
// *sw().printInfo(); // error: request for member 'printInfo' in 'sw.StudentWrapper::operator()()'
(*sw()).printInfo(); // correct:先sw(),再*sw(),再(*sw()).printInfo()
// 下面这条语句报错:先结合sw(),再结合(),最后结合*,因此会报错
// *sw()(); // error: expression cannot be used as a function
(*sw())(); // correct:先sw(),再*sw(),再(*sw())()
// (int)和()/[]:先结合()和[],再强转
cout << "-----------------4------------------" << endl;
Student2 ss("Alice", 999), sss("Jason", 998), ssArray[2] = {ss, sss};
ss(); // 调用Student2::operator()
// 下面这条语句报错,因为会先结合ss(),再强制转换
// (Student)ss(); // error: invalid use of void expression
((Student)ss)(); // correct: 调用Student::operator()
// 下面这条语句报错,因为会先结合ssArray[0], 再ssArray[0](),再强制转换
// (Student)ssArray[0](); // error: invalid use of void expression
((Student)ssArray[1])(); // correct:将ssArray[1]强制转换为Student类型后,调用其()方法
// ()高于.*和->*:先结合()
cout << "-----------------5------------------" << endl;
void (Student::*fp)();
fp = Student::printInfo;
// s1.*fp(); // error: must use '.*' or '->*' to call pointer-to-member function in 'fp (...)'
(s1.*fp)();
(s2.*fp)();
// ps1->*fp(); // error: must use '.*' or '->*' to call pointer-to-member function in 'fp (...)'
(ps1->*fp)();
// (int)高于.*和->*:先结合(int)
cout << "-----------------6------------------" << endl;
Student2 *ssp = &sss; // Jason
void (Student2::*fp2)();
fp2 = Student2::printInfo;
(ss.*fp2)();
((Student)ss.*fp)(); // 先将ss强转为Student,然后调用Student::printInfo(),注意是.*fp而不是.*fp2
((Student*)ssp->*fp)(); // 先将ssp强转为Student*,然后调用Student::printInfo(),注意是.*fp而不是.*fp2
// *高于.*和->*:先结合*
cout << "-----------------7------------------" << endl;
(*ssp.*fp2)(); // 先*ssp,再.*fp2
Student2 **sspp = &ssp;
(*sspp->*fp2)(); // 先*sspp,再->fp2
delete ps1;
return 0;
}
输出:
-----------------1------------------
I am a student. My name is Bob. My id is 101
I am a student. My name is Jack. My id is 102
I am a student. My name is Eric. My id is 103
-----------------2------------------
I am a student. My name is Eric. My id is 103
-----------------3------------------
I am a student. My name is Eric. My id is 103
I am a student. My name is Eric. My id is 103
-----------------4------------------
I am Student2!!!
I am a student. My name is Alice. My id is 999
I am a student. My name is Jason. My id is 998
-----------------5------------------
I am a student. My name is Bob. My id is 101
I am a student. My name is Jack. My id is 102
I am a student. My name is Eric. My id is 103
-----------------6------------------
I am a super Student!!!
I am a student. My name is Alice. My id is 999
I am a student. My name is Jason. My id is 998
-----------------7------------------
I am a super Student!!!
I am a super Student!!!
推荐阅读
-
C++ 中指针和数组相关操作符的优先级
-
在C/C++中,如何轻松获取数组的长度?(使用宏和模板)
-
3/5法则:理解C++中的拷贝构造函数和拷贝赋值操作符
-
设循环队列中数组的下标范围是1 n 其头尾指针分别为f和r 则其元素个数为
-
C++编程:实战教程 - 实现栈与队列(Deque) 容器适配器,探讨Stack和Queue的实现及Queueue(Deque)的问题。深入理解:如何模拟实现Priority Queue (优先级队列) - 实例与练习。续集:仿函数在C++中的二次探索
-
深入理解C++中引用和指针的差异:实战实例解析
-
深入了解指针 -> 结构中的成员数组和指针
-
常量指针,C++ 中的指针常量(const 和 *),如何使用和理解它们
-
包婷婷 (201550484)作业一 统计软件简介与数据操作-SPSS(Statistical Product and Service Solutions),"统计产品与服务解决方案"软件。最初软件全称为"(SolutionsStatistical Package for the Social Sciences),但是随着SPSS产品服务领域的扩大和服务深度的增加,SPSS公司已于2000年正式将英文全称更改为"统计产品与服务解决方案",标志着SPSS的战略方向正在做出重大调整。为IBM公司推出的一系列用于统计学分析运算、数据挖掘、预测分析和决策支持任务的软件产品及相关服务的总称SPSS,有Windows和Mac OS X等版本。 1984年SPSS总部首先推出了世界上第一个统计分析软件微机版本SPSS/PC+,开创了SPSS微机系列产品的开发方向,极大地扩充了它的应用范围,并使其能很快地应用于自然科学、技术科学、社会科学的各个领域。世界上许多有影响的报刊杂志纷纷就SPSS的自动统计绘图、数据的深入分析、使用方便、功能齐全等方面给予了高度的评价。 R统计软件介绍 R是一套完整的数据处理、计算和制图软件系统。其功能包括:数据存储和处理系统;数组运算工具(其向量、矩阵运算方面功能尤其强大);完整连贯的统计分析工具;优秀的统计制图功能;简便而强大的编程语言:可操纵数据的输入和输出,可实现分支、循环,用户可自定义功能。 与其说R是一种统计软件,还不如说R是一种数学计算的环境,因为R并不是仅仅提供若干统计程序、使用者只需指定数据库和若干参数便可进行一个统计分析。R的思想是:它可以提供一些集成的统计工具,但更大量的是它提供各种数学计算、统计计算的函数,从而使使用者能灵活机动的进行数据分析,甚至创造出符合需要的新的统计计算方法。 该语言的语法表面上类似 C,但在语义上是函数设计语言(functional programming language)的变种并且和Lisp 以及 APL有很强的兼容性。特别的是,它允许在"语言上计算"(computing on the language)。这使得它可以把表达式作为函数的输入参数,而这种做法对统计模拟和绘图非常有用。 R是一个免费的*软件,它有UNIX、LINUX、MacOS和WINDOWS版本,都是可以免费下载和使用的。在R主页那儿可以下载到R的安装程序、各种外挂程序和文档。在R的安装程序中只包含了8个基础模块,其他外在模块可以通过CRAN获得。 二、R语言 R是用于统计分析、绘图的语言和操作环境。R是属于GNU系统的一个*、免费、源代码开放的软件,它是一个用于统计计算和统计制图的优秀工具。 R作为一种统计分析软件,是集统计分析与图形显示于一体的。它可以运行于UNIX,Windows和Macintosh的操作系统上,而且嵌入了一个非常方便实用的帮助系统,相比于其他统计分析软件,R还有以下特点: 1.R是*软件。这意味着它是完全免费,开放源代码的。可以在它的网站及其镜像中下载任何有关的安装程序、源代码、程序包及其源代码、文档资料。标准的安装文件身自身就带有许多模块和内嵌统计函数,安装好后可以直接实现许多常用的统计功能。[2] 2.R是一种可编程的语言。作为一个开放的统计编程环境,语法通俗易懂,很容易学会和掌握语言的语法。而且学会之后,我们可以编制自己的函数来扩展现有的语言。这也就是为什么它的更新速度比一般统计软件,如,SPSS,SAS等快得多。大多数最新的统计方法和技术都可以在R中直接得到。[2] 3. 所有R的函数和数据集是保存在程序包里面的。只有当一个包被载入时,它的内容才可以被访问。一些常用、基本的程序包已经被收入了标准安装文件中,随着新的统计分析方法的出现,标准安装文件中所包含的程序包也随着版本的更新而不断变化。在另外版安装文件中,已经包含的程序包有:base一R的基础模块、mle一极大似然估计模块、ts一时间序列分析模块、mva一多元统计分析模块、survival一生存分析模块等等.[2] 4.R具有很强的互动性。除了图形输出是在另外的窗口处,它的输入输出窗口都是在同一个窗口进行的,输入语法中如果出现错误会马上在窗口口中得到提示,对以前输入过的命令有记忆功能,可以随时再现、编辑修改以满足用户的需要。输出的图形可以直接保存为JPG,BMP,PNG等图片格式,还可以直接保存为PDF文件。另外,和其他编程语言和数据库之间有很好的接口。[2] 5.如果加入R的帮助邮件列表一,每天都可能会收到几十份关于R的邮件资讯。可以和全球一流的统计计算方面的专家讨论各种问题,可以说是全世界最大、最前沿的统计学家思维的聚集地.[2] R是基于S语言的一个GNU项目,所以也可以当作S语言的一种实现,通常用S语言编写的代码都可以不作修改的在R环境下运行。 R的语法是来自Scheme。R的使用与S-PLUS有很多类似之处,这两种语言有一定的兼容性。S-PLUS的使用手册,只要稍加修改就可作为R的使用手册。所以有人说:R,是S-PLUS的一个“克隆”。 但是请不要忘了:R是免费的(R is free)。R语言源代码托管在github,具体地址可以看参考资料。[3] 。 R语言的下载可以通过CRAN的镜像来查找。 R语言有域名为.cn的下载地址,有六个,其中两个由Datagurn,由 中国科学技术大学提供的。R语言Windows版,其中由两个下载地点是Datagurn和 USTC提供的。 三、stata Stata 是一套提供其使用者数据分析、数据管理以及绘制专业图表的完整及整合性统计软件。它提供许许多多功能,包含线性混合模型、均衡重复反复及多项式普罗比模式。用Stata绘制的统计图形相当精美。 新版本的STATA采用最具亲和力的窗口接口,使用者自行建立程序时,软件能提供具有直接命令式的语法。Stata提供完整的使用手册,包含统计样本建立、解释、模型与语法、文献等超过一万余页的出版品。 除此之外,Stata软件可以透过网络实时更新每天的最新功能,更可以得知世界各地的使用者对于STATA公司提出的问题与解决之道。使用者也可以透过Stata. Journal获得许许多多的相关讯息以及书籍介绍等。另外一个获取庞大资源的管道就是Statalist,它是一个独立的listserver,每月交替提供使用者超过1000个讯息以及50个程序。 四、PYTHON
-
C++ 中的基本矩阵运算 ( +、-、=、 使用二维指针打开空间,形成二维数组; 最初的设计题目是设计一个矩阵类来实现基本的矩阵操作; 最初的设计是一个存放矩阵元素的 HL[10][10] 数组,后来改为二维指针; 主要问题在于人们对二维指针的理解还不够深入,不知道它们是如何打开空间的; HL = 新的 double *[行]; for(i = 0;i < row;i++) HL[i] = new double [list]; 其次,没有找到合适的处理方法来添加不同类型的矩阵