Вверх ↑
Ответов: 9906
Рейтинг: 351
#1: 2007-07-16 15:55:48 ЛС | профиль | цитата
Но все-таки не все ладно в нашем королевстве, мне продолжает казаться...
Специально повторюсь: это все я не специально придумываю, мысли сами в голову лезут


Вроде бы вышеупомянутого перечня зависимости точек для элемента (не важно, каким образом добытого: парсингом, или из конфига) - не достаточно.
В смысле: понятно, что любая информация - это лучше, чем ее отсутствие. Но лучше иметь максимально полную информацию.
Не задаваясь вопросом, нужна ли она именно сегодня. Принимая за аксиому, что все равно понадобится (хотя, возможно - и не сразу). Главное - определить объем этой информации, и сформировать механизмы доступа к ней.
Мне (сегодня) такая информация об элементе представляется примерно в таком виде. Для каждого метода элемента указываем не только вызываемые события, но и их последовательность
doStrCat
  <Str1>,<Str2>,onStrCat
Более того, можем указать альтернативные варианты последовательностей
doCompare: <Op1>Type<Op2>
  <Op1>,<Op2>,onTrue
doCompare: not <Op1>Type<Op2>
<Op1>,<Op2>,onFalse
Смысл необязательных условий на "варианте" имеется только тогда, когда они могут быть вычислены в Design-Time
Естественно.
Но когда это происходит - это оптимизация типа Dead End Elimination.
Ну, к примеру, если никогда не происходящий onFalse меняет какие-то данные, то сия "подключенность" уже никак не добавит точку doCompare в список "достижимых" при определении обсуждаемого ранее качества Volatile

А разбиение на "вариантность", позволит установить в таком примере
Add(Hub,15612280,182,140)
{
link(onEvent1,9960986:doClear,[(294,146)(294,111)])
link(onEvent2,1693346:doCompare,[])
}
Add(If_else,1693346,238,147)
{
Op2=String(!!!)
link(onTrue,9960986:doValue,[(286,153)(286,104)])
link(onFalse,6964309:doCompare,[])
link(Op1,9158563:Text,[])
}
Add(Memory,9960986,301,98)
{
Default=Integer(0)
}
Add(Edit,9158563,238,98)
{
Left=230
Top=70
}
Add(If_else,6964309,301,154)
{
Op2=Integer(0)
link(Op1,9960986:Value,[])
}
"недостижимость" точки Memory.doValue при вычислении Op1 во втором If_Else И в нем, кстати, условие и посчитается в Design-Time
Если, конечно, мы умеем устанавливать вышеупомянутую "недостижимость"


Это скажем информация о классе элемента. И она должна быть превращена CG в информацию для КОНКРЕТНОГО элемента.
Парсингом, или еще как-то, но в этих "строках" должна появиться информация об использовании входных данных, и именно в нужном месте. Ну например
doCompare:
  Op1,D0,onTrue(data)
doCompare:
Op1,D0,onFalse(data)
- здесь имеется в виду, что: а) условие не разрешимо в design-time б) первый операнд идет с верхней точки в) второй из потока (первые данные из MT-списка) г) на выход попадают данные потока. Если факт одного использования установлен (второй операнд), и точно известно, что он ПОСЛЕ события Op1, и ДО событий onTrue или onFalse, то факт использования в выходном событии - только на последующих проходах (кстати, кроме самого факта использования, лучше бы знать и к какому типу данные приводились при использовании).
А ведь это влияет на принятие решения про "буферизацию"
Если Op1 не "портит" входной код, то буферизацию все равно надо провести, при двойном (или больше) использовании входного кода...
Т.е., при этой реализации, принятие решения зависит от фактора "использования" выходным потоком

Тут ситуация такая: точно сказать, как надо получать это "использование" даже "внутри" конкретного элемента (да еще на каждую ветку реализации) - пока не могу
Пока могу совершенно точно сказать - это очень надо
Подозреваю лишь, что факты "использования" почти синхронны "авто приведению типа"...

И вроде бы, в обсуждаемый "вариантный" способ вписываются почти все наши сегодняшние знания о функционировании элемента. Хоть хелп такой канонизируй - именно таких знаний не хватает сегодня пользователю для понимания "чего куды попадет" на схеме


А если еще подумать, когда нам нужна "буферизация", то начинает доходить, что ровно тот же разговор применим и к верхним событиям
Т.е., код, полученный сверху, может прямо в коде элемента использоваться неоднократно
Ну, скажем в неком гипотетическом скрипте написано:
  fvar(s1,s2)
  s1 = Str1
s2 = Str2
println(s1,s2,s1)
Где и кто должен писать код буферизации?
Мне представляется, что именно CG при обработке точки Str1 по информации MultiUsed от предыдущих проходов.
Так же как и для обработки event - мне здесь тоже так кажется.


И все-таки информации, заложенной в SubType - явно не хватает. Ну хорошо, пусть мы установили факт MultiUsed и провели "буферизацию". Все равно в данных у нас код, но того, что дальше заниматься буферизацией уже не умно - у нас ведь никакой...
Add(Hub,15612280,182,140)
{
link(onEvent1,4880539:doWork2,[])
link(onEvent2,1693346:doCompare,[])
}
Add(If_else,1693346,238,147)
{
link(onTrue,4880539:doWork3,[(291,153)])
link(Op1,9158563:Text,[])
}
Add(Memory,9960986,308,140)
{
Default=Integer(0)
}
Add(Edit,9158563,238,98)
{
Left=230
Top=70
}
Add(HubEx,4880539,287,133)
{
link(onEvent,9960986:doValue,[])
}
Add(StrCat,5994167,126,140)
{
Str2="222"
link(onStrCat,15612280:doEvent1,[])
}
Установили мы, что для HUB есть факт MultiUsed
Отлично, проводим "буферизацию"
Но для If_else тоже есть факт MultiUsed
А ведь причин для "буферизации" - уже никаких
Вопрос ведь...
карма: 9

0