Лафа с пешком до работы кончилась и я решил узнать какой всё же проездной на метро выгоднее. Естественно, лёгких путей мы не ищем и на интуицию полагаться не будем.

Итак, модель. Для простоты смоделируем “год” в 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; }
Для сборки нужен компилятор с поддержкой C++11. Актуальную версию (если замечу баги или неточности) можно найти на code.sunchaser.info

Leave a Reply

Your email address will not be published. Required fields are marked *

This site uses Akismet to reduce spam. Learn how your comment data is processed.