题解 | #购物单#

购物单

https://www.nowcoder.com/practice/f9c6f980eeec43ef85be20755ddbeaf4

#include <iostream>
#include <vector>
using namespace std;

//物品类
class Object {
  protected:
    //物品编号
    int id;

    //物品价格
    int v;

    //物品重要度
    int p;

    //0或所属主件编号
    int q;

  public:
    //构造函数
    Object(const int j = 0, const int v = 0, const int p = 0, const int q = 0) {
        this->id = j;
        this->v = v;
        this->p = p;
        this->q = q;
        return;
    }

    //析构函数
    ~Object() {}

  public:
    //获取物品编号
    int getid(void)const {
        return this->id;
    }

    //获取物品价格
    int getvalue(void)const {
        return this->v;
    }

    //获取物品重要度
    int getprime(void)const {
        return this->p;
    }

    //判断是否是主件
    bool isSubject(void)const {
        if (this->q)
            return false;
        else return true;
    }

    //获取所属主件编号
    int getSubjectid(void)const {
        if (!this->q)
            cout << "这是主件。" << endl;
        return this->q;
    }

    //获取满意度
    int getSatisfaction(void)const {
        return this->v * this->p;
    }
};

class ShopList {
  private:
    //总钱数
    int N;

    //记录满意度(vvvi[i]j[j][k]表示购买用(j*10)的钱购买前i件物品的状态k的满意度)
    vector<vector<vector<int>>> vvvi;

    //物品列表
    vector<Object> vo;
    vector<vector<Object>> vvo;

  public:
    //构造函数(N为总钱数,m为物品总数
    ShopList(const int N, const int m);

    //析构函数
    ~ShopList();

    //求满意度
    int getSatisfaction(const int i, const int j, const int k)const;

    //最大满意度
    const int getMaxSatisfaction(void);

    //用j*10的钱购买新物品列表前i件物品所能获得的最大满意度
    const int getMaxSatisfaction(const int i, const int j)const;

    //物品列表重构
    void refactObjList(void);

    //填充满意度列表
    void fillSatisfactionVector(void);
};

ShopList::ShopList(const int N, const int m) {
    this->N = N;

    //清空物品列表
    this->vo.clear();

    //用于存储数值
    int v = 0;
    int p = 0;
    int q = 0;

    //录入
    for (int i = 1; i <= m; i++) {
        cin >> v >> p >> q;
        this->vo.push_back(Object(i, v, p, q));
    }
    return;
}

ShopList::~ShopList() {
}

int ShopList::getSatisfaction(const int i, const int j, const int k)const {
    //如果没有钱
    if (j == 0)
        return 0;

    //如果没有物品
    if (i == 0)
        return 0;

    //如果不买
    if (k == 4)
        return this->getMaxSatisfaction(i - 1, j);

    //物品及其附件的价格
    int v = this->vvo[i][0].getvalue() / 10;
    int v1 = this->vvo[i][1].getvalue() / 10;
    int v2 = this->vvo[i][2].getvalue() / 10;

    //如果钱不够买主件
    if (j < v)
        return this->getMaxSatisfaction(i - 1, j);

    //物品及其附件的满意度
    int s = this->vvo[i][0].getSatisfaction();
    int s1 = this->vvo[i][1].getSatisfaction();
    int s2 = this->vvo[i][2].getSatisfaction();

    //如果只买主件
    if (k == 0)
        return s + this->getMaxSatisfaction(i - 1, j - v);

    //如果买主件和附件1
    if (k == 1) {

        //如果钱不够
        if (j < v + v1)
            return this->getMaxSatisfaction(i - 1, j);
        return s + s1 + this->getMaxSatisfaction(i - 1, j - v - v1);
    }

    //如果买主件和附件2
    if (k == 2) {

        //如果钱不够
        if (j < v + v2)
            return this->getMaxSatisfaction(i - 1, j);
        return s + s2 + this->getMaxSatisfaction(i - 1, j - v - v2);
    }

    //如果买主件、附件1和附件2
    if (k == 3) {

        //如果钱不够
        if (j < v + v1 + v2)
            return this->getMaxSatisfaction(i - 1, j);
        return s + s1 + s2 + this->getMaxSatisfaction(i - 1, j - v - v1 - v2);
    }
    return 0;
}

const int ShopList::getMaxSatisfaction(void) {
    this->refactObjList();
    this->fillSatisfactionVector();
    return this->getMaxSatisfaction(this->vvvi.size() - 1, this->N / 10);
}

const int ShopList::getMaxSatisfaction(const int i, const int j) const {
    int max = 0;
    for (int k = 0; k < 5; k++) {
        if (this->vvvi[i][j][k] > max)
            max = this->vvvi[i][j][k];
    }
    return max;
}

void ShopList::refactObjList(void) {
    //在下标为0处置一空数组
    this->vvo.clear();
    this->vvo.push_back(vector<Object>());

    //遍历原物品列表
    for (vector<Object>::iterator voit = this->vo.begin(); voit != this->vo.end();
            voit++) {

        //如果是主件
        if (voit->isSubject()) {
            vector<Object> tmp(1, *voit);
            this->vvo.push_back(tmp);
        }
    }
    for (vector<Object>::iterator voit = this->vo.begin(); voit != this->vo.end();
            voit++) {

        //是附件
        if (!voit->isSubject()) {
            int id = voit->getSubjectid();
            for (int i = 1; i < this->vvo.size(); i++)
                if (this->vvo[i][0].getid() == id) {
                    this->vvo[i].push_back(*voit);
                    break;
                }
        }
    }

    //调整数组大小,避免下标溢出
    for (int i = 0; i < this->vvo.size(); i++)
        this->vvo[i].resize(3);
    return;
}

void ShopList::fillSatisfactionVector(void) {
    for (int i = 0; i < this->vvo.size(); i++) {
        vector<vector<int>> tmp;
        for (int j = 0; j <= this->N / 10; j++) {

            //五种状态:买主件、买主件和附件1、买主件和附件2、买主件和附件1和附件2、不买
            vector<int> vi;
            for (int k = 0; k < 5; k++) {
                vi.push_back(this->getSatisfaction(i, j, k));
            }
            tmp.push_back(vi);
        }
        this->vvvi.push_back(tmp);
    }
    return;
}

int main() {
    int N, m;
    while (cin >> N >> m) { // 注意 while 处理多个 case
        cout << ShopList(N, m).getMaxSatisfaction() << endl;
    }
}
// 64 位输出请用 printf("%lld")

全部评论

相关推荐

04-08 16:35
门头沟学院 Java
站队站对牛:实在是恶心的公司
点赞 评论 收藏
分享
上周组里招人,我面了六个候选人,回来跟同事吃饭的时候聊起一个让我挺感慨的现象。前三个候选人,算法题写得都不错。第一道二分查找,五分钟之内给出解法,边界条件也处理得干净。第二道动态规划,状态转移方程写对了,空间复杂度也优化了一版。我翻他们的简历,力扣刷题量都在300以上。后三个呢,就有点参差不齐了。有的边界条件没处理好,有的直接说这道题没刷过能不能换个思路讲讲。其中有一个女生,我印象特别深——她拿到题之后没有马上写,而是先问我:“面试官,我能先跟你确认一下我对题目的理解吗?”然后她把自己的思路讲了一遍,虽然最后代码写得不是最优解,但整个沟通过程非常顺畅。这个女生的代码不是最优的,但当我问她“如果这里是线上环境,你会怎么设计’的时候,她给我讲了一套完整的方案——异常怎么处理、日志怎么打、怎么平滑发布。她对这是之前在实习的时候踩过的坑。”我在想LeetCode到底在筛选什么?我自己的经历可能有点代表性。我当年校招的时候,也是刷了三百多道题才敢去面试。那时候大家都刷,你不刷就过不了笔试关。后来工作了,前三年基本没再打开过力扣。真正干活的时候,没人让你写反转链表,也没人让你手撕红黑树。更多的是:这个接口为什么慢了、那个服务为什么OOM了、线上数据对不上了得排查一下。所以后来我当面试官,慢慢调整了自己的评判标准。算法题我还会出,但目的变了。我出算法题,不是想看你能不能背出最优解。而是想看你拿到一个陌生问题的时候,是怎么思考的。你会先理清题意吗?你会主动问边界条件吗?你想不出来的时候会怎么办?你写出来的代码,变量命名乱不乱、结构清不清楚?这些才是工作中真正用得到的能力。LeetCode是一个工具,不是目的。它帮你熟悉数据结构和常见算法思路,这没问题。但如果你刷了三百道题,却说不清楚自己的项目解决了什么问题、遇到了什么困难、你是怎么解决的,那这三百道题可能真的白刷了。所以还要不要刷LeetCode?要刷,但别只刷题。刷题的时候,多问自己几个为什么:为什么用这个数据结构?为什么这个解法比那个好?如果换个条件,解法还成立吗?把刷题当成锻炼思维的方式,而不是背答案的任务。毕竟面试官想看到的,从来不是一台背题机器,而是一个能解决问题的人。
国企上岸了的向宇同桌...:最害怕答非所问了,但是频繁反问确定意思又害怕面试官觉得我笨
AI时代还有必要刷lee...
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务