#include <iostream>
#include <map>
#include <list>
#include <string>
#include <vector>
using namespace std;
struct OrderNode
{
int order_id;
char type; // B: 买,S:卖
int num; //股票数量
int price; //价格
OrderNode(int order_id, char type, int num, int price) : order_id(order_id), type(type), num(num), price(price) {}
};
using OrderList = list<OrderNode>;
using PriceMap = map<int, OrderList>;
using OrderMap = map<int, OrderNode>;
struct Context
{
PriceMap buy_map;
PriceMap sell_map;
OrderMap order_map;
void print() const
{
cout << "buy list:" << endl;
for (auto it : buy_map)
{
cout << it.first << ": ";
for (auto i : it.second)
{
cout << i.num << ", ";
}
cout << endl;
}
cout << "sell list:" << endl;
for (auto it : sell_map)
{
cout << it.first << ": ";
for (auto i : it.second)
{
cout << i.num << ", ";
}
cout << endl;
}
}
};
class ExeOrder
{
public:
// ExeOrder() {};
virtual bool Exe(int order_id, char type, int num, int price) = 0;
};
class AddOrder : public ExeOrder
{
public:
AddOrder(Context *ctx) : ctx_(ctx) {}
bool Exe(int order_id, char type, int num, int price)
{
PriceMap *price_map;
if (deal(type, num, price))
{
cout << "deal done :"
<< "A," << order_id << "," << type << "," << num << "," << price << endl;
cout <<"1050 ::"<< &ctx_->sell_map[1050].front().num<<endl;
return true;
}
if (type == 'B')
{
price_map = &ctx_->buy_map;
}
else
{
price_map = &ctx_->sell_map;
}
OrderNode item(order_id, type, num, price);
if (price_map->find(price) == price_map->end())
{
price_map->insert(make_pair(price, OrderList()));
}
price_map->at(price).push_back(item);
ctx_->order_map.insert(make_pair(order_id, item));
return true;
}
bool deal(char type, int &num, int price)
{
PriceMap *price_map;
if (type == 'B')
{
price_map = &ctx_->sell_map;
}
else
{
price_map = &ctx_->buy_map;
}
if (price_map->find(price) == price_map->end())
{
return false;
}
OrderList* order_list = &price_map->at(price);
for (auto it : *order_list)
{
if (it.num > num)
{
it.num = it.num - num;
cout << "deal done :"
<< "aaa," << it.order_id << "," << type << "," << num << "," << price << endl;
num = 0;
break;
}
if (it.num == num)
{
cout << "deal done :"
<< "bbb," << it.order_id << "," << type << "," << num << "," << price << endl;
ctx_->order_map.erase(it.order_id);
order_list->pop_front();
if (order_list->size() == 0)
{
price_map->erase(price);
}
num = 0;
break;
}
num -= it.num;
ctx_->order_map.erase(it.order_id);
order_list->pop_front();
}
if (num)
{
return false;
}
cout <<"testbbb"<<order_list->size()<<endl;
return true;
}
Context *ctx_;
};
class DelOrder : public ExeOrder
{
public:
DelOrder(Context *ctx) : ctx_(ctx) {}
bool Exe(int order_id, char type, int num, int price)
{
PriceMap *price_map;
if (type == 'B')
{
price_map = &ctx_->buy_map;
}
else
{
price_map = &ctx_->sell_map;
}
if (ctx_->order_map.find(order_id) == ctx_->order_map.end())
{
cout << "delete cmd ("
<< "X," << order_id << "," << type << "," << num << "," << price << ") failed" << endl;
return false;
}
auto it = ctx_->order_map.at(order_id);
if (it.num == num && it.price == price && it.type == type)
{
price_map->at(price).remove_if([num, price, type](OrderNode it){
if (it.num == num && it.price == price && it.type == type)
{
return true;
}
return false;
});
ctx_->order_map.erase(order_id);
cout << "delete cmd ("
<< "X," << order_id << "," << type << "," << num << "," << price << ") success" << endl;
return true;
}
cout << "delete cmd ("
<< "X," << order_id << "," << type << "," << num << "," << price << ") failed" << endl;
return false;
}
Context *ctx_;
};
using ExeMap = std::map<char, ExeOrder *>;
void registerCommand(ExeMap& m_exe_map_, Context* m_ctx_)
{
m_exe_map_.insert(make_pair('A', new AddOrder(m_ctx_)));
m_exe_map_.insert(make_pair('X', new DelOrder(m_ctx_)));
}
class ExeManage
{
public:
ExeManage()
{
registerCommand(m_exe_map_, &m_ctx_);
}
~ExeManage()
{
for (auto it : m_exe_map_)
{
delete it.second;
}
m_exe_map_.clear();
}
void do_it(const string &cmd)
{
char cmd_type = '\0';
int order_id = 0;
char type = '\0';
int num = 0;
int price = 0;
sscanf(cmd.c_str(), "%c,%d,%c,%d,%d", &cmd_type, &order_id, &type, &num, &price);
printf("%c,%d,%c,%d,%d\n", cmd_type, order_id, type, num, price);
if (m_exe_map_.find(cmd_type) == m_exe_map_.end())
{
cout << "command error11, " << cmd << endl;
return;
}
m_exe_map_[cmd_type]->Exe(order_id, type, num, price);
cout<<"test"<<m_ctx_.sell_map.size()<<endl;
}
void print()
{
m_ctx_.print();
}
private:
Context m_ctx_;
ExeMap m_exe_map_;
};
int main()
{
string s[] = {"A,100000,S,1,1075","A,100001,B,9,1000","A,100002,B,30,975","A,100003,S,10,1050","A,100004,B,10,950","A,100005,S,2,1025",
"A,100006,B,1,1000","X,100004,B,10,950","A,100007,S,5,1025","A,100008,B,3,1050","X,100008,B,3,1050","X,100005,S,2,1025"};
vector<string> input(s, s + 12);
ExeManage exe;
for (auto it : input)
{
exe.do_it(it);
}
exe.print();
}
#include <iostream>
#include <map>
#include <list>
#include <string>
#include <vector>
using namespace std;
struct OrderNode
{
    int order_id;
    char type; // B: 买，S:卖
    int num;   //股票数量
    int price; //价格
    OrderNode(int order_id, char type, int num, int price) : order_id(order_id), type(type), num(num), price(price) {}
};
using OrderList = list<OrderNode>;
using PriceMap = map<int, OrderList>;
using OrderMap = map<int, OrderNode>;

struct Context
{
    PriceMap buy_map;
    PriceMap sell_map;
    OrderMap order_map;
    void print() const
    {
        cout << "buy list:" << endl;
        for (auto it : buy_map)
        {
            cout << it.first << ": ";
            for (auto i : it.second)
            {
                cout << i.num << ", ";
            }
            cout << endl;
        }
        cout << "sell list:" << endl;
        for (auto it : sell_map)
        {
            cout << it.first << ": ";
            for (auto i : it.second)
            {
                cout << i.num << ", ";
            }
            cout << endl;
        }
    }
};

class ExeOrder
{
public:
    // ExeOrder() {};
    virtual bool Exe(int order_id, char type, int num, int price) = 0;
};

class AddOrder : public ExeOrder
{
public:
    AddOrder(Context *ctx) : ctx_(ctx) {}
    bool Exe(int order_id, char type, int num, int price)
    {
        PriceMap *price_map;
        if (deal(type, num, price))
        {
            cout << "deal done :"
                 << "A," << order_id << "," << type << "," << num << "," << price << endl;
            cout <<"1050 ::"<< &ctx_->sell_map[1050].front().num<<endl;
            return true;
        }
        if (type == 'B')
        {
            price_map = &ctx_->buy_map;
        }
        else
        {
            price_map = &ctx_->sell_map;
        }
        OrderNode item(order_id, type, num, price);
        if (price_map->find(price) == price_map->end())
        {
            price_map->insert(make_pair(price, OrderList()));
        }
        price_map->at(price).push_back(item);
        ctx_->order_map.insert(make_pair(order_id, item));
        return true;
    }
    bool deal(char type, int &num, int price)
    {
        PriceMap *price_map;
        if (type == 'B')
        {
            price_map = &ctx_->sell_map;
        }
        else
        {
            price_map = &ctx_->buy_map;
        }
        if (price_map->find(price) == price_map->end())
        {
            return false;
        }
        OrderList* order_list = &price_map->at(price);
        for (auto it : *order_list)
        {
            if (it.num > num)
            {
                it.num = it.num - num;
                cout << "deal done :"
                 << "aaa," << it.order_id << "," << type << "," << num << "," << price << endl;
                num = 0;
                break;
            }
            if (it.num == num)
            {
                cout << "deal done :"
                 << "bbb," << it.order_id << "," << type << "," << num << "," << price << endl;
                ctx_->order_map.erase(it.order_id);
                order_list->pop_front();
                if (order_list->size() == 0)
                {
                    price_map->erase(price);
                }
                num = 0;
                break;
            }
            num -= it.num;
            ctx_->order_map.erase(it.order_id);
            order_list->pop_front();
        }
        if (num)
        {
            return false;
        }
        cout <<"testbbb"<<order_list->size()<<endl;
        return true;
    }
    Context *ctx_;
};

class DelOrder : public ExeOrder
{
public:
    DelOrder(Context *ctx) : ctx_(ctx) {}
    bool Exe(int order_id, char type, int num, int price)
    {
        PriceMap *price_map;
        if (type == 'B')
        {
            price_map = &ctx_->buy_map;
        }
        else
        {
            price_map = &ctx_->sell_map;
        }
        if (ctx_->order_map.find(order_id) == ctx_->order_map.end())
        {
            cout << "delete cmd ("
                 << "X," << order_id << "," << type << "," << num << "," << price << ") failed" << endl;
            return false;
        }
        auto it = ctx_->order_map.at(order_id);
        if (it.num == num && it.price == price && it.type == type)
        {
            price_map->at(price).remove_if([num, price, type](OrderNode  it){
                if (it.num == num && it.price == price && it.type == type)
                {
                    return true;
                }
                return false;
            });
            ctx_->order_map.erase(order_id);
            cout << "delete cmd ("
                 << "X," << order_id << "," << type << "," << num << "," << price << ") success" << endl;
            return true;
        }
        cout << "delete cmd ("
             << "X," << order_id << "," << type << "," << num << "," << price << ") failed" << endl;
        return false;
    }
    Context *ctx_;
};
using ExeMap = std::map<char, ExeOrder *>;
void registerCommand(ExeMap& m_exe_map_, Context* m_ctx_)
{
    m_exe_map_.insert(make_pair('A', new AddOrder(m_ctx_)));
    m_exe_map_.insert(make_pair('X', new DelOrder(m_ctx_)));
}
class ExeManage
{
public:
    ExeManage()
    {
        registerCommand(m_exe_map_, &m_ctx_);
    }
    ~ExeManage()
    {
        for (auto it : m_exe_map_)
        {
            delete it.second;
        }
        m_exe_map_.clear();
    }
    void do_it(const string &cmd)
    {
        char cmd_type = '\0';
        int order_id = 0;
        char type = '\0';
        int num = 0;
        int price = 0;
        sscanf(cmd.c_str(), "%c,%d,%c,%d,%d", &cmd_type, &order_id, &type, &num, &price);
        printf("%c,%d,%c,%d,%d\n", cmd_type, order_id, type, num, price);
        if (m_exe_map_.find(cmd_type) == m_exe_map_.end())
        {
            cout << "command error11, " << cmd << endl;
            return;
        }
        m_exe_map_[cmd_type]->Exe(order_id, type, num, price);
        cout<<"test"<<m_ctx_.sell_map.size()<<endl;
    }
    void print()
    {
        m_ctx_.print();
    }

private:
    Context m_ctx_;
    ExeMap m_exe_map_;
};
int main()
{
    string s[] = {"A,100000,S,1,1075","A,100001,B,9,1000","A,100002,B,30,975","A,100003,S,10,1050","A,100004,B,10,950","A,100005,S,2,1025",
		"A,100006,B,1,1000","X,100004,B,10,950","A,100007,S,5,1025","A,100008,B,3,1050","X,100008,B,3,1050","X,100005,S,2,1025"};
    vector<string> input(s, s + 12);
    ExeManage exe;
    for (auto it : input)
    {
        exe.do_it(it);
    }
    exe.print();
}