后缀表达式

后缀表达式包含主表达式或者其中的后缀运算符跟在主表达式之后的表达式。 下表列出了后缀运算符。

后缀运算符

运算符名称 运算符表示法
下标运算符 [ ]
函数调用运算符 ( )
显式类型转换运算符 type-name( )
成员访问运算符 or ->
后缀递增运算符 ++
后缀递减运算符 --

以下语法描述了可能的后缀表达式:

primary-expression
postfix-expression[expression]postfix-expression(expression-list)simple-type-name(expression-list)postfix-expression.namepostfix-expression->namepostfix-expression++postfix-expression--cast-keyword < typename > (expression )typeid ( typename )

上面的 postfix-expression 可能是主表达式或另一个后缀表达式。 后缀表达式从左到右进行分组,这允许表达式按如下方式链接起来:

func(1)->GetValue()++

在上面的表达式中,func 是主表达式,func(1) 是函数后缀表达式,func(1)->GetValue 是指定类成员的后缀表达式,func(1)->GetValue() 是另一个函数后缀表达式,整个表达式是增加 GetValue 的返回值的后缀表达式。 该表达式的整体含义是作为自变量传递 1 的 "call func,并作为返回值获取一个指向类的指针。 然后调用此类上的 GetValue(),接着递增返回的值。

上面列出的表达式是赋值表达式,这意味着这些表达式的结果必须为右值。

后缀表达式形式

simple-type-name ( expression-list )

指示构造函数的调用。 如果 simple-type-name 是基本类型,则表达式列表必须是单个表达式,并且该表达式指示表达式的值将转换为基础类型。 此类强制转换表达式模仿构造函数。 由于此形式允许使用相同的语法来构造基本类型和类,因此它在定义模板类时特别有用。

cast-keyword 是 dynamic_caststatic_castreinterpret_cast 之一。 更多信息可在 dynamic_caststatic_castreinterpet_cast 中找到。

typeid 运算符被视为后缀表达式。 请参阅 typeid 运算符

形式和实际自变量

调用程序将“实参”中的信息传递给被调用的函数。被调用的函数使用相应的“形参”访问该信息。

当调用函数时,将执行以下任务:

  • 计算所有实参(调用方提供的参数)。 没有计算这些自变量的隐含顺序,但所有自变量都会计算,并且所有副作用都会在进入该函数前完成。

  • 使用每个形参在表达式列表中对应的实参来初始化该形参。 (形参是在函数头中声明并在函数体中使用的参数。)转换就像通过初始化完成的一样 - 标准和用户定义的转换都是在将实参转换为正确类型时执行的。 以下代码从概念上演示了所执行的初始化:

    void Func( int i ); // Function prototype
    ...
    Func( 7 );          // Execute function call
    

    调用前的概念性初始化为:

    int Temp_i = 7;
    Func( Temp_i );
    

    请注意,初始化就像使用等号语法(而不是括号语法)一样执行。 在将值传递到函数之前制作了 i 的副本。 (有关详细信息,请参阅初始化表达式转换)。

    因此,如果函数原型(声明)对 long 类型的参数进行调用,并且调用程序提供了 int 类型的实参,则会使用到 long 类型的标准类型转换提升该实参(请参阅标准转换)。

    如果提供了一个实际自变量,但它没有到形式自变量类型的标准的或用户定义的转换,则是一个错误。

    对于类类型的实际自变量,将通过调用类的构造函数初始化形式自变量。 (有关这些特殊类成员函数的详细信息,请参阅构造函数。)

  • 执行函数调用。

以下程序片段演示了函数调用:

// expre_Formal_and_Actual_Arguments.cpp
void func( long param1, double param2 );

int main()
{
    long i = 1;
    double j = 2;

    // Call func with actual arguments i and j.
    func( i, j );
}

// Define func with formal parameters param1 and param2.
void func( long param1, double param2 )
{
}

当从 main 调用 func 时,将使用 ii 将转换为 long 类型以对应使用标准转换的正确类型)的值初始化形参 param1,并使用 jj 将转换为使用标准转换的 double 类型)的值初始化形参 param2

自变量类型的处理

不能在函数主体内更改声明为 const 类型的形参。 函数可以更改类型不是 const 的任何参数。 但是,更改对于函数而言是本地进行的,且不会影响实参的值,除非实参是对非 const 类型的对象的引用。

以下函数阐释了其中的一些概念:

// expre_Treatment_of_Argument_Types.cpp
int func1( const int i, int j, char *c ) {
   i = 7;   // C3892 i is const.
   j = i;   // value of j is lost at return
   *c = 'a' + j;   // changes value of c in calling function
   return i;
}

double& func2( double& d, const char *c ) {
   d = 14.387;   // changes value of d in calling function.
   *c = 'a';   // C3892 c is a pointer to a const object.
    return d;
}

省略号和默认参数

通过使用下列两种方法之一,可以声明函数以接受比函数定义中指定的自变量更少的自变量:省略号 (...) 或默认自变量。

省略号表示可能需要参数,但声明中未指定数目和类型。 这通常是较差的 C++ 编程做法,因为它使您无法获得 C++ 的一个优点,即类型安全。 不同的转换将应用于使用省略号声明的函数,而不是应用于那些形参和实参类型的已知函数:

  • 如果实参的类型为 float,则在函数调用前将其提升为 double 类型。

  • 使用整数提升将任何 signed charunsigned charsigned shortunsigned short、枚举类型或位字段转换为 signed intunsigned int

  • 类类型的所有参数都作为数据结构通过值进行传递;副本是由二进制复制创建的,而不是通过调用类的复制构造函数(如果存在)创建的。

如果使用省略号,则必须在参数列表中最后声明它。 有关传递可变数量的参数的详细信息,请参阅《运行时库参考》中对 va_arg, va_start, and va_list 的讨论

有关 CLR 编程中默认参数的信息,请参阅变量参数列表 (...) (C++/CLI)

如果函数调用中没有提供值,则可通过默认参数指定参数应采用的值。 以下代码片段演示默认自变量的工作方式。 有关指定默认参数的限制的详细信息,请参阅默认参数

// expre_Ellipsis_and_Default_Arguments.cpp
// compile with: /EHsc
#include <iostream>

// Declare the function print that prints a string,
// then a terminator.
void print( const char *string,
            const char *terminator = "\n" );

int main()
{
    print( "hello," );
    print( "world!" );

    print( "good morning", ", " );
    print( "sunshine." );
}

using namespace std;
// Define print.
void print( const char *string, const char *terminator )
{
    if( string != NULL )
        cout << string;

    if( terminator != NULL )
        cout << terminator;
}

上面的程序声明一个采用两个参数的函数 print。 而第二个参数 terminator 具有默认值 "\n"。 在 main 中,对 print 的前两个调用允许默认的第二个参数提供新行以终止打印的字符串。 第三个调用为第二个参数指定显式值。 该程序的输出为

hello,
world!
good morning, sunshine.

另请参阅

表达式类型