C++中this指针用法详解及实例

C++中this指针用法详解及实例

概要:

本文简单介绍this指针的基本概念,并通过一个实际例子介绍this指针用于防止变量命名冲突和用于类中层叠式调用的两个用法。

this指针概览

C++中,每个类 对应了一个对象,每个对象指向自己所在内存地址的方式即为使用this指针。在类中,this指针作为一个变量通过编译器隐式传递给非暂存(non-static)成员函数。因为this指针不是对象本身,因此sizeof函数并不能用于确定this指针所对应的对象大小。this指针的具体类型与具体对象的类型以及对象是否被const关键字修饰 有关。例如,在类Employee的非常量函数中,this指针类型为Employee ,若为常量函数,则this指针类型为const Employee 。由于this本身是一个指向对象的指针,因此*this这类去指针操作则得到本类中对象的地址。关于this指针的使用,举例如下:

本文代码引用和免责声明:

/**************************************************************************
 * (C) Copyright 1992-2012 by Deitel & Associates, Inc. and        *
 * Pearson Education, Inc. All Rights Reserved.              *
 *                                    *
 * DISCLAIMER: The authors and publisher of this book have used their   *
 * best efforts in preparing the book. These efforts include the     *
 * development, research, and testing of the theories and programs    *
 * to determine their effectiveness. The authors and publisher make    *
 * no warranty of any kind, expressed or implied, with regard to these  *
 * programs or to the documentation contained in these books. The authors *
 * and publisher shall not be liable in any event for incidental or    *
 * consequential damages in connection with, or arising out of, the    *
 * furnishing, performance, or use of these programs.           *
 **************************************************************************/

Test.h文件:

#ifndef TEST_H
#define TEST_H

class Test
{
public:
  explicit Test( int = 0 ); // default constructor
  void print() const;
private:
  int x;
}; // end class Test

#endif /* TEST_H */

Test.cpp文件:

#include "Test.h"
#include <iostream>
using namespace std;

// constructor
Test::Test( int value ) : x( value ){}

// print x using implicit and explicit this pointers;
// the parentheses around *this are required
void Test::print() const
{
  // implicitly use the this pointer to access the member x
  cout << "    x = " << x;
  // explicitly use the this pointer and the arrow operator
  // to access the member x
  cout << "\n this->x = " << this->x;
  // explicitly use the dereferenced this pointer and
  // the dot operator to access the member x
  cout << "\n(*this).x = " << ( *this ).x << endl;
} // end function print

main.cpp中的调用示例:

#include "Test.h"
int main()
{
  Test testObject( 12 ); // instantiate and initialize testObject
  testObject.print();
  return 0;
} // end main

本例中,由于this本身是指针,因此类中变量x的读写方式即为this->x。注意由于this变量是隐式传递的,因此在同一个类中的成员函数中直接调用x变量其效果等同于通过this指针调用x。使用去指针化的this变量则获得对象地址,因此通过对象地址调用变量的方式是用点号操作符。

介绍完this指针获取变量的方式之后,接下来本文将介绍this指针的两个作用。

一、this指针用于防止类中的变量冲突

this指针可以用来防止数据域与传入参数变量名相同可能导致的问题。以下列程序为例:

Time.h文件

//此处省略定义头

class Time
{
public:
  //...此处省略若干行非重点部分
  Time &setHour( int ); // set hour
  Time &setMinute( int ); // set minute
  Time &setSecond( int ); // set second

  //...此处省略若干行非重点部分
private:
  unsigned int hour; // 0 - 23 (24-hour clock format)
  unsigned int minute; // 0 - 59
  unsigned int second; // 0 - 59
}; // end class Time

Time.cpp文件:

// set hour value
Time &Time::setHour( int hour ) // note Time & return
{
  if ( hour >= 0 && hour < 24 )
   this->hour = hour;
  else
   throw invalid_argument( "hour must be 0-23" );

  return *this; // enables cascading
} // end function setHour

// set minute 和 set second写法类似

此处代码传入参数名为hour,hour被赋值对象也是本类私有变量hour,此时用this指针指向hour变量的方式就防止了命名重复。注意到前述代码的返回值为指向这个对象的指针,这与接下来本文要分析的第二点有关。

二、this指针用于层叠式调用

通过返回类的去指针化的this指针*this,事实上就是返回了类所在的地址。那么此类就可以被层叠调用。如上述Time这个对象,主程序调用示例如下:

main.cpp文件:

//省略非重要的预处理指令和using命令
#include "Time.h" // Time class definition

int main()
{
  Time t; // create Time object

  // cascaded function calls
  t.setHour( 18 ).setMinute( 30 ).setSecond( 22 );

  //省略其余非重要部分
} // end main

此处t.setHour其实得到的返回值为&t,那么获取setMinute的方法就是t.setMinute。同样运行t.setHour(18).setMinute(30)之后返回值仍为&t,因此可以继续调用setSecond。

那么,这样返回指向对象的类安全性有问题么?注意,此处类是以整个对象的形式被返回的,并没有出现类中的私有成员地址被返回的情况,因此返回对象地址与返回变量的地址本质是不同的。返回对象之后,对象仍然确保了私有变量的封装性,因此就变量地址造成的安全性问题,此处是不必考虑的。

感谢 阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • 深度理解c++中的this指针

    1.this指针,就是一个指向当前对象的指针.我们知道,定义出一个类,它在内存中是不占空间的,只有定义了该类类型的对象时,系统就会为该对象分配一段存储空间,这段空间里只存储成员变量,对于成员函数,是存放在代码区的.(复习:内存分为5大区:静态区.常量区.栈.堆.代码区).下边给出一个日期类,通过这个实例,深度理解this指针. #define _CRT_SECURE_NO_WARNINGS 1 #include using namespace std; class Date { public:

  • C++编程指向成员的指针以及this指针的基本使用指南

    指向成员的指针 指向成员的指针的声明是指针声明的特例.使用以下序列来声明它们: [storage-class-specifiers] [cv-qualifiers] type-specifiers [ms-modifier] qualified-name ::* [cv-qualifiers] identifier [= & qualified-name :: member-name]; 声明说明符: 可选存储类说明符. 可选 const 和/或 volatile 说明符. 类型说明符:类型的名

  • C/C++静态类和this指针详解及实例代码

     C/C++静态类和this指针详解 1.静态类 C++的静态成员不仅可以通过对象来访问,还可以直接通过类名来访问. class CBook{ public: static double price;//需要通过类外来进行初始化 } int main(void){ CBook book; book.price;//通过对象来访问 CBook::price//通过类名来访问 return 0; } 静态成员变量 对应静态成员有以下几点需要注意: (1)静态数据成员可以是当前类的类型,而其他数据成员

  • C++中this指针的用法及介绍

    this指针只能在一个类的成员函数中调用,它表示当前对象的地址.下面是一个例子:   复制代码 代码如下: void Date::setMonth( int mn )     {      month = mn; // 这三句是等价的      this->month = mn;      (*this).month = mn;     } 1. this只能在成员函数中使用.全局函数,静态函数都不能使用this.实际上,成员函数默认第一个参数为T* const register this.如:

  • 详解C++中的this指针与常对象

    C++ this指针详解 this 是C++中的一个关键字,也是一个常量指针,指向当前对象(具体说是当前对象的首地址).通过 this,可以访问当前对象的成员变量和成员函数. 所谓当前对象,就是正在使用的对象,例如对于stu.say();,stu 就是当前对象,系统正在访问 stu 的成员函数 say(). 假设 this 指向 stu 对象,那么下面的语句中,this 就和 pStu 的值相同: Student stu; //通过Student类来创建对象 Student *pStu = &s

  • C++基础之this指针与另一种“多态”

    一.引入定义一个类的对象,首先系统已经给这个对象分配了空间,然后会调用构造函数. 一个类有多个对象,当程序中调用对象的某个函数时,有可能要访问到这个对象的成员变量.而对于同一个类的每一个对象,都是共享同一份类函数.对象有单独的变量,但是没有单独的函数,所以当调用函数时,系统必须让函数知道这是哪个对象的操作,从而确定成员变量是哪个对象的.这种用于对成员变量归属对像进行区分的东西,就叫做this指针.事实上它就是对象的地址,这一点从反汇编出来的代码可以看到. 二.分析1.测试代码: 复制代码 代码如

  • C++ 中的this指针详解及实例

    C++ this 指针详解 学习 C++ 的指针既简单又有趣.通过指针,可以简化一些 C++ 编程任务的执行,还有一些任务,如动态内存分配,没有指针是无法执行的.所以,想要成为一名优秀的 C++ 程序员,学习指针是很有必要的. 正如您所知道的,每一个变量都有一个内存位置,每一个内存位置都定义了可使用连字号(&)运算符访问的地址,它表示了在内存中的一个地址. this指针是类的一个自动生成.自动隐蔽的私有成员,它存在于类的非静态成员中,指向被调用函数所在的对象. 全局仅有一个this指针,当一个对

  • C++中this指针用法详解及实例

    C++中this指针用法详解及实例 概要: 本文简单介绍this指针的基本概念,并通过一个实际例子介绍this指针用于防止变量命名冲突和用于类中层叠式调用的两个用法. this指针概览 C++中,每个类 对应了一个对象,每个对象指向自己所在内存地址的方式即为使用this指针.在类中,this指针作为一个变量通过编译器隐式传递给非暂存(non-static)成员函数.因为this指针不是对象本身,因此sizeof函数并不能用于确定this指针所对应的对象大小.this指针的具体类型与具体对象的类型

  • 浅谈c++中的stl中的map用法详解

    Map是STL的一个关联容器,它提供一对一(其中第一个可以称为关键字,每个关键字只能在map中出现一次,第二个可能称为该关键字的值)的数据处理能力,由于这个特性,它完成有可能在我们处理一对一数据的时候,在编程上提供快速通道.这里说下map内部数据的组织,map内部自建一颗红黑树(一种非严格意义上的平衡二叉树),这颗树具有对数据自动排序的功能,所以在map内部所有的数据都是有序的,后边我们会见识到有序的好处. 下面举例说明什么是一对一的数据映射.比如一个班级中,每个学生的学号跟他的姓名就存在着一一

  • 基于C++中setiosflags()的用法详解

    cout<<setiosflags(ios::fixed)<<setiosflags(ios::right)<<setprecision(2); setiosflags 是包含在命名空间iomanip 中的C++ 操作符,该操作符的作用是执行由有参数指定区域内的动作:   iso::fixed 是操作符setiosflags 的参数之一,该参数指定的动作是以带小数点的形式表示浮点数,并且在允许的精度范围内尽可能的把数字移向小数点右侧:   iso::right 也是se

  • Angular 中 select指令用法详解

    最近在angular中使用select指令时,出现了很多问题,搞得很郁闷.查看了很多资料后,发现select指令并不简单,决定总结一下. select用法: <select ng-model="" [name=""] [required=""] [ng-required=""] [ng-options=""]> </select> 属性说明: 发现并没有ng-change属性 ng-

  • java 中 ChannelHandler的用法详解

    java 中 ChannelHandler的用法详解 前言: ChannelHandler处理一个I/O event或者拦截一个I/O操作,在它的ChannelPipeline中将其递交给相邻的下一个handler. 通过继承ChannelHandlerAdapter来代替 因为这个接口有许多的方法需要实现,你或许希望通过继承ChannelHandlerAdapter来代替. context对象 一个ChannelHandler和一个ChannelHandlerContext对象一起被提供.一个

  • Java中isAssignableFrom的用法详解

    class1.isAssignableFrom(class2) 判定此 Class 对象所表示的类或接口与指定的 Class 参数所表示的类或接口是否相同,或是否是其超类或超接口.如果是则返回 true:否则返回 false.如果该 Class 表示一个基本类型,且指定的 Class 参数正是该 Class 对象,则该方法返回 true:否则返回 false. 1. class2是不是class1的子类或者子接口 2. Object是所有类的父类 一个例子搞定: package com.auuz

  • php 中的closure用法详解

    Closure,匿名函数,是php5.3的时候引入的,又称为Anonymous functions.字面意思也就是没有定义名字的函数.比如以下代码(文件名是do.php) <?php function A() { return 100; }; function B(Closure $callback) { return $callback(); } $a = B(A()); print_r($a);//输出:Fatal error: Uncaught TypeError: Argument 1

  • JavaScript中eval()函数用法详解

    eval() 函数计算 JavaScript 字符串,并把它作为脚本代码来执行. 如果参数是一个表达式,eval() 函数将执行表达式.如果参数是Javascript语句,eval()将执行 Javascript 语句. 语法 复制代码 代码如下: eval(string) 参数 描述 string 必需.要计算的字符串,其中含有要计算的 JavaScript 表达式或要执行的语句. eval()函数用法详解: 此函数可能使用的频率并不是太高,但是在某些情况下具有很大的作用,下面就介绍一下eva

  • C# 中string.split用法详解

    第一种方法 string s=abcdeabcdeabcde; string[] sArray=s.Split('c') ; foreach(string i in sArray) Console.WriteLine(i.ToString()); 输出下面的结果: ab deab deab de 第二种方法 我们看到了结果是以一个指定的字符进行的分割.使用另一种构造方法对多个字 符进行分割: string s="abcdeabcdeabcde"; string[] sArray1=s.

  • JS、jQuery中select的用法详解

    1.js var obj=document.getElementById(selectid); obj.options.length = 0; //清除所有内容 obj.options[index] = new Option("three",3); //更改对应的值 obj.options[index].selected = true; //保持选中状态 obj.add(new Option("4","4")); "文本",&

随机推荐