C++ - 复制(copy) 和 虚复制(virtual copy) 的 区别

复制(copy) 和 虚复制(virtual copy) 的 区别

 

本文地址: http://blog.csdn.net/caroline_wendy/article/details/16120397

 

在继承过程中, 需要区分复制(copy)虚复制(virtual copy);

在派生类转换为基类时, 复制(copy)有可能切掉(sliced down)派生对象的派生部分, 只保留基类部分, 使派生类的虚函数无法使用;

为了避免此情况, 如果传入是对象, 则可以定义虚函数clone, 使派生类继承此虚函数, 再传入容器, 可以重载方法;

另外, 如果传入是实参, 使用shared_ptr<Base>配合make_shared<Derived>添加容器, 也可以进行动态绑定;

再传入容器中, 容器会自动调用派生类的重载方法, 实现动态绑定;

注意: 引用限定符(reference qualifier) GCC 4.8.1 才能支持;

 

代码:

/*
 * CppPrimer.cpp
 *
 *  Created on: 2013.11.12
 *      Author: Caroline
 */

/*eclipse cdt*/

#include <iostream>
#include <string>
#include <vector>
#include <set>

#include <utility>
#include <memory>
#include <cstddef>

using namespace std;

class Quote;

class Quote {
public:
	Quote() = default;
	Quote (const std::string& book, double sales_price) :
		bookNo (book), price (sales_price) {}
	void add_item (const Quote& sale);
	std::string isbn() const { return bookNo; }
	virtual double net_price (std::size_t n) const { return n* price; } //虚函数
	virtual Quote* clone() const & {return new Quote(*this);}
	virtual Quote* clone() && {return new Quote(std::move(*this));}
	virtual ~Quote() = default; //动态绑定析构器
private:
	std::string bookNo;
protected: //受保护类型
	double price = 0.0;
};

class Disc_quote : public Quote { //抽象基类
public:
	Disc_quote() = default;
	Disc_quote (const std::string& book, double price, std::size_t qty, double disc) :
		Quote(book, price), quantity (qty), discount (disc) {}
	double net_price (std::size_t) const = 0; //纯虚函数
protected:
		std::size_t quantity = 0;
		double discount = 0.0;
};

class Bulk_quote final : public Disc_quote { //final限定词, 无法被继承
public:
	Bulk_quote() = default;
	Bulk_quote(const std::string& book, double p, std::size_t qty, double disc) :
		Disc_quote(book, p, qty, disc) {} //使用基类的构造器
	double net_price(std::size_t cnt) const override;
	virtual Bulk_quote* clone() const & {return new Bulk_quote(*this);}
	virtual Bulk_quote* clone() && {return new Bulk_quote(std::move(*this));}
};

double Bulk_quote::net_price(std::size_t cnt) const
{
	if (cnt >= quantity)
		return cnt * (1-discount) * price;
	else
		return cnt * price;
}

double print_total(std::ostream &os, const Quote& item, std::size_t n)
{
	double ret = item.net_price(n);
	os << "ISBN: " << item.isbn() << " # sold: " << n << " total due: " << ret << std::endl;
	return ret;
}

class Basket {
public:
	void add_item (const std::shared_ptr<Quote> &sale) { items.insert(sale); }
	void add_item (const Quote& sale)
	{
		//items.insert(std::shared_ptr<Quote>(new Quote(sale))); //不会动态绑定
		items.insert(std::shared_ptr<Quote>(sale.clone()));
	}
	void add_item (Quote&& sale)
	{
		//items.insert(std::shared_ptr<Quote>(new Quote(std::move(sale)))); //不会动态绑定
		items.insert(std::shared_ptr<Quote>(std::move(sale).clone()));
	}
	double total_reciept (std::ostream&) const;
private:
	static bool compare (const std::shared_ptr<Quote> &lhs,
			const std::shared_ptr<Quote> &rhs)
	{ return lhs->isbn() < rhs->isbn(); };
	std::multiset<std::shared_ptr<Quote>, decltype(compare)*> items{compare};
};

double Basket::total_reciept(std::ostream &os) const
{
	double sum = 0.0;
	for(auto iter = items.cbegin(); iter != items.cend();
			iter = items.upper_bound(*iter)) { //跳过同名书, 直接计算count
		sum += print_total(os, **iter, items.count(*iter)); //*it是shared_ptr; **it是object
	}
	os << "Total Sale: " << sum << std::endl;
	return sum;
}

int main (void) {

	Basket bsk;
	/*bsk.add_item(std::make_shared<Quote>("CppPrimer", 45));
	bsk.add_item(std::make_shared<Bulk_quote>("EffectiveCpp", 50, 2, 0.15));
	bsk.add_item(std::make_shared<Bulk_quote>("EffectiveCpp", 50, 2, 0.15));
	bsk.add_item(std::make_shared<Bulk_quote>("EffectiveCpp", 50, 2, 0.15));*/

	bsk.add_item(Quote("CppPrimer", 45));
	bsk.add_item(Bulk_quote("EffectiveCpp", 50, 2, 0.15));
	bsk.add_item(Bulk_quote("EffectiveCpp", 50, 2, 0.15));
	bsk.add_item(Bulk_quote("EffectiveCpp", 50, 2, 0.15));

	bsk.total_reciept(std::cout);

	return 0;

}


输出:

ISBN: CppPrimer # sold: 1 total due: 45
ISBN: EffectiveCpp # sold: 3 total due: 127.5
Total Sale: 172.5


 

 

 

posted on 2013-11-14 21:16  you Richer  阅读(530)  评论(0编辑  收藏  举报