牛客-华为机试-困难-字符串 模拟-HJ98自动售货系统-C++

描述

1 总体说明

考生需要模拟实现一个简单的自动售货系统,实现投币、购买商品、退币、查询库存商品及存钱盒信息的功能。

系统初始化时自动售货机中商品为6种商品,商品的单价参见1.1规格说明,存钱盒内放置1元、2元、5元、10元钱币,商品数量和钱币张数通过初始化命令设置,参见2.1 系统初始化。

1.1规格说明

1. 商品:每种商品包含商品名称、单价、数量三种属性,其中商品名不重复。考生不能修改商品名称和单价,初始化命令设置商品数量。这些信息在考试框架中进行定义,考生在实现功能代码时可直接使用。

商品 名称

单价

数量

A12X
A23X
A34X
A45X
A58X
A66X

2. 存钱盒信息:钱币面额、张数两种属性。初始化命令设置各种面额钱币张数。这些信息在考试框架中进行定义,考生在实现功能代码时可直接使用。

钱币面额

张数

10元

X

5元

X
2元X
1元X

3. 退币原则

1) 根据系统存钱盒内钱币的 信息 ,按钱币总张数最少的原则进行退币。

2) 如果因零钱不足导致不能退币,则尽最大可能退币,以减少用户损失。

例如:假设存钱盒内只有4张2元,无其它面额钱币。如果需要退币7元,系统因零钱不足无法退币,则继续尝试退币6元,最终系统成功退币3张2元,用户损失1元钱币。

4. 投币操作说明:每次投币成功,投入的钱币面额累加到投币余额;同时,本次投入的钱币放入存钱盒中,存钱盒相应面额钱币增加。

5. 投币余额:指当前自动售货机中用户剩余的可购买商品的钱币总额;例如:投入2元面额的钱币,投币余额增加2元;购买一件价格2元的商品,投币余额减少2元;

6. 退币操作说明:退币操作需要遵守 退币原则 ;退币成功后,投币余额清零,同时扣除存钱盒相应的金额。

7. 购买商品操作说明:一次仅允许购买一件商品;购买商品成功后,自动售货机中对应商品数量减1,投币余额扣除本次购买商品的价格。

2 操作说明

命令字与第一个参数间使用一个空格分隔,多条命令采用分号隔开。考试系统会对输入命令格式进行处理,考生不需要关注输入命令格式的合法性,只需要实现命令处理函数。

2.1 系统初始化

命令格式

r A1 数量 -A2 数量 -A3 数量 -A4 数量 -A5 数量 -A6 数量 1 元张数 -2 元张数 -5 元张数 -10 元张数

参数名称

参数说明

类型

取值范围

A1数量

商品A1数量

整数

[0,30]

A2数量

商品A2数量

整数

[0,30]

A3数量

商品A3数量

整数

[0,30]

A4数量

商品A4数量

整数

[0,30]

A5数量

商品A5数量

整数

[0,30]

A6数量

商品A6数量

整数

[0,30]

1元张数

面额1元钱币张数

整数

[0,30]

2元张数

面额2元钱币张数

整数

[0,30]

5元张数

面额5元钱币张数

整数

[0,30]

10元张数

面额10元钱币张数

整数

[0,30]

商品和各种面额钱币取值范围只是作为初始化命令的限制,其它场景下不限制取值范围;考试框架已经实现取值范围的检查,考生不需要关注。

功能说明:设置自动售货机中商品数量和存钱盒各种面额的钱币张数;

约束说明:系统在任意阶段均可执行r初始化系统;考生不需要关注参数的合法性,不需要关注增加或缺少参数的场景;

输出说明:输出操作成功提示(执行完r命令后系统会自动输出操作结果,考生不需要再次调用输出函数),例:

命令输出含义
r 6-5-4-3-2-1 4-3-2-1;S001:Initialization is successful初始化成功

2.2 投币

命令格式p 钱币面额

功能说明

(1) 如果投入非1元、2元、5元、10元的钱币面额(钱币面额不考虑负数、字符等非正整数的情况),输出“E002:Denomination error”;

(2) 如果存钱盒中1元和2元面额钱币总额小于本次投入的钱币面额,输出“E003:Change is not enough, pay fail”,但投入1元和2元面额钱币不受此限制。

(3) 如果自动售货机中商品全部销售完毕,投币失败。输出“E005:All the goods sold out”;

(4) 如果投币成功,输出“S002:Pay success,balance=X”;

约束说明

(1) 系统在任意阶段都可以投币;

(2) 一次投币只能投一张钱币;

(3) 同等条件下,错误码的优先级:E002 > E003 > E005;

输出说明:如果投币成功,输出“S002:Pay success,balance=X”。

例:

命令

输出

p 10;

S002:Pay success,balance=10

2.3 购买商品

命令格式b 商品名称

功能说明

(1) 如果购买的商品不在商品列表中,输出“E006:Goods does not exist”;

(2) 如果所购买的商品的数量为0,输出“E007:The goods sold out”;

(3) 如果投币余额小于待购买商品价格,输出“E008:Lack of balance”;

(4) 如果购买成功,输出“S003:Buy success,balance=X”;

约束说明

(1) 一次购买操作仅能购买一件商品,可以多次购买;

(2) 同等条件下,错误码的优先级:E006 > E007 > E008;

输出说明:

如果购买成功,输出“S003:Buy success,balance=X”。

例:

命令

输出

b A1;

S003:Buy success,balance=8

2.4 退币

命令格式c

功能说明

(1) 如果投币余额等于0的情况下,输出“E009:Work failure”;

(2) 如果投币余额大于0的情况下,按照 退币原则 进行“找零”,输出退币信息;

约束说明

(1) 系统在任意阶段都可以退币;

(2) 退币方式必须按照 退币原则 进行退币;

输出说明:如果退币成功,按照 退币原则 输出退币信息。

例,退5元钱币:

命令

输出

c;

1 yuan coin number=0

2 yuan coin number=0

5 yuan coin number=1

10 yuan coin number=0

2.5 查询

命令格式q 查询类别

功能说明

(1) 查询自动售货机中商品信息,包含商品名称、单价、数量。 根据商品数量从大到小进行排序;商品数量相同时,按照商品名称的先后顺序进行排序

例如:A1的商品名称先于A2的商品名称,A2的商品名称先于A3的商品名称。

(2) 查询存钱盒信息,包含各种面额钱币的张数;

(3) 查询类别如下表所示:

查询类别

查询内容

0

查询商品信息

1查询存钱盒信息

如果“查询类别”参数错误,输出“E010:Parameter error”。“查询类别”参数错误时,不进行下面的处理;

输出说明

“查询类别”为0时,输出自动售货机中所有商品信息(商品名称单价数量)例:

命令

输出

q 0;

A1 2 6

A2 3 5

A3 4 4

A4 5 3

A5 8 2

A6 6 0

“查询类别”为1时,输出存钱盒信息(各种面额钱币的张数),格式固定。例:

命令

输出

q 1;

1 yuan coin number=4

2 yuan coin number=3

5 yuan coin number=2

10 yuan coin number=1

输入描述:

依照说明中的命令码格式输入命令。

输出描述:

输出执行结果

示例1

输入:

r 22-18-21-21-7-20 3-23-10-6;c;q0;p 1;b A6;c;b A5;b A1;c;q1;p 5;

输出:

S001:Initialization is successful
E009:Work failure
E010:Parameter error
S002:Pay success,balance=1
E008:Lack of balance
1 yuan coin number=1
2 yuan coin number=0
5 yuan coin number=0
10 yuan coin number=0
E008:Lack of balance
E008:Lack of balance
E009:Work failure
E010:Parameter error
S002:Pay success,balance=5

代码

#include<iostream>
#include<vector>
#include<cstring>
#include<list>
#include<map>
using namespace std;
//A1 ~ A6
class good{
public:
    string name;
    int price;
    int count;
    good(){    
    }
    good(const char* n, int p, int c){
        name = n;
        price = p;
        count = c;
    }
};
//1 2 5 10
class money{
public:
    int denomination;
    int count;
    
    money()
    {
        
    }
    
    money(int d, int c)
    {
        denomination = d;
        count = c;
    }
};
//贩售机相关属性
//货物
map<int, good> goods;
//贩售机钱盒中的钱
map<int, money> all_moneys;
//用户投入的总金额
int user_amount;
void init(){
    //初始化物品
    goods.insert(pair<int, good>(1, good("A1", 2, 0)));
    goods.insert(pair<int, good>(2, good("A2", 3, 0)));
    goods.insert(pair<int, good>(3, good("A3", 4, 0)));
    goods.insert(pair<int, good>(4, good("A4", 5, 0)));
    goods.insert(pair<int, good>(5, good("A5", 8, 0)));
    goods.insert(pair<int, good>(6, good("A6", 6, 0)));
    
    //初始化钱
    all_moneys.insert(pair<int, money>(1, money(1, 0)));
    all_moneys.insert(pair<int, money>(2, money(2, 0)));
    all_moneys.insert(pair<int, money>(5, money(5, 0)));
    all_moneys.insert(pair<int, money>(10, money(10, 0)));
    
    //用户投入的金额
    user_amount = 0;
}
//处理r 命令相关
bool parse_init(char* str){
    str += 2;
    char* tmp = str;
    int kind = 1;
    while(*str){
        if (*str == '-' || *str == ' '){
            *str = '\0';
            //初始化商品
            if (kind <= 6){
                goods[kind].count = stoi(tmp);
            }
            else if (kind == 7){
                all_moneys[1].count = stoi(tmp);
            }
            else if (kind == 8){
                all_moneys[2].count = stoi(tmp);
            }
            else if (kind == 9){
                all_moneys[5].count = stoi(tmp);
                
                //最后一个没有分隔符
                str++;
                all_moneys[10].count = stoi(str);
                break;
            }
            kind++;
            
            str += 1;
            tmp = str;
            continue;
        }
        str++;
    } 
    cout << "S001:Initialization is successful" << endl;
    return true;
}
//获取单个物品数量
//A1 ~ A6 去掉A转换成数字
int get_good_count(int i){
    return goods[i].count;
}
//获取所有物品数量
int get_goods_count(){
    int ret = 0;
    for (auto kv: goods){
        ret += kv.second.count;
    }
    
    return ret;
}
//处理p命令
bool parse_pay(char* str){
    if (strlen(str) < 3){
        return false;
    }
    str += 2;
    int amount = stoi(str);
    if (!(amount == 1 || amount == 2 || amount == 5 || amount == 10)){
        cout << "E002:Denomination error" << endl;
        return false;
    }
    if (all_moneys[1].count + all_moneys[2].count * 2 < amount){
        cout << "E003:Change is not enough, pay fail" << endl;
        return false;
    } 
    if (get_goods_count() <= 0){
        cout << "E005:All the goods sold out" << endl;
        return false;
    }
    //投币成功
    user_amount += amount;
    cout << "S002:Pay success,balance=" << user_amount << endl;
    //钱盒中相应面额增加
    all_moneys[amount].count++;
    return true;
}
//退钱
int pay_back(){
    //计算需要找令的各面额情况
    int one = 0;
    int two = 0;
    int five = 0;
    int ten = 0;
    if (user_amount >= 10 && all_moneys[10].count > 1){
        //计算找零的十元张数
        int need = user_amount / 10;
        need = need < all_moneys[10].count? need: all_moneys[10].count;
        
        ten = need;
        user_amount -= ten * 10;
        all_moneys[10].count -= need;
    }
    if (user_amount >= 5 && all_moneys[5].count > 1){
        //计算找零的十元张数
        int need = user_amount / 5;
        need = need < all_moneys[5].count? need: all_moneys[5].count;
        
        five = need;
        user_amount -= five * 5;
        all_moneys[5].count -= need;
    }
    if (user_amount >= 2 && all_moneys[2].count > 1){
        //计算找零的十元张数
        int need = user_amount / 2;
        need = need < all_moneys[2].count? need: all_moneys[2].count;
        
        two = need;
        user_amount -= two * 2;
        all_moneys[2].count -= need;
    }
    if (user_amount >= 1 && all_moneys[1].count > 1){
        //计算找零的十元张数
        int need = user_amount;
        need = need < all_moneys[1].count? need: all_moneys[1].count;
        
        one = need;
        user_amount -= one;
        all_moneys[1].count -= need;
    }
    cout << "1 yuan coin number=" << one << endl;
    cout << "2 yuan coin number=" << two << endl;
    cout << "5 yuan coin number=" << five << endl;
    cout << "10 yuan coin number=" << ten << endl;
    
    //退钱完成后用户的钱归零(是否需要归零)
    //user_amount -= ten * 10 + five * 5 + two * 2 + one;
    
    return ten * 10 + five * 5 + two * 2 + one;
}
//退出命令c
bool parse_payback(){
    if (user_amount <= 0){
        cout << "E009:Work failure" << endl;
        return false;
    }
    
    return pay_back() > 0;
}
void show_money(){
    cout << "1 yuan coin number=" << all_moneys[1].count << endl;
    cout << "2 yuan coin number=" << all_moneys[2].count << endl;
    cout << "5 yuan coin number=" << all_moneys[5].count << endl;
    cout << "10 yuan coin number=" << all_moneys[10].count << endl;
}
void show_goods(){
    //获取所有商品到列表中
    std::list<good> lst;
    for (auto kv: goods){
        lst.push_back(kv.second);
    }
    //排序
    lst.sort([](good a, good b)->bool {
        if (a.count < b.count){
            return false;
        }
        else if (a.count == b.count){
            return a.name < b.name;
        }
        
        return true;
    });
    //输出
    for (std::list<good>::iterator it = lst.begin(); it != lst.end(); it++){
        cout << it->name << " " << it->price << " " << it->count << endl;
    }
}
//查询命令
bool parse_query(char* str){
    if (strlen(str) < 3){
        cout << "E010:Parameter error" << endl;
        return false;
    }  
    str += 2;
    int parm = stoi(str);
    if (parm != 0 && parm != 1){
        cout << "E010:Parameter error" << endl;
        return false;
    }
    if (parm == 1){
        show_money();
    }
    else{
        show_goods();
    }
    
    return true;
}
//买
bool parse_buy(char* str){
    str += 2;
    if (str[0] != 'A' || !(str[1] >= '1' && str[1] <= '6')){
        cout << "E006:Goods does not exist" << endl;
        return false;
    } 
    int i = str[1] - '0';
    if (goods[i].count <= 0){
        cout << "E007:The goods sold out" << endl;
        return false;
    }
    if (user_amount < goods[i].price){
        cout << "E008:Lack of balance" << endl;
        return false;
    } 
    //用户扣钱
    user_amount -= goods[i].price;
    //物品数量减
    goods[i].count--;
    cout << "S003:Buy success,balance=" << user_amount << endl;
    return true;
}
/*
 r 初始化命令: r A1 数量 -A2 数量 -A3 数量 -A4 数量 -A5 数量 -A6 数量 1 元张数 -2 元张数 -5 元张数 -10 元张数
 p 投币命令:   p 钱币面额
 b 购买命令:   b 商品名称
 c 退币   :   c
 q 查询命令:   q 0 / q 1
 */
bool parse_cmd(char* str)
{
    switch(str[0])
    {
        case 'r':
            return parse_init(str);
            break;
        case 'p':
            return parse_pay(str);
            break;
        case 'b':
            return parse_buy(str);
            break;
        case 'c':
            return parse_payback();
            break;
        case 'q':
            return parse_query(str);
            break;
    }  
    return true;
}
int main()
{
    //初始化贩售机
    init();
    while(true)
    {
        char* inp = new char[1026];
        if (cin.getline(inp, 1025))
        {
            //用户金额清空
            user_amount = 0;      
            //遍历拆分单词
            char *start, *tmp;
            start = tmp = inp;        
            while(*tmp)
            {
                //分隔单词和处理
                if (*tmp == ';')
                {
                    *tmp = '\0';
                    
                    //进行处理
                    parse_cmd(start);
                    
                    start = tmp + 1;
                    tmp = start;
                    continue;
                }
                
                tmp++;
                if (*tmp == '\0')
                {
                    parse_cmd(start);
                }
            }
        }
        else
        {
            delete[] inp;
            break;
        }
    }
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

画弋

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值