详解C++中二进制求补运算符与下标运算符的用法_C 语言

二进制求补运算符:~ 
语法

~ cast-expression

备注
二进制反码运算符 (~)(有时称为“按位反码”运算符)将生成其操作数的按位二进制反码。即,操作数中为 1 的每个位在结果中为 0。相反,操作数中为 0 的每个位在结果中为 1。二进制反码运算符的操作数必须为整型。
~ 的运算符关键字
compl 运算符是 ~ 的文本等效项。访问程序中的 compl 运算符有两种方式:包括头文件 iso646.h,或使用 /Za 进行编译。

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

using namespace std;

int main () {
  unsigned short y = 0xFFFF;
  cout << hex << y << endl;
  y = ~y;  // Take one's complement
  cout << hex << y << endl;
}

在此示例中,分配给 y 的新值是无符号值 0xFFFF 或 0x0000 的二进制反码。
将对整型操作数执行整型提升,并且结果类型将是操作数将提升到的类型。

下标运算符:[]

 postfix-expression [ expression ]

备注
后跟下标运算符 [ ] 的后缀表达式(也可为主表达式)指定数组索引。
通常,postfix-expression 表示的值是一个指针值(如数组标识符),expression 是一个整数值(包括枚举类型)。 但是,从语法上来说,只需要一个表达式是指针类型,另一个表达式是整型。 因此整数值可以位于 postfix-expression 位置,指针值可以位于 expression 的方括号中或下标位置。 考虑以下代码片断:

  int nArray[5] = { 0, 1, 2, 3, 4 };
  cout << nArray[2] << endl;      // prints "2"
  cout << 2[nArray] << endl;      // prints "2"

在前面的示例中,表达式 nArray[2] 与 2[nArray] 相同。 原因是下标表达式 e1[ e2 ] 的结果由以下所示给定:

*( ( e2 ) + (e1) )

该表达式生成的地址不是 e1 地址中的 e2 字节。 相反,该地址将进行缩放以生成数组 e2 中的下一个对象。 例如:

double aDbl[2];

aDb[0] 和 aDb[1] 的地址相距 8 字节 - double 类型的对象的大小。 根据对象类型进行的缩放将由 C++ 语言自动完成,并在其中讨论了指针类型的操作数的加减法的相加运算符中定义。
下标表达式还可以有多个下标,如下所示:

expression1 [expression2] [expression3]...

下标表达式从左至右关联。 首先计算最左侧的下标表达式 expression1[expression2]。 通过添加 expression1 和 expression2 得到的地址构成一个指针表达式;然后 expression3 将添加到此指针表达式,从而构成一个新的指针表达式,依此类推,直到添加最后一个下标表达式。 在计算了最后的 subscripted 表达式后,将应用间接寻址运算符 (*),除非最终指针值将为数组类型寻址。
具有多个下标的表达式引用多维数组的元素。 多维数组是其元素为数组的数组。 例如,三维数组的第一个元素是一个具有两个维度的数组。 以下示例声明并初始化字符的简单二维数组:

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

using namespace std;
#define MAX_ROWS 2
#define MAX_COLS 2

int main() {
  char c[ MAX_ROWS ][ MAX_COLS ] = { { 'a', 'b' }, { 'c', 'd' } };
  for ( int i = 0; i < MAX_ROWS; i++ )
   for ( int j = 0; j < MAX_COLS; j++ )
     cout << c[ i ][ j ] << endl;
}

正下标和负下标
数组的第一个元素是元素 0。 C++ 数组的范围是从 array[0] 到 array[size – 1]。 但是,C++ 支持正负下标。 负下标必须在数组边界内;否则结果不可预知。 以下代码显示了正数组和负数组下标:

#include <iostream>
using namespace std;

int main() {
  int intArray[1024];
  for (int i = 0, j = 0; i < 1024; i++)
  {
    intArray[i] = j++;
  }

  cout << intArray[512] << endl;// 512

  int *midArray = &intArray[512]; // pointer to the middle of the array

  cout << midArray[-256] << endl;  // 256

  cout << intArray[-256] << endl; // unpredictable
}

上一行中的负下标可能产生运行时错误,因为它在内存中指向比数组的原点低 256 个字节的地址。 指针 midArray 会初始化为 intArray 的中点;因此可以对其使用正数组和负数组索引。 数组下标错误不会产生编译时错误,但它们会产生不可预知的结果。
下标运算符是可交换的。 因此,只要没有重载下标运算符(请参阅重载运算符 ),表达式 array[index] 和 array[array] 就一定等效。 第一种形式是最常见的编码做法,但它们都有效。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索c++
, 运算符
, 下标
二进制求补
下标运算符、下标运算符重载、c下标运算符重载、c语言下标运算符、c 下标 运算符 赋值,以便于您获取更多的相关知识。

时间: 2025-01-02 18:18:18

详解C++中二进制求补运算符与下标运算符的用法_C 语言的相关文章

详解C++中的指针、数组指针与函数指针_C 语言

C++中一个重要的特性就是指针,指针不仅具有获得地址的能力,还具有操作地址的能力.指针可以用于数组.或作为函数的参数,用来访问内存和对内存的操作,指针的使用使得C++很高效,但是指针也非常危险,使用不当会带来比较严重的问题. 1.指针 程序中所有的变量和常量都存在一个内存地址中,当然,函数也有对应的内存地址,内存地址的不同会导致程序执行时有所不同. 指针就是用来控制和存储内存地址的变量,它指向单个对象的地址,除了void之外,指针的数据类型与所指向地址的变量数据类型保持一致. 2.如何定义指针.

详解C语言中freopen()函数和fclose()函数的用法_C 语言

C语言freopen()函数:打开文件函数,并获得文件句柄 头文件: #include <stdio.h> 定义函数: FILE * freopen(const char * path, const char * mode, FILE * stream); 函数说明: 参数 path 字符串包含欲打开的文件路径及文件名. 参数mode 请参考fopen()说明.. 参数stream 为已打开的文件指针. Freopen()会将原stream 所打开的文件流关闭, 然后打开参数path 的文件.

详解C++中对构造函数和赋值运算符的复制和移动操作_C 语言

复制构造函数和复制赋值运算符从 C++ 11 中开始,该语言支持两种类型的分配:复制赋值和移动赋值. 在本文中,"赋值"意味着复制赋值,除非有其他显式声明. 赋值操作和初始化操作都会导致对象被复制. 赋值:在将一个对象的值赋给另一个对象时,第一个对象将复制到第二个对象中. 因此, Point a, b; ... a = b; 导致 b 的值被复制到 a 中. 初始化:在以下情况下将进行初始化:声明新对象.参数通过值传递给函数或值通过值从函数返回. 您可以为类类型的对象定义"复

详解C++成员函数的override和final说明符的用法_C 语言

override 说明符 可使用 override 关键字来指定在基类中重写虚函数的成员函数. 语法 function-declaration override; 备注 override 仅在成员函数声明之后使用时才是区分上下文的且具有特殊含义:否则,它不是保留的关键字. 使用 override 有助于防止您的代码中出现意外的继承行为.以下示例演示在未使用 override 的情况下,可能不打算使用派生类的成员函数行为.编译器不会发出此代码的任何错误. class BaseClass { vir

详解C语言中telldir()函数和seekdir()函数的用法_C 语言

C语言telldir()函数:取得目录流的读取位置头文件: #include <dirent.h> 定义函数: off_t telldir(DIR *dir); 函数说明:telldir()返回参数dir 目录流目前的读取位置. 此返回值代表距离目录文件开头的偏移量返回值返回下个读取位置, 有错误发生时返回-1. 错误代码:EBADF 参数dir 为无效的目录流. 范例 #include <sys/types.h> #include <dirent.h> #includ

详解Android中用于线程处理的AsyncTask类的用法及源码_Android

为什么要用AsyncTask我们写App都有一个原则,主线程不能够运行需要占用大量CPU时间片的任务,如大量复杂的浮点运算,较大的磁盘IO操作,网络socket等,这些都会导致我们的主线程对用户的响应变得迟钝,甚至ANR,这些会使应用的用户体验变差,但是有时又的确需要执行这些耗时的任务,那么我们通常可以使用AsyncTask或者new Thread 来处理,这样把任务放入工作线程中执行,不会占用主线程的时间片,所以主线程会及时响应用户的操作,如果使用new Thread来执行任务,那么如果需要中

详解C#中的定时器Timer类及其垃圾回收机制_C#教程

关于C# Timer类  在C#里关于定时器类就有3个 C# Timer使用的方法1.定义在System.Windows.Forms里 C# Timer使用的方法2.定义在System.Threading.Timer类里  " C# Timer使用的方法3.定义在System.Timers.Timer类里 下面我们来具体看看这3种C# Timer用法的解释: (1)System.Windows.Forms.Timer 应用于WinForm中的,它是通过Windows消息机制实现的,类似于VB或D

详解Android中ViewPager的PagerTabStrip子控件的用法_Android

我们先来看一个小例子: 可以看到,效果实现的也是很棒,比之前自定义的标签指示器更加的流畅.下面,简单介绍一下 PagerTabStrip和它的使用. PagerTabStrip是v4支持包里面的类,是ViewPager专用的类,不能在其他地方使用.在使用的时候,我们只需要在ViewPager的布局里面声明即可.     如下面的代码 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns

详解C++中的增量运算符++和减量运算符--的用法_C 语言

前缀增量和减量运算符:++ 和 --  语法 ++ unary-expression –– unary-expression 备注 前缀递增运算符 (++) 向其操作数添加 1:此递增值是表达式的结果.操作数必须是类型不为 const 的左值.结果是与操作数相同类型的左值. 前缀递减运算符 (––) 与前缀递增运算符类似,只不过操作数将减少 1,并且结果是递减值. 前缀和后缀递增和递减运算符均会影响其操作数.它们之间的主要差异是递增或递减在表达式的计算中出现的顺序.在前缀形式中,将在表达式计算中