Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- #include <array>
- #include <cassert>
- #include <chrono>
- #include <iomanip>
- #include <iostream>
- #include <sstream>
- #include <string>
- #include <tuple>
- #include <unordered_map>
- #include <vector>
- using namespace std;
- class VehiclePlate {
- private:
- auto AsTuple() const {
- return tie(letters_, digits_, region_);
- }
- public:
- bool operator==(const VehiclePlate& other) const {
- return AsTuple() == other.AsTuple();
- }
- VehiclePlate(char l0, char l1, int digits, char l2, int region)
- : letters_{l0, l1, l2}
- , digits_(digits)
- , region_(region) {
- }
- string ToString() const {
- ostringstream out;
- out << letters_[0] << letters_[1];
- // чтобы дополнить цифровую часть номера слева нулями
- // до трёх цифр, используем подобные манипуляторы:
- // setfill задаёт символ для заполнения,
- // right задаёт выравнивание по правому краю,
- // setw задаёт минимальное желаемое количество знаков
- out << setfill('0') << right << setw(3) << digits_;
- out << letters_[2] << setw(2) << region_;
- return out.str();
- }
- private:
- array<char, 3> letters_;
- int digits_;
- int region_;
- };
- ostream& operator<<(ostream& out, VehiclePlate plate) {
- out << plate.ToString();
- return out;
- }
- // возьмите реализацию хешера из прошлого задания
- class VehiclePlateHasher {
- public:
- size_t operator()(const VehiclePlate& plate) const {
- return hash<string>()(plate.ToString());
- }
- };
- // выбросьте это исключение в случае ошибки парковки
- struct ParkingException {};
- template <typename Clock>
- class Parking {
- // при обращении к типу внутри шаблонного параметра мы обязаны использовать
- // typename; чтобы этого избежать, объявим псевдонимы для нужных типов
- using Duration = typename Clock::duration;
- using TimePoint = typename Clock::time_point;
- public:
- Parking(int cost_per_second) : cost_per_second_(cost_per_second) {}
- // запарковать машину с указанным номером
- void Park(VehiclePlate car) {
- // место для вашей реализации
- if (now_parked_.count(car)) {
- throw ParkingException;
- }
- now_parked_[car] = "сейчас" // запарковали машину
- }
- // забрать машину с указанным номером
- void Withdraw(const VehiclePlate& car) {
- // место для вашей реализации
- if (!now_parked_.count(car)) {
- throw ParkingException;
- }
- complete_parks_[car] += "сейчас" - now_parked_.at(car); // накопили машине секунды
- now_parked_.erase(car); // "забрали" машину с парковки
- }
- // получить счёт за конкретный автомобиль
- int64_t GetCurrentBill(const VehiclePlate& car) const {
- // место для вашей реализации
- bool isCarInParking = now_parked_.count(car);
- bool HaveCarDebt = complete_parks_.count(car);
- Duration total;
- if (!isCarInParking && !HaveCarDebt) { // машины нет в системе
- return 0;
- }
- if (isCarInParking) { // если машина на парковке, вписать ей текущий долг
- total += "сейчас" - now_parked_.at(car);
- if (HaveCarDebt) { // если машина имеет сформированный долг, вписать и его
- total += complete_parks_.at(car);
- }
- return chrono::duration_cast<chrono::seconds>(total).count() * cost_per_second_;
- }
- // завершить расчётный период
- // те машины, которые находятся на парковке на данный момент, должны
- // остаться на парковке, но отсчёт времени для них начинается с нуля
- unordered_map<VehiclePlate, int64_t, VehiclePlateHasher> EndPeriodAndGetBills() {
- // место для вашей реализации
- // сколько машины "настояли" -- записать в complete_parks_
- unordered_map<VehiclePlate, int64_t, VehiclePlateHasher> debts;
- for (const auto& [car, time_start] : now_parked_) {
- complete_parks_[car] += "сейчас" - time_start;
- time_start = 0;
- debts[car] = chrono::duration_cast<chrono::seconds>(complete_parks_[car]).count() * cost_per_second_;
- }
- return debts;
- }
- // не меняйте этот метод
- auto& GetNowParked() const {
- return now_parked_;
- }
- // не меняйте этот метод
- auto& GetCompleteParks() const {
- return complete_parks_;
- }
- private:
- int cost_per_second_;
- unordered_map<VehiclePlate, TimePoint, VehiclePlateHasher> now_parked_;
- unordered_map<VehiclePlate, Duration, VehiclePlateHasher> complete_parks_;
- };
- // эти часы удобно использовать для тестирования
- // они покажут столько времени, сколько вы задали явно
- class TestClock {
- public:
- using time_point = chrono::system_clock::time_point;
- using duration = chrono::system_clock::duration;
- static void SetNow(int seconds) {
- current_time_ = seconds;
- }
- static time_point now() {
- return start_point_ + chrono::seconds(current_time_);
- }
- private:
- inline static time_point start_point_ = chrono::system_clock::now();
- inline static int current_time_ = 0;
- };
- int main() {
- Parking<TestClock> parking(10);
- TestClock::SetNow(10);
- parking.Park({'A', 'A', 111, 'A', 99});
- TestClock::SetNow(20);
- parking.Withdraw({'A', 'A', 111, 'A', 99});
- parking.Park({'B', 'B', 222, 'B', 99});
- TestClock::SetNow(40);
- assert(parking.GetCurrentBill({'A', 'A', 111, 'A', 99}) == 100);
- assert(parking.GetCurrentBill({'B', 'B', 222, 'B', 99}) == 200);
- parking.Park({'A', 'A', 111, 'A', 99});
- TestClock::SetNow(50);
- assert(parking.GetCurrentBill({'A', 'A', 111, 'A', 99}) == 200);
- assert(parking.GetCurrentBill({'B', 'B', 222, 'B', 99}) == 300);
- assert(parking.GetCurrentBill({'C', 'C', 333, 'C', 99}) == 0);
- parking.Withdraw({'B', 'B', 222, 'B', 99});
- TestClock::SetNow(70);
- {
- // проверим счёт
- auto bill = parking.EndPeriodAndGetBills();
- // так как внутри макроса используется запятая,
- // нужно заключить его аргумент в дополнительные скобки
- assert((bill
- == unordered_map<VehiclePlate, int64_t, VehiclePlateHasher>{
- {{'A', 'A', 111, 'A', 99}, 400},
- {{'B', 'B', 222, 'B', 99}, 300},
- }));
- }
- TestClock::SetNow(80);
- {
- // проверим счёт
- auto bill = parking.EndPeriodAndGetBills();
- // так как внутри макроса используется запятая,
- // нужно заключить его аргумент в дополнительные скобки
- assert((bill
- == unordered_map<VehiclePlate, int64_t, VehiclePlateHasher>{
- {{'A', 'A', 111, 'A', 99}, 100},
- }));
- }
- try {
- parking.Park({'A', 'A', 111, 'A', 99});
- assert(false);
- }
- catch (ParkingException) {
- }
- try {
- parking.Withdraw({'B', 'B', 222, 'B', 99});
- assert(false);
- }
- catch (ParkingException) {
- }
- cout << "Success!"s << endl;
- }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement