Чистый код. Создание, анализ и рефакторинг — страница 93 из 94

146     return date.getOrdinalDay() == getOrdinalDay();

147   }

148

149   public int hashCode() {

150     return getOrdinalDay();

151   }

152

153   public int compareTo(Object other) {

154     return daysSince((DayDate) other);

155   }

156

157   private int calcOrdinal(int day, Month month, int year) {

158     int leapDaysForYear = DateUtil.leapYearCount(year - 1);

159     int daysUpToYear = (year - MINIMUM_YEAR_SUPPORTED) * 365 + leapDaysForYear;

160     int daysUpToMonth = AGGREGATE_DAYS_TO_END_OF_PRECEDING_MONTH[month.toInt()];

161     if (DateUtil.isLeapYear(year) && month.toInt() > FEBRUARY.toInt())

162       daysUpToMonth++;

163     int daysInMonth = day - 1;

164     return daysUpToYear + daysUpToMonth + daysInMonth + EARLIEST_DATE_ORDINAL;

165   }

166

167   private void calcDayMonthYear() {

168     int days = ordinalDay - EARLIEST_DATE_ORDINAL;

169     int overestimatedYear = MINIMUM_YEAR_SUPPORTED + days / 365;

170     int nonleapdays = days - DateUtil.leapYearCount(overestimatedYear);

171     int underestimatedYear = MINIMUM_YEAR_SUPPORTED + nonleapdays / 365;

172

173     year = huntForYearContaining(ordinalDay, underestimatedYear);

174     int firstOrdinalOfYear = firstOrdinalOfYear(year);

175     month = huntForMonthContaining(ordinalDay, firstOrdinalOfYear);

176     day = ordinalDay - firstOrdinalOfYear - daysBeforeThisMonth(month.toInt());

177   }

178

179   private Month huntForMonthContaining(int anOrdinal, int firstOrdinalOfYear) {

180     int daysIntoThisYear = anOrdinal - firstOrdinalOfYear;

181     int aMonth = 1;

182     while (daysBeforeThisMonth(aMonth) < daysIntoThisYear)

183       aMonth++;

184

185     return Month.fromInt(aMonth - 1);

186   }

187

188   private int daysBeforeThisMonth(int aMonth) {

189     if (DateUtil.isLeapYear(year))

190       return LEAP_YEAR_AGGREGATE_DAYS_TO_END_OF_PRECEDING_MONTH[aMonth] - 1;

191     else

192       return AGGREGATE_DAYS_TO_END_OF_PRECEDING_MONTH[aMonth] - 1;

193   }

194

195   private int huntForYearContaining(int anOrdinalDay, int startingYear) {


Листинг Б.16 (продолжение)

196     int aYear = startingYear;

197     while (firstOrdinalOfYear(aYear) <= anOrdinalDay)

198       aYear++;

199

200     return aYear - 1;

201   }

202

203   private int firstOrdinalOfYear(int year) {

204     return calcOrdinal(1, Month.JANUARY, year);

205   }

206

207   public static DayDate createInstance(Date date) {

208     GregorianCalendar calendar = new GregorianCalendar();

209     calendar.setTime(date);

210     return new SpreadsheetDate(calendar.get(Calendar.DATE),

211                                Month.fromInt(calendar.get(Calendar.MONTH) + 1),

212                                calendar.get(Calendar.YEAR));

213

214   }

215 }

Приложение B. Перекрестные ссылки

Перекрестные ссылки для «запахов кода» и эвристических правил из приложения А организованы по принципу Номер_главы-Номер_страницы.

C116–306, 16–309, 17–323
C216–309, 16–312, 16–320, 17–323
C316-<$R[P#,R16–10]>, 16–312, 16–314, 17–323
C417–323
C517–324
E117–324
E217–324
F114–266, 17–325
F217–325
F317–325
F416–304, 16–314>, 17–325
G116–307, 17–325
G216–305, 17–326
G316–305, 17–326
G416–308, 17–326
G516–308, 16–313, 16–317, 16–320, 17–327
G66–128, 16–309, 16–311, 16–315, 16–318, 16–319, 17–328
G716–309, 17–329
G816–311, 17–329
G916–311, 16–312, 16–314, 17–330
G105–107, 16–311, 17–330
G1115–295, 16–311, 16–314, 16–317, 17–330
G1216–312, 16–313, 16–314, 16–315, 16–320, 17–331
G1316–313, 16–314, 17–331
G1416–314, 17–331
G1516–315, 17–333
G1616–315, 17–333
G1716–315, 17–334, 17–337
G1816–315, 16–316, 16–317, 17–334
G1916–316, 16–317, 17–335
G2016–316, 17–335
G2116–317, 17–336
G2216–318, 17–336
G233–63, 14–265, 16–319, 17–338
G2416–320, 17–338
G2516–320, 17–339
G2617–340
G2717–340
G2815–293, 17–341
G2915–394, 17–341
G3015–295, 17–341
G3115–296, 17–342
G3215–296, 17–343
G3315–298, 17–343
G343–60, 6–128, 17–344
G355–111, 17–345
G366–126, 17–346
J116–306, 17–347
J216–308, 17–347
J316–311, 16–312, 17–348
N115–296, 16–307, 16–309, 16–310, 16–314, 16–315, 16–316, 16–319, 17–349
N216–307, 17–351
N316–312, 16–314, 17–352
N415–294, 16–316, 17–352
N52–45, 14–248, 17–353
N615–293, 17–353
N715–294, 17–354
T116–304, 16–305, 17–354
T216–304, 17–354
T316–305, 17–354
T417–355
T516–305, 16–306, 17–355
T616–305, 17–355
T716–306, 17–355
T816–306, 17–355
T917–356

Эпилог

На конференции по гибким методологиям, проходившей в Денвере в 2005 году, Элизабет Хедриксон[83] вручила мне зеленый браслет наподобие тех, которые стали такими популярными после Ланса Армстронга. На браслете было написано «Одержим тестированием». Я с гордостью носил этот браслет. С тех пор как Кент Бек научил меня методологии разработки через тестирование (TDD) в 1999 году, я действительно стал буквально одержим этой темой.