通常情况下,类的成员函数都只涉及一个对象,即调用它的对象。但有时候方法可能涉及到两个对象,在这种情况就需要使用到C++的this指针。
class Stock
{
private:
...
double total_val;
...
public:
double total() const {return total_val;}
}
如上述类,total_val为私有成员,外部程序不可以直接访问该数据成员,只能通过total()方法获取total_val值。如果要知道哪个Stock对象total_val的值最大,方法一是对象调用total()方法来较total_val值。
另外一种方法就是使用this指针。定义一个成员函数,该函数可以查看两个Stock对象,并返回total_val值较高对象的引用。拟定成员函数名为topval(),stock1.topval()访问stock1对象的数据成员;stock2.topval()访问stock2对象的数据成员。要对两个对象进行比较,则必须将其中一个对象作为参数,传递给成员函数topval(),则topval()的参数类型应为const Stock &(出于效率原因选择使用引用来传递参数)。topval()需要返回total_val值较大的对象,则可以直接返回一个Stock对象的引用。topval()方法的原型如下:
const Stock & topval(const Stock & s)const;
topval()方法隐式的访问一个对象,显式地访问一个对象,并返回其中一个对象的引用。括号中的const表示该函数不会修改被显式访问的对象;括号后面的cosnt表明,该函数不会修改被隐式访问的对象。由于该函数返回了两个const对象之一的引用,因此返回类型也应使用const修饰。
假设都要对Stock对象stock1和stock2进行比较,将total_val值较大的对象赋值给top对象,可以使用以下两条语句实现:
top = stock1.topval(stock2);
top = stock2.topval(stock1);
第一种格式:隐式地访问stock1,显式地访问stock2;第二种格式:隐式地访问stock2,显式地访问stock1。无论使用哪一种格式,都会返回total_val值较大的那个对象。
C++中this指针指向用来调用成员函数的对象(this被作为隐藏参数传递给方法)。stock1.topval(stock2);语句将this设置为stock1对象的地址。一般来说,类方法都将this指针设置为调用它的对象的地址。topval()中的total_val只不过是this->total_val的简写。topval()的实现如下:
const Stock & Stock::topval(const Stock & s)const
{
if(s.total_val > total_val)
return s;
else
return *this;
}
特别提醒: 每个成员函数(包括构造函数和析构函数)都有一个this指针。this指针指向调用对象。如果方法需要引用整个调用对象,则可以使用表达式 * this 。在函数的括号后面使用const限定符将this限定为const,这样就不能使用this来修改对象的值。this是对象的地址,而不是对象本身。* this才是对象本身, * this可以作为调用对象的别名。( * this将解除引用运算符*用于指针,将得到指针指向的值)
示例代码如下:
1、定义文件stock20.h
//stock20.h--augmented version
#pragma once
#include
class Stock
{
private:
std::string company;
int shares;
double share_val;
double total_val;
void set_tot() { total_val = shares * share_val; }
public:
Stock();
Stock(const std::string & c0, int n = 0, double pr = 0.0);
~Stock();
void buy(long num, double price);
void sell(long num, double price);
void update(double price);
void show()const;
const Stock& topval(const Stock & s)const;
};
2、实现文件stock20.cpp
//stock20.cpp--augement version
#include "stock20.h"
#include
Stock::Stock()
{
company = "no name";
shares = 0;
share_val = 0.0;
total_val = 0.0;
}
Stock::Stock(const std::string& co, int n, double pr)
{
company = co;
if (n < 0)
{
std::cout << "Number of shares can't be negative;"
<< company << " shares set to 0.\n";
shares = 0;
}
else
shares = n;
share_val = pr;
set_tot();
}
Stock::~Stock()
{
}
void Stock::buy(long num, double price)
{
if (num < 0)
{
std::cout << "Number of shares purchased can't be nagative." << "Transacyion is aborted.\n";
}
else
{
shares += num;
share_val = price;
set_tot();
}
}
void Stock::sell(long num, double price)
{
using std::cout;
if (num < 0)
{
cout << "Number of shares sold can't be nagative."
<< "Transaction is abored.\n";
}
else if(num > shares)
{
cout << "You can't sell more than you have! "
<< "Transaction is aborted.\n";
}
else
{
shares -= num;
share_val = price;
set_tot();
}
}
void Stock::update(double price)
{
share_val = price;
set_tot();
}
void Stock::show()const
{
using std::cout;
using std::ios_base;
//set format #.###
ios_base::fmtflags orig = cout.setf(ios_base::fixed, ios_base::floatfield);
std::streamsize prec = cout.precision(3);
cout << "Company: " << company << "Shares: " << shares << '\n';
cout << "Shares Price:$" << share_val;
//set format to #.##
cout.precision(2);
cout << " Total Worth:$" << total_val << '\n';
//restore original format
cout.setf(orig, ios_base::floatfield);
cout.precision(prec);
}
const Stock & Stock::topval(const Stock & s)const
{
if (s.total_val > total_val)
return s;
else
return *this;
}
3、使用文件usestok2.cpp
//usestock2.cpp-- using the Stock class
//compilewith stock20.cpp
#include
#include "stock20.h"
int main()
{
Stock kate("NanoSmart", 12, 20.0);
kate.show();
Stock joe("Fleep Enterprises", 60, 6.5);
joe.show();
Stock top = kate.topval(joe);
std::cout << "\nMost valuable holding:\n";
top.show();
return 0;
}