Итак, модель. Для простоты смоделируем "год" в 360 дней из 12 месяцев по 30 дней. Ну, для простоты. Для примерного результата этого достаточно. И заставим модель кататься по проездным этот год
- Не брались проездные на 90 дней — один из них не может быть выгоднее жетонов (1890/70 = 27 рублей за каждую поездку), другой — подорожника (1560/60 = 26 рублей за каждую поездку)
- Не брался месячный проездной на 70 поездок. Сначала я его неправильно смоделировал, а потом заломало/ Про него — отдельно
Результат:
Абсцисса — число поездок в неделю, ордината — средняя стоимость одной поездки
- Не стоит париться с проездными, если вы ездите меньше раза в день. Думаю, это и так очевидно.
- Подорожник рулит необычайно
- Нет особого смысла в "толстых" проездных (25/15, 50/30) — они незначительно выгоднее вариантов "полегче" на тот же период
- Проездной на 20 рулит над подорожником с 9 до 14 поездок
- Проездной на 40 рулит над подорожником с 9 до 17 поездок и безусловно рулит над проездным на 20
- Если вы ездите больше 1 и меньше 3 раз в день, скорее всего вам будет выгоднее проездной на 40 поездок
- В остальных случаях берите подорожник. Вообще, берите подорожник, с ним не надо паритсья с проездными
- Проездной на месяц привязан к календарному месяцу. Может он и выгоден, но мне с такой привязкой париться неудобно
- Все эти тарифы — или от балды, или разводилово
Код модели под катом (C++)
/**
* Saint Petersburg subway payments calc
*/
#include
#include
class Calendar
{
int day;
public:
Calendar(): day(0) {}
int getDay()
{
return this->day;
}
int getMonth()
{
return this->day / 30;
}
int nextDay()
{
this->day++;
}
};
class TicketSystem
{
double sum;
double last_payment;
int rides;
virtual double processPayment() = 0;
protected:
Calendar *cal;
public:
TicketSystem(Calendar* c): cal(c), sum(0), last_payment(0), rides(0) {}
double ride()
{
this->rides++;
this->sum += this->last_payment = this->processPayment();
return this->last_payment;
}
double getLastPayment()
{
return this->last_payment;
}
double getCurrentSum()
{
return this->sum;
}
int getCurrentRides()
{
return this->rides;
}
};
class PayByToken: public TicketSystem
{
double token_price;
double processPayment()
{
return this->token_price;
}
public:
PayByToken(Calendar* c, double token_price = 27): token_price(token_price), TicketSystem(c) {}
};
class PayByPodorozhnik: public TicketSystem
{
int current_month;
int rides;
double processPayment()
{
if (this->cal->getMonth() != this->current_month)
{
this->current_month = this->cal->getMonth();
this->rides = 0;
}
this->rides++;
if (rides > 40)
return 19;
else if (rides > 30)
return 20.5;
else if (rides > 20)
return 22;
else if (rides > 10)
return 24.5;
else
return 26;
}
public:
PayByPodorozhnik(Calendar* c): TicketSystem(c), current_month(-1), rides(-1) {}
};
class PayByCard: public TicketSystem
{
double price;
int rides, days, days_left, rides_left, prev_day;
public:
PayByCard(Calendar *cal, double price, int rides, int days):
TicketSystem(cal), rides(rides), days(days), price(price), days_left(0), rides_left(0), prev_day(-1) {}
double processPayment()
{
if (prev_day != cal->getDay())
{
prev_day = cal->getDay();
days_left--;
}
rides_left--;
if (days_left <= 0 || rides_left <= 0)
{
rides_left = rides;
days_left = days;
return price;
}
return 0;
}
};
typedef std::list MethList;
void run_sim(double);
int main()
{
for (double rides = 1.0/7; rides < 8; rides += 1.0/7)
{
run_sim(rides);
}
return 0;
}
void run_sim(double rides_per_day)
{
double rides = 0;
Calendar *cal = new Calendar();
MethList list;
list.push_back(new PayByToken(cal, 27));
list.push_back(new PayByPodorozhnik(cal));
//list.push_back(new PayByCard(cal, 1890, 70, 90));
//list.push_back(new PayByCard(cal, 1560, 60, 90));
list.push_back(new PayByCard(cal, 230, 10, 7));
list.push_back(new PayByCard(cal, 430, 20, 15));
list.push_back(new PayByCard(cal, 530, 25, 15));
list.push_back(new PayByCard(cal, 830, 40, 30));
list.push_back(new PayByCard(cal, 1025, 50, 30));
//list.push_back(new PayByCard(cal, 1290, 70, 30)); // wrong!!!
while(cal->getDay() < 360)
{
if (rides > 1)
{
rides -= 1;
for (TicketSystem *token: list)
{
token->ride();
}
}
else
{
rides += rides_per_day;
cal->nextDay();
}
}
std::cout << rides_per_day * 7 << "\t";
for (TicketSystem *token: list)
{
std::cout << token->getCurrentSum() / token->getCurrentRides() << "\t";
}
std::cout << std::endl;
for (TicketSystem *token: list)
{
delete token;
token = nullptr;
}
delete cal;
}
Актуальную версию можно найти на code.sunchaser.info