{ return cbase + (v–vbase)*scale; } // см. раздел 21.4
};
Мы хотим создать класс, поскольку вычисление зависит от трех констант, которые не обязательно
повторяются. В этих условиях можно определить следующие функции:
Scale xs(xoffset,base_year,xscale);
Scale ys(ymax–yoffset,0,–yscale);
Обратите внимание на то, что мы сделали масштабирующий множитель
ys
отрицательным, чтобы отразить тот факт, что координаты y возрастают в направлении вниз, хотя мы привыкли, что они возрастают в направлении вверх. Теперь можем использовать функцию
xs
для преобразования лет в координату
x
. Аналогично можно использовать функцию
ys
для преобразования процентов в координату
y
.
15.6.4. Построение графика
Итак, у нас есть все предпосылки для создания элегантной программы. Начнем с создания окна и размещения осей.
Line current_year(Point(xs(2008),ys(0)),Point(xs(2008),ys(100)));
current_year.set_style(Line_style::dash);
Оси пересекаются в точке
Point(xoffset,ymax–yoffset)
, соответствующей паре (
1960,0
). Обратите внимание на то, как деления отражают данные. На оси y отложено десять делений, каждое из которых соответствует десяти процентам населения. На оси x каждое деление соответствует десяти годам. Точное количество делений вычисляется по значениям переменных
base_year
и
end_year
, поэтому, если мы изменим диапазон, оси автоматически будут вычислены заново. Это одно из преимуществ отсутствия “магических констант” в коде. Метка на оси x нарушает это правило, потому что размещать метки, пока числа на окажутся на правильных позициях, бесполезно. Возможно, лучше было бы задать набор индивидуальных меток для каждого деления.
Пожалуйста, обратите внимание на любопытное форматирование этой метки, представляющей собой строку. Мы использовали два смежных строковых литерала.
"year 1960 1970 1980 1990"
"2000 2010 2020 2030 2040"
Компилятор конкатенирует такие строки, поэтому это эквивалентно
Этот трюк может оказаться полезным при размещении длинных строк, поскольку он позволяет сохранить читабельность текста.
Объект
current_year
соответствует вертикальной линии, разделяющей реальные данные и прогнозируемые. Обратите внимание на то, как используются функции
xs
и
ys
для правильного размещения и масштабирования этой линии.
Построив оси, мы можем обработать данные. Определим три объекта класса
Open_polyline
и заполним их в цикле чтения.
Open_polyline children;
Open_polyline adults;
Open_polyline aged;
Distribution d;
while (ifs>>d) {
if (d.year<base_year || end_year<d.year)
error("Год не попадает в диапазон");
if (d.young+d.middle+d.old != 100)
error("Проценты не согласованы");
int x = xs(d.year);
children.add(Point(x,ys(d.young)));
adults.add(Point(x,ys(d.middle)));
aged.add(Point(x,ys(d.old)));
}
Использование функций
xs
и
ys
делает проблему масштабирования и размещения данных тривиальной. “Небольшие классы”, такие как
Scale
, могут оказаться очень важными для упрощения кода и устранения лишних повторов — тем самым они повышают читабельность и увеличивают шансы на создание правильной программы.
Для того чтобы графики были более ясными, мы пометили их и раскрасили в разные цвета.
Text children_label(Point(20,children.point(0).y),"age 0-15");
children.set_color(Color::red);
children_label.set_color(Color::red);
Text adults_label(Point(20,adults.point(0).y),"age 15-64");
adults.set_color(Color::blue);
adults_label.set_color(Color::blue);
Text aged_label(Point(20,aged.point(0).y),"age 65+");
aged.set_color(Color::dark_green);
aged_label.set_color(Color::dark_green);
В заключение нам нужно связать разные объекты класса
Shape
с объектом класса
Window
и передать управление системе графического пользовательского интерфейса (см. раздел 15.2.3).