php专区

 首页 > php专区 > PHP进阶 > 算法 > 编程之美中的买书最优惠问题

编程之美中的买书最优惠问题

分享到:
【字体:
导读:
         摘要:题目:在节假日的时候,书店一般都会做促销活动。由于《哈利波特》系列相当畅销,店长决定通过促销活动来回馈读者。在销售的《哈利波特》平装本系列中,一共有五卷,用编号0,1,2,3,4来表示。假设每一卷单独销售均需要8欧元。如果读者一次购买不同的两卷,就可以扣除5%的费用,三卷则更多。 ...

编程之美中的买书最优惠问题

题目:在节假日的时候,书店一般都会做促销活动。由于《哈利波特》系列相当畅销,店长决定通过促销活动来回馈读者。在销售的《哈利波特》平装本系列中,一共有五卷,用编号0, 1, 2, 3, 4来表示。假设每一卷单独销售均需要8欧元。如果读者一次购买不同的两卷,就可以扣除5%的费用,三卷则更多。假设具体折扣的情况如下:

本数          折扣
2          	5%
3          	10%
4           20%
5           25%

在一份订单中,根据购买的卷数以及本书,就会出现可以应用不同折扣规则的情况。但是,一本书只会应用一个折扣规则。比如,读者一共买了两本卷一,一本卷二。那么,可以享受到5%的折扣。另外一本卷一则不能享受折扣。如果有多种折扣,希望能够计算出的总额尽可能的低。

要求根据这样的需求,设计出算法,能够计算出读者所购买一批书的最低价格。

分析:

首先假设五册书分别为A,B,C,D,E,不失一般性可以假设Na>=Nb>=Nc>=Nd>=Ne,设所有书可以划分为k组,每组中书不重复,则在最优解中有如下性质:

  1. 所有只包含一本书的组均只包含同一本书
  2. 若有两组包含一本书的组包含的书不同,则这两组合并,能得到更优解,与最优解矛盾.

  3. 包含一本书的组只包含A
  4. 若包含书为A',若Na'==Na,则A,A'对调即可,若Na'

  5. 所有只包含两本书的组均包含相同两本书
  6. 组(A', B'), (A',C')折扣0.2重组为(A',B',C')与(A)折扣0.3

  7. 包含两本书的组必包含A
  8. 若存在包含一本A的组,而两本书的组不包含A,则合并能得更优解。若不存在包含一本A的组,而两本书组包含为A',A'',则包含三本书,四本组必含A而缺A'者。(A', A''), (AXY), 折扣0.4不如(A''),(AXYA')0.8。(A',A''),(AXYZ)折扣0.9不如(A''),(AXYZA')1.25。

  9. 包含两本书的组必包含B
  10. 同上证。

  11. 所有只包含三本书的组均包含相同三本书
  12. (A,B,C),(A,B,D)折扣0.6,(A,B),(A,B,C,D)折扣0.9

  13. 所有包含三本书的组均包含A
  14. 若存在只包含A的组,合并得更优。若存在只包含A, B的组(A,B)(XYZ)折扣0.4,不如(B)(AXYZ)折扣0.8。若不存在一,二本组,假设三本书组为(X,Y,Z),则必有包含A的四本组缺X,(X,Y,Z)(A,A',A'',A''')折扣1.1,不如(Y,Z),(A,A',A'',A''',X)折扣1.35

  15. 所有包含三本的组均包含B,C
  16. 同上可证

  17. 所有包含四本书的组均包含A,B,C
  18. 设XYZW为(ABC)(BCDE)折扣1.1,不如(B,C),(A,B,C,D,E)折扣1.35。其它情况同理可证。

  19. 包含五本书与包含三本书情况不会同时出现
  20. (A,B,C),(A,B,C,D,E)折扣1.55,不如(A,B,C,D),(A,B,C,E)折扣1.6

由以上证明可得如下结论:

  • 每组均包含A,所有组数与A相同
  • 所有包含两本及以上的组均包含B,组数与B同
  • 所有包含三本及以上的组均包含C,组数与C同
  • 三,五不并存

由此可得解法如下:

const double BuyBook::UNIT_PRICE = 8;
const double BuyBook::DISCOUNTS[5] = {1, 0.95, 0.9, 0.8, 0.75};
static const int BOOK_KINDS = 5;
double BuyBook::SearchFast(int* books)
{
    Sort(books);
    int g[5];
    g[0] = books[0] - books[1];
    g[1] = books[1] - books[2];
    g[2] = books[2] - books[3];
    g[3] = books[3] - books[4];
    g[4] = books[4];
    int t = min(g[2], g[4]);
    if (t > 0)
    {
        g[2] -= t;
        g[4] -= t;
        g[3] += 2 * t;
    }
    double sum = 0;
    for (int i = 0; i < BOOK_KINDS; ++i)
    {
        sum += g[i] * (i+1) * UNIT_PRICE * DISCOUNTS[i];
    }
    return sum;
}

本文地址:http://www.nowamagic.net/librarys/veda/detail/966,欢迎访问原出处。

编程之美中的买书最优惠问题
分享到:
JavaScript各种排序的性能比较
JavaScript各种排序的性能比较 排序是经常使用的编程例子,在JavaScript里各种排序的性能又如何呢?每个浏览器测试得出的数据会不一样。比如我用chrome 测试 一般快速排序都会最快,IE 则根据数组长度有可能希尔最快。 不要用太大数据去测试冒泡排序(浏览器崩溃了我不管)。 [4,2,5,6,8,9,7,0,1,3] ...
什么是极大极小博弈树?
什么是极大极小博弈树? 极大极小博弈树(Minimax Game Tree,简写为MGT)用于编写电脑之间的游戏程序,这类程序由两个游戏者轮流,每次执行一个步骤。当然,所有可能的步骤构成了一个树的结构。例如下面的图就是一个MGT,它表示了Tic-Tac-Toe游戏的前两步所有可能的步骤。Tic-Tac-Toe是一种简单的九宫格游戏,玩法是...
  •         php迷,一个php技术的分享社区,专属您自己的技术摘抄本、收藏夹。
  • 在这里……