-
Notifications
You must be signed in to change notification settings - Fork 0
/
fabrics.html
340 lines (297 loc) · 32.9 KB
/
fabrics.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
<!doctype html>
<html>
<head>
<title>Про фреймворки</title>
<meta charset="utf-8">
<link rel="shortcut icon" href="/assets/images/Chi's Sweet Home.ico" type="image/x-icon">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="IE=edge;" />
<link rel="stylesheet" href="/assets/css/main.css" />
<link rel="stylesheet" href="/assets/css/pygment.css" />
<script type="text/javascript" src="https://yastatic.net/jquery/2.2.0/jquery.min.js"></script>
<script type="text/javascript" src="https://yastatic.net/underscore/1.8.3/underscore.js"></script>
</head>
<body>
<h4 class="site-name"><a href="/index.html">Магия на кончиках пальцев</a></h4>
<article>
<header style="background-image: url(
/assets/images/stardust.jpeg
);
background-position: center center;
">
<div class="content">
<h1>Про фреймворки</h1>
<div class="summary">
Несколько статей о пробемах сложности и фреймворков
</div>
</div>
</header>
<div class="text content">
<div class="info">
<div>Публикация от
<b><time pubdate datetime="2016-04-13T00:00:00">
13 Apr 2016
</time></b>
</div>
<div class="tags">
#<a href="/index.html?q=%23Braithwaite">Braithwaite</a>
#<a href="/index.html?q=%23frameworks">frameworks</a>
#<a href="/index.html?q=%23fun">fun</a>
</div>
<div class="category">
/<a href="/index.html?q=%2Flibrary">library</a>
</div>
</div>
<h2>Про фреймворки</h2>
<p><small>
<em><strong>Оригинал:</strong> <a href="http://discuss.joelonsoftware.com/default.asp?joel.3.219431">http://discuss.joelonsoftware.com/default.asp?joel.3.219431</a> (BenjiSmith)
<br>
<strong>Перевод:</strong> <a href="https://habrahabr.ru/post/141477/">https://habrahabr.ru/post/141477/</a> (truezemez)</em>
</small></p>
<p>Я собираюсь сделать Java-веб-приложение (да, это будет Java, по некоторым причинам, которые сейчас озвучивать не хочу). В процессе работы, я оцениваю кучу J2EE portlet-enabled JSR-compliant MVC role-based CMS web service application container фреймворков.</p>
<p>После кучи потраченного времени на прочтение документации, я уже готов выколоть себе глаза.</p>
<hr />
<p>Давайте представим, что я решил сделать шкафчик для специй.</p>
<p>Я уже делал небольшие поделки из дерева и думаю, что знаю, что мне нужно: немного дощечек и несколько базовых инструментов: рулетка, пила, уровень и молоток.</p>
<p>Если бы я строил целый дом, а не просто шкафчик для специй, то мне все также требовались бы рулетка, пила, уровень и молоток (не считая остального).</p>
<p>Итак, я иду в магазин за инструментами и спрашиваю продавца, где я могу найти молоток.</p>
<p>— Молоток? — спросил он. — Никто уже давно не покупает молотки. Это старомодно.</p>
<p>Я был очень удивлен и спросил почему.</p>
<p>— Ну, проблема молотков в том, что их очень много видов. Кувалды, столярные молотки, с круглым бойком и т.д. Что если Вы купите один тип молотка, а потом поймете, что Вам нужен другой? Вам придется покупать отдельный молоток для следующей задачи. Как выяснилось, большинство людей хотят иметь один молоток, который бы справлялся со всеми типами задач, с которыми можно столкнуться.</p>
<p>— Хммммм. Ну, звучит разумно. Можете показать мне такой Универсальный Молоток?</p>
<p>— Нет. Мы их больше не продаем. Они устарели.</p>
<p>— Серьезно? С Ваших слов я понял, что Универсальный Молоток – это технология будущего.</p>
<p>— Как выяснилось, если сделать только один тип молотка, способный выполнять те же задачи, что и все виды молотков, тогда он будет не очень хорош для каждой из них. Забивать гвозди кувалдой не очень эффективно. И если Вы хотите убить свою бывшую девушку, то ничто не заменит молотка с круглым бойком.</p>
<p>— Верно. Но если никто больше не покупает Универсальные Молотки и если Вы больше не продаете все старомодные типы молотков, то какие же молотки Вы продаете?</p>
<p>— Вообще-то, мы не продаем никакие молотки.</p>
<p>— Но…</p>
<p>— Наши исследования показали, что людям совсем не нужен Универсальный Молоток. Всегда лучше использовать нужный тип молотка для работы. Поэтому, мы начали продавать фабрики молотков, способные создать любой молоток, какой Вам нужен. Все что Вам нужно — это укомплектовать фабрику рабочими, запустить механизм, купить сырье, оплатить расходы и — БАЦ — у Вас есть именно такой молоток, какой Вам нужен.</p>
<p>— Но мне как-то не хочется покупать фабрику молотков…</p>
<p>— Это хорошо. Потому что мы их больше не продаем.</p>
<p>— Но Вы же только что сказали…</p>
<p>— Мы обнаружили, что большинству людей не нужна целая фабрика. Некоторым людям, например, никогда не понадобится молоток с круглым бойком. (Может у них нет бывших девушек. Или они убили их ледорубом.) Поэтому, нет смысла кому-либо покупать фабрику молотков, которая может произвести любой тип молотка.</p>
<p>— Да, похоже на то.</p>
<p>— Вместо этого мы начали продавать чертежи фабрик молотков, позволяющие нашим клиентам построить их собственные фабрики молотков, специально разработанные для производства только тех типов молотков, которые им нужны.</p>
<p>— Дайте угадаю. Вы их больше не продаете.</p>
<p>— Нет. Конечно же нет. Как выяснилось, люди не хотят строить целую фабрику, только ради того, чтоб произвести пару молотков. Оставьте строительство профессионалам, вот что я всегда говорю.</p>
<p>— И здесь я с вами соглашусь.</p>
<p>— Ага. Поэтому мы прекратили продавать эти чертежи и начали продавать фабрики фабрик молотков. Каждая фабрика фабрик молотков строится для Вас экспертами своего дела, и вам не нужно волноваться о деталях постройки фабрики. Вы получаете все преимущества обладания собственной фабрикой молотков, производящей молотки для ваших потребностей.</p>
<p>— Ну, это как-бы, не совсем…</p>
<p>— Я знаю что Вы скажете! Мы их уже больше не продаем. По каким-то причинам, довольно мало людей покупали фабрики фабрик молотков, поэтому мы придумали новое решение проблемы.</p>
<p>— Угу.</p>
<p>— Когда мы сделали шаг назад и посмотрели глобально на инфраструктуру инструментов, мы обнаружили что люди разочарованы, т.к. им нужно управлять фабрикой фабрик молотков и произведенной фабрикой молотков. Это довольно обременительно, в случае если вам еще нужно управлять фабрикой фабрик рулеток, фабрикой фабрик пил, фабрикой фабрик уровней, не говоря уже о производстве пиломатериалов. Взглянув на ситуацию, мы поняли, что это слишком сложно для тех, кто просто хочет сделать шкафчик для специй.</p>
<p>— Да. Это уж точно.</p>
<p>— Поэтому, на этой неделе мы представим единую фабрику фабрик фабрик инструментов, чтобы каждую фабрику фабрик инструментов Вы могли произвести с помощью одной, объединенной фабрики. Фабрика фабрик фабрик будет производить только те фабрики фабрик инструментов, какие Вам нужны, и каждая из этих фабрик фабрик будет производить фабрику, основанную на Ваших требованиях к инструменту. Окончательный набор инструментов, полученных в результате этого процесса, будет идеален для вашего конкретного проекта. У Вас будет именно тот молоток, который вам нужен, и рулетка, подходящая именно под эту задачу, и все это по нажатию одной кнопки (конечно вам придется немного поработать с конфигурацией, чтобы заставить все это работать именно так как вам надо).</p>
<p>— Так у вас нет никаких молотков? Совсем никаких?</p>
<p>— Нет. Если Вы по-настоящему хотите высококачественный, промышленно разработанный шкафчик для специй, вам нужно нечто более продвинутое, нежели обычный молоток из магазина.</p>
<p>— И что, все сейчас так делают? Все используют единую фабрику фабрик фабрик инструментов, всякий раз, когда им нужен молоток?</p>
<p>— Да.</p>
<p>— Ладно… Я кажется понял, что мне нужно делать. Если сейчас это делается таким образом, будет, наверное, лучше и мне научиться так делать.</p>
<p>— Очень хорошо!</p>
<p>— Эта вещь идет с документацией, верно?</p>
<h2>Остерегайтесь Тьюринговской трясины</h2>
<p><small>
<em><strong>Оригинал:</strong> <a href="http://weblog.raganwald.com/2004/10/beware-of-turing-tar-pit.html">http://weblog.raganwald.com/2004/10/beware-of-turing-tar-pit.html</a> (Reginald Braithwaite)
<br>
<strong>Перевод:</strong> <a href="https://habrahabr.ru/post/139535/">https://habrahabr.ru/post/139535/</a> (tangro), Николай Волков</em>
</small></p>
<blockquote>
<p>...в которой всё возможно, но ничего конкретного нельзя сделать просто.
-- <cite>Алан Перлис <sup id="fnref2:Perlis"><a class="footnote-ref" href="#fn:Perlis">1</a></sup></cite></p>
</blockquote>
<p>Что такое Тьюринговская трясина? Это состояние, в котором программа становится столь могущественной, столь обобщенной, что усилия по решению с её помощью какой-либо конкретной задачи равны или превосходят затраты на написание с нуля программы, которая решает только данную задачу.</p>
<p>Я частенько страдаю от желания пойти поплавать в этой трясине. Начинается всё с необходимости решить какую-то конкретную проблему. Я смотрю на существующие инструменты и вижу в каждом из них свои недостатки. И вот, забыв о первоначальной задаче, я уже набрасываю архитектуру универсального инструмента, который будет решать общий класс подобных задач, будет невероятно гибким, универсальным и… </p>
<p>(Особенно опасна эта трясина для разработчиков языков программирования. Есть огромный соблазн урезать язык до самого маленького, самого элегантного ядра аксиом.)</p>
<blockquote>
<p>Я думаю, язык программирования должен быть большей частью самоопределяем. Спецификация языка должна быть разделена на 2 части: небольшое ядро операторов (которые выполняют роль аксиом) и остальная часть языка, которая, как теоремы, определяется в терминах аксиом.
-- <cite>Пол Грэм</cite></p>
</blockquote>
<p>Опасность трясины в том, что вместо решения проблемы вы начинаете разрабатывать инструмент для создания решений проблем. И чем шире круг задач, в которых он пытается быть полезным, тем менее полезным он окажется в каждой конкретной задаче. </p>
<p>В моем случае, я часто ловлю себя на том, что при "программировании снизу вверх", пытаюсь изобрести идеальный язык программирования для реализации программы, которую я пытаюсь написать. Когда я вдруг обнаруживаю себя работающим над спецификацией нового языка программирования, я понимаю, что попал в трясину. Пора остановиться и передохнуть.</p>
<p>Привлекательность Тьюринговской трясины в том, что она кажется потенциальным решением всех проблем. Инструмент, который решает любые задачи общими методами — что может быть лучше? Например, VisiCalc<sup id="fnref:vc"><a class="footnote-ref" href="#fn:vc">2</a></sup>: это приложение действительно превратило своих пользователей в программистов!</p>
<blockquote>
<p>В программировании всё, что мы делаем, является всего-лишь частным случаем более общей проблемы. И иногда мы осознаём это слишком быстро.
-- <cite>Алан Перлис <sup id="fnref:Perlis"><a class="footnote-ref" href="#fn:Perlis">1</a></sup></cite></p>
</blockquote>
<p>Попытка понять отношение вашей конкретной проблемы и общего класса подобных проблем может действительно помочь увидеть некоторые скрытые ранее детали, и найти ряд замечательных моментов «Ага!».</p>
<p>Таким образом, я верю, что в идеальном случае вы должны быть всегда готовы пройтись по краю Тьюринговской трясины, но в то же время иметь железную силу воли, чтобы не свалиться в неё. </p>
<p>Лично у меня подобной силы воли нет, и, что еще хуже, есть огромное любопытство. Для хоть какой-то борьбы с этим всем я придумал ряд эвристик по определению края трясины:</p>
<ol>
<li><strong>Помните мысль Алана Кэя</strong>: <em>«Мы стараемся сделать простые вещи простыми, а сложные — возможными»</em>. Когда решение общей задачи делает сложные вещи возможными — это хорошо. Но не ценой потери простоты простых вещей.</li>
<li><strong>Сколь бы высоко вы не летали в облаках теории, время от времени спускайтесь на грешную землю</strong>. Лично я всегда имею в наличии определенное количество пожеланий пользователей и регулярно возвращаюсь к ним с вопросом «а поможет ли этот восхитительный виджет решить вот эту банальную практическую проблему?».</li>
<li><strong>Eat your own dog food/go round trip on an infrequent basis</strong>. Okay, you’re convinced you need to run your application inside a special purpose virtual machine. Fine. Pick one small operation and make it work. Program for your virtual machine, end to end. Don’t skip any of the steps. It’s better to go thin (few functions) and deep (use the entire stack) than to go wide and shallow. If your VM is a pain in the rump, you’ve discovered something. (This is a cornerstone of Ken Schwaber’s Scrum methodology).</li>
<li><strong>Не занимайтесь самобичеванием, когда вдруг обнаружите, что всё-таки погрязли в трясине</strong>. Даже если Вы начали решать проблему А и вдруг обнаружили себя решающим общий класс задач С, который даже не включает А, снова вспомните Алана Кэя: <em>«Успешная технология создаёт проблемы, которые только она и может решить»</em>. Не будьте сразу уверенными в том, что создали что-то ненужное. Если осталось немного — закончите и посмотрите, что вышло. Вы никогда не узнаете, полезная ли получилась штука, пока не начнёте пользоваться.</li>
</ol>
<blockquote>
<p>Уверены ли вы, что все эти свистелки и дуделки, все эти восхитительные средства, составляющие ваш «мощный язык программирования», на самом деле являются частью решения проблемы, а не частью самой проблемы?
-- <cite>Эдсгер Дейкстра</cite></p>
</blockquote>
<p>Аминь.</p>
<h3>Комментарий переводчика</h3>
<p>Тьюринговская трясина на самом деле встречается чаще, чем кажется. И ладно бы в неё затягивало лишь великие академические умы, решающие задачи вселенского масштаба. Но нет — универсальные инструменты стараются создавать все! Каждая программа по записи DVD зачем-то обзаводится своими аудио\видео\фото-редакторами, средствами создания обложек, плеерами и конверторами. Каждый мессенджер считаем своим долгом поддерживать все возможные протоколы связи, вплоть до электронной почты, социальных сетей, СМС, бумажных писем и отправки сообщений внеземным цивилизациям. Все операционки лезут на все типы устройств, каждый кодек-пак включает в себя тонну хлама, документация на некоторые консольные программы <sup id="fnref:mplayer"><a class="footnote-ref" href="#fn:mplayer">3</a></sup> имеет по 150 страниц формата А4 и пару тысяч ключей командной строки. Каждый второй сайт в интернете обрастает мхом из погоды\гороскопов\знакомств\работы\чатов\форумов. Стараясь привлечь лишнего пользователя новой фичей, программы и сайты теряют десяток других, которые заколебались выискивать в образовавшейся куче хлама то рациональное зерно, ради которого когда-то эта программа или сайт были выбраны.</p>
<p>Серебряных пуль нет. Лично мне единственным способом не скатится из полезного узконаправленного средства в разрозненный набор малофункциональной чуши кажется система плагинов. Хорошими примерами являются современные браузеры, некоторые онлайн-игры, IDE и другие модульные программы (я уверен — вы сможете дополнить список), которые, оставаясь весьма аскетичными в своей основе, дают тем не менее возможность сотворить боевой комбайн любого уровня сложности.
Перед тем как добавить в свою программу новую фичу — подумайте, а нужна ли она хотя бы четверти пользователей? Если нет — может быть стоит просто вывести наружу API и дать возможность желающим написать и подключить плагин? </p>
<div class="footnote">
<hr />
<ol>
<li id="fn:Perlis">
<p><span>Алан Джей Перлис (англ. Alan Jay Perlis; 1 апреля 1922 — 7 февраля 1990) — американский учёный в области компьютерных технологий, известен своими работами в области языков программирования и как первый лауреат премии Тьюринга (в 1966 году, согласно цитате, «за его влияние в области передовых методов программирования и создание компиляторов». Имеется в виду его работа в группе, разработавшей язык программирования Алгол).</span> <a class="footnote-backref" href="#fnref:Perlis" title="Jump back to footnote 1 in the text">↩</a><a class="footnote-backref" href="#fnref2:Perlis" title="Jump back to footnote 1 in the text">↩</a></p>
</li>
<li id="fn:vc">
<p><strong>VisiCalc</strong> -- первая программа для редактирования таблиц (а-ля <em>Excel</em>). <a href="http://www.bricklin.com/visicalc.htm">http://www.bricklin.com/visicalc.htm</a> <a class="footnote-backref" href="#fnref:vc" title="Jump back to footnote 2 in the text">↩</a></p>
</li>
<li id="fn:mplayer">
<p>Например, <a href="http://www.mplayerhq.hu/DOCS/man/en/mplayer.1.html">документация</a> <em>Mplayer</em> <a class="footnote-backref" href="#fnref:mplayer" title="Jump back to footnote 3 in the text">↩</a></p>
</li>
</ol>
</div>
</div>
</article>
<footer class="content">
<span>© 2016
Николай Волков</span>
<span class="separator">::</span>
<a href="https://github.com/FRiMN">GitHub</a>
<span class="separator">::</span>
<a href="https://bitbucket.org/FRiMN/">Bitbucket</a>
</footer>
<script type="text/javascript">
var $headers = $("body > article > .text h2, body > article > .text h3");
var $headersH3 = $("body > article > .text h3");
var $mainHeader = $("body > article > header");
// Подготовка списка содержания
var id = 0;
var toc_list = []
$headers.each(function (index) {
id = index+1;
$this = $(this);
$(this).attr('id', id);
var level = 2;
_.each($headersH3, function($item) {
if ( $this[0].id === $item.id ) {
level = 3;
}
})
// $(this).data('level', level);
toc_list.push([id, $(this).text(), level]);
})
// Генерация боковой панели
var tocT = _.template('\
<div class="toc">\
<h4>Содержание</h4>\
<% var prevlevel = 2; %>\
<ol> <% _.each(list, function(item) { %>\
<% if (item[2] === 3) { %>\
<% if (prevlevel === 2) { %>\
<ol>\
<% prevlevel = 3; } %>\
<li class="level-3"><a href="#<%= item[0] %>"><%= item[1] %></a></li>\
<% } else { %>\
<% if (prevlevel === 3) { %>\
</ol>\
<% prevlevel = 2; } %>\
<li><a href="#<%= item[0] %>"><%= item[1] %></a></li>\
<% } %>\
<% }) %> </ol>\
<a class="up" href="#">Наверх</a>\
</div>\
');
if (toc_list.length > 1) {
var tocContainer = tocT({list : toc_list});
var $tocContainer = $(tocContainer);
$("body > article").append($tocContainer);
// Позиционирование боковой панели
var reposition = function() {
if ( $(window).scrollTop() < $mainHeader.height() ) {
$tocContainer.css('top', $mainHeader.height() - $(window).scrollTop());
$tocContainer.css('max-height',
$(window).height() - $mainHeader.height() + $(window).scrollTop() - 16);
} else {
$tocContainer.css('top', '');
$tocContainer.css('max-height', '98%');
}
}
reposition();
$(window).scroll(reposition);
}
// Установка анкоров на заголовки
$headers.each(function(index) {
$(this).prepend('<span class="anchor-link"><a href="#'+$(this).attr('id')+'">#</a></span>')
});
// Копирование сносок к тексту
var footnotesList = {};
var $footnotes = $("div.footnote li > p");
$footnotes.each(function(index) {
footnotesList[ $(this).find('a.footnote-backref').attr('href') ] = $(this);
});
var paragraphs = {}
$('a.footnote-ref').each(function(index) {
var id;
var fnLink = $(this).parent('sup').attr('id');
var $container = $(this).parents('p');
if ( $container.children('.side-footnotes').length === 0 ) {
id = _.uniqueId("sfn-");
$container.append('<div id="'+id+'" class="side-footnotes">');
} else {
id = $container.children('.side-footnotes').attr('id');
}
paragraphs[id] = paragraphs[id] || [];
paragraphs[id].push(fnLink);
paragraphs[id] = _.uniq(paragraphs[id]);
})
_(paragraphs).each(function(value, key, plist) {
var $container = $('#'+key);
var $p = $container.parent();
var fnnums = [];
$p.find('a.footnote-ref').each(function() {
fnnums.push( $(this).parent('sup') );
}).hover(function() {
$container.addClass('hover');
}, function() {
$container.removeClass('hover');
});;
_(value).each(function(element, index, fnlist) {
var fnnum = _(fnnums).find(function(item) {
return item.attr('id') === element;
});
var E = $( footnotesList[ '#'+element ].clone() );
E.prepend('<span class="fnnum">'+fnnum.children('a.footnote-ref').text()+'</span>');
E.find('a.footnote-backref').remove();
$container.append( E );
fnnum.hover(function() {
E.find('a').addClass('hover');
}, function() {
E.find('a').removeClass('hover');
});
})
})
</script>
<!-- Yandex.Metrika counter -->
<script type="text/javascript">
(function (d, w, c) {
(w[c] = w[c] || []).push(function() {
try {
w.yaCounter36832450 = new Ya.Metrika({
id:36832450,
clickmap:true,
trackLinks:true,
accurateTrackBounce:true,
trackHash:true
});
} catch(e) { }
});
var n = d.getElementsByTagName("script")[0],
s = d.createElement("script"),
f = function () { n.parentNode.insertBefore(s, n); };
s.type = "text/javascript";
s.async = true;
s.src = "https://mc.yandex.ru/metrika/watch.js";
if (w.opera == "[object Opera]") {
d.addEventListener("DOMContentLoaded", f, false);
} else { f(); }
})(document, window, "yandex_metrika_callbacks");
</script>
<noscript><div><img src="https://mc.yandex.ru/watch/36832450" style="position:absolute; left:-9999px;" alt="" /></div></noscript>
<!-- /Yandex.Metrika counter -->
</body>
</html>