c++链式调用
链式调用(Chaining)是一种设计方法,它允许你通过一个对象的多个方法连续调用,而不需要在每次调用后重新引用对象。这种方法在构建复杂对象时尤其有用,特别是在使用建造者模式时。
在C++中,实现链式调用的常见方法是让每个方法返回对象本身的引用(通常是 *this)。这样,调用者就可以在同一个表达式中继续调用其他方法。
示例:链式调用在C++中的实现
下面是一个简单的示例,展示如何在C++中使用链式调用来设置一个类的属性。
示例:设置一个简单的 Person 类
#include <iostream>
#include <string>
class Person {
public:
Person& setName(const std::string& name) {
this->name = name;
return *this;
}
Person& setAge(int age) {
this->age = age;
return *this;
}
Person& setAddress(const std::string& address) {
this->address = address;
return *this;
}
void display() const {
std::cout << "Name: " << name << ", Age: " << age << ", Address: " << address << std::endl;
}
private:
std::string name;
int age;
std::string address;
};
int main() {
Person person;
person.setName("John Doe").setAge(30).setAddress("123 Main St").display();
return 0;
}
在这个示例中,Person 类的每个 setter 方法都返回对当前对象的引用(*this),允许我们链式调用这些方法。display 方法用于显示对象的属性。
结合建造者模式和链式调用
链式调用在建造者模式中也非常有用,特别是当你需要构建复杂对象时。下面是一个结合建造者模式和链式调用的示例:
示例:建造者模式与链式调用
#include <iostream>
#include <string>
class Product {
public:
void show() const {
std::cout << "Product [PartA: " << partA << ", PartB: " << partB << ", PartC: " << partC << "]" << std::endl;
}
// 友元类 Builder 以便能访问 Product 的私有成员
friend class Builder;
private:
std::string partA;
std::string partB;
std::string partC;
};
class Builder {
public:
Builder& buildPartA(const std::string& partA) {
product.partA = partA;
return *this;
}
Builder& buildPartB(const std::string& partB) {
product.partB = partB;
return *this;
}
Builder& buildPartC(const std::string& partC) {
product.partC = partC;
return *this;
}
Product getResult() {
return product;
}
private:
Product product;
};
int main() {
Builder builder;
Product product = builder.buildPartA("PartA1").buildPartB("PartB1").buildPartC("PartC1").getResult();
product.show();
return 0;
}
在这个示例中,Builder 类的每个方法都返回 *this,使得我们可以链式调用这些方法来逐步构建 Product 对象。最终的 getResult 方法返回构建的 Product 对象。
在C++中,*this 返回的是当前对象的引用。为了更具体地解释,让我们看一下 *this 是如何工作的。
*this 的含义
this 是一个指针,它指向调用成员函数的对象。*this 则是对这个对象的解引用,因而得到的是该对象本身。因此,当你在成员函数中返回 *this 时,实际上是返回调用该函数的对象自身的引用。
示例:解释 *this
让我们通过一个简单的例子来解释它的工作原理。
#include <iostream>
#include <string>
class Person {
public:
Person& setName(const std::string& name) {
this->name = name; // 使用 this 指针访问成员变量
return *this; // 返回对象本身的引用
}
void display() const {
std::cout << "Name: " << name << std::endl;
}
private:
std::string name;
};
int main() {
Person person;
person.setName("John Doe").display();
return 0;
}
在这个例子中,Person 类有一个 setName 方法,该方法接受一个字符串参数并设置对象的 name 属性。然后,该方法返回 *this,即当前对象的引用。这使得你可以链式调用 display() 方法。
详细解读
- person.setName("John Doe"):当你调用 setName 方法时,this 指针指向 person 对象。
- this->name = name:使用 this 指针修改对象的成员变量。
- return *this:返回当前对象的引用,即 person 对象的引用。
- .display():由于 setName 返回的是 person 对象的引用,你可以继续调用 display 方法。
对象的生命周期
需要注意的是,返回 *this 时,返回的是对象的引用,而不是对象的副本,这意味着你在链式调用中操作的是同一个对象。例如:
Person& p = person.setName("Jane Doe");
在这个语句中,p 是对 person 对象的引用,而不是一个新的对象。任何对 p 的操作实际上都是对 person 的操作。
*this 返回的是调用成员函数的对象自身的引用。这使得链式调用成为可能,因为每个函数调用都返回相同的对象,可以继续调用其他成员函数。这种设计模式在需要设置或修改多个属性时特别有用,从而使代码更简洁和易读。
下面是进一步针对创建者模式,讲解*this 指针
示例:Builder 模式与链式调用
#include <iostream>
#include <string>
class Product {
public:
void show() const {
std::cout << "Product [PartA: " << partA << ", PartB: " << partB << ", PartC: " << partC << "]" << std::endl;
}
// 友元类 Builder 以便能访问 Product 的私有成员
friend class Builder;
private:
std::string partA;
std::string partB;
std::string partC;
};
class Builder {
public:
Builder& buildPartA(const std::string& partA) {
product.partA = partA;
return *this;
}
Builder& buildPartB(const std::string& partB) {
product.partB = partB;
return *this;
}
Builder& buildPartC(const std::string& partC) {
product.partC = partC;
return *this;
}
Product getResult() {
return product;
}
private:
Product product;
};
int main() {
Builder builder;
Product product = builder.buildPartA("PartA1").buildPartB("PartB1").buildPartC("PartC1").getResult();
product.show();
return 0;
}
详细解读
Builder 类中的方法
buildPartA, buildPartB, 和 buildPartC 方法在 Builder 类中被定义。这些方法修改 Builder 类的成员变量 product,并且每个方法都返回 *this,即 Builder 的引用。
这使得我们可以链式调用这些方法,例如:
builder.buildPartA("PartA1").buildPartB("PartB1").buildPartC("PartC1")。
product 成员变量
Builder 类有一个私有成员变量 product,它是一个 Product 对象。每个 buildPartX 方法都会修改这个 Product 对象的相应部分。
返回 Builder 对象的引用
每个 buildPartX 方法返回 *this,即当前 Builder 对象的引用。这使得我们可以链式调用这些方法,从而逐步构建 Product 对象。
最终生成 Product 对象
最后,通过调用 getResult() 方法,返回 product 对象。此时,product 对象已经通过之前的链式调用被完全构建好。
为什么 Builder 调用写进了 Product 内
- Builder 类的作用是逐步构建 Product 对象。每次调用 buildPartX 方法时,都是在修改 Builder 类内部的 Product 对象,而不是创建新的 Builder 或 Product 对象。
- 最终,通过调用 getResult() 方法,返回构建好的 Product 对象。
在 Builder 模式中,链式调用的返回值是 Builder 对象的引用(*this),这使得我们可以连续调用 Builder 类的方法来设置 Product 对象的属性。Builder 内部持有一个 Product 对象,并在链式调用中逐步构建这个对象。最终,通过 getResult() 方法返回构建好的 Product 对象。