AXForum  
Вернуться   AXForum > Microsoft Dynamics AX > DAX: Программирование
All
Забыли пароль?
Зарегистрироваться Правила Справка Пользователи Сообщения за день Поиск Все разделы прочитаны

 
 
Опции темы Поиск в этой теме Опции просмотра
Старый 29.07.2014, 12:13   #1  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
В чем смысл подобной конструкции?
Я щас глупость спрошу. Но весь мозг сломал в размышлениях "зачем?"

AX2012 R3.
Класс RetailCDXChannelSpecificData, метод trackChannelDBAssignment()

вот очищенный от лишнего код:
X++:
public void trackChannelDBAssignment()
{
    container channelDB;
    int i;
    RefRecId channelDBRecId;

    while select RefRecId
        from ref1
        where // условия
    {
        channelDB += ref1.RefRecID;
    }

    for (i = 1; i <= conLen(channelDB); ++i)
    {
        channelDBRecId = conPeek(channelDB, i);

        // некие действия
    }
}
ВОПРОС:
нафига разработчики Майкрософт сначала собирают контейнер, а затем делают цикл по контейнеру?

контейнер - внутренняя переменная, никуда не возвращается, никаких побочных эффектов он не дает. никаких действий со счетчиком не выполняется. счетчик не возвращается. break'ов из циклов тоже не бывает.

ЗАЧЕМ?
Почему не сразу один цикл?

X++:
public void trackChannelDBAssignment()
{
    RefRecId channelDBRecId;

    while select RefRecId
        from ref1
        where // условия
    {
        channelDBRecId = ref1.RefRecID;

        // некие действия
    }
}
Старый 29.07.2014, 12:19   #2  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,429 / 1772 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
Может для того чтобы меньше блокировать таблицу ref1. Побыстрее "отпустить". Если операции вставка/удаление/изменение записей в этой таблице происходят часто, а "некие действия" достаточно продолжительны, то вполне возможно - это такая оптимизация.

Последний раз редактировалось S.Kuskov; 29.07.2014 в 12:22.
Старый 29.07.2014, 12:28   #3  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от S.Kuskov Посмотреть сообщение
Может для того чтобы меньше блокировать таблицу ref1. Побыстрее "отпустить". Если операции вставка/удаление/изменение записей в этой таблице происходят часто, а "некие действия" достаточно продолжительны, то вполне возможно - это такая оптимизация.
метод работает глубоко внутри транзакции.
никаких особо длинных операций не выполняется.
вот полный код:

X++:
/// <summary>
/// Tracks channel database assignment.
/// </summary>
public void trackChannelDBAssignment()
{
    container channelDB;
    int i;
    RefRecId channelDBRecId;
    RefRecId channelRecId;
    RetailCDXDataStoreChannel dataStoreChannel;

    while select RefRecId
        from ref1
        where ref1.NodeID == 0
           && ref1.RefTableID == tableNum(RetailCDXDataStoreChannel)
           && ref1.SYS_CHANGE_OPERATION == 'I'
    {
        channelDB += ref1.RefRecID;
    }

    for (i = 1; i <= conLen(channelDB); ++i)
    {
        channelDBRecId = conPeek(channelDB, i);

        select Channel from dataStoreChannel
            where dataStoreChannel.RecId == channelDBRecId;

        channelRecId = dataStoreChannel.Channel;

        select firstOnly RecId from ref1
        where ref1.NodeID == 0
           && ref1.RefTableID == tableNum(RetailChannelTable)
           && ref1.SYS_CHANGE_OPERATION == 'I'
           && ref1.RefRecID == channelRecId;

        if (!ref1)
        {
            ref1.initValue();
            ref1.NodeID = 0;
            ref1.RefTableID = tableNum(RetailChannelTable);
            ref1.RefRecID = channelRecId;
            ref1.SYS_CHANGE_OPERATION = 'I';
            ref1.insert();
        }
    }
}
чувствую, что туплю. и наверняка причина на поверхности.
но даже предположить не могу ЗАЧЕМ?
Старый 29.07.2014, 12:29   #4  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,429 / 1772 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
Ну так они же во втором цикле вносят изменения в данные влияющие на выборку в первом.

UPD.: Хотя нет, показалось. Там RefTableID другой. Ну может быть им так же как и мне показалось

Последний раз редактировалось S.Kuskov; 29.07.2014 в 12:34.
Старый 29.07.2014, 12:33   #5  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от S.Kuskov Посмотреть сообщение
Ну так они же во втором цикле вносят изменения в данные влияющие на выборку в первом.

не совсем так, в условии выборки присуствует tablenum одной таблицы, а вставляют они tablenum другой таблицы. поэтому выборку они не меняют.

если бы это был программист на каком-нибудь проекте, я бы подумал, что он не умеет объявлять еще одну локальную табличную переменную вместо повторного использования ref1... подумал бы... и уволил программиста нафиг.

но здесь вроде майкрософтовский код... и может быть, я чего не знаю?
Старый 29.07.2014, 12:36   #6  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
покопал на предмет использования побочных эффектов с ref1.
ref1 определен как переменная класса.

но не, ref1 после вызова не используется. побочных эффектов нет.

ЗАЧЕМ?
Старый 29.07.2014, 12:36   #7  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,429 / 1772 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
А метод ref1.insert() не перекрыт? Там ничего существенного не происходит?
Старый 29.07.2014, 12:41   #8  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от S.Kuskov Посмотреть сообщение
А метод ref1.insert() не перекрыт? Там ничего существенного не происходит?
нет, метод не перекрыт.

эта таблица временная. спасибо за направление. сюда несомненно стоит подумать.

но ведь ничто не мешает объявить локальную переменную и направить ее на тот же набор временных записей, что и переменная класса ref1. или таки нет?
Миниатюры
Нажмите на изображение для увеличения
Название: 1.PNG
Просмотров: 448
Размер:	59.4 Кб
ID:	8936  
Старый 29.07.2014, 12:42   #9  
online
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,892 / 5650 (194) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Цитата:
Сообщение от mazzy Посмотреть сообщение

не совсем так, в условии выборки присуствует tablenum одной таблицы, а вставляют они tablenum другой таблицы. поэтому выборку они не меняют.

если бы это был программист на каком-нибудь проекте, я бы подумал, что он не умеет объявлять еще одну локальную табличную переменную вместо повторного использования ref1... подумал бы... и уволил программиста нафиг.

но здесь вроде майкрософтовский код... и может быть, я чего не знаю?
А ведь, насколько я помню, Microsoft ритейловскую функциональность у кого-то прикупил по случаю кризиса - недорого и со скидкой. Так что вполне возможно что код унаследован от партнера, который его написал еще во времена v 2.50
Конечно там insert_recordset напрашивается...
Старый 29.07.2014, 12:45   #10  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от fed Посмотреть сообщение
А ведь, насколько я помню, Microsoft ритейловскую функциональность у кого-то прикупил по случаю кризиса - недорого и со скидкой. Так что вполне возможно что код унаследован от партнера, который его написал еще во времена v 2.50
Конечно там insert_recordset напрашивается...
не, не, не.
это код из R3. это код, который наваяли уже в Майкрософте весной этого года.

унаследованный был в r2 раньше.
в унаследованном порядка и логичности было больше. Хотя и там были места для рефакторинга. Но хоть какое-то логичное объяснение тогдашнему коду было.
Старый 29.07.2014, 12:51   #11  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,429 / 1772 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
Цитата:
Сообщение от mazzy Посмотреть сообщение
эта таблица временная. спасибо за направление. сюда несомненно стоит подумать.

но ведь ничто не мешает объявить локальную переменную и направить ее на тот же набор временных записей, что и переменная класса ref1. или таки нет?
Таки да, очень похоже на то что они не знали о такой возможности SetTmpData. Других объяснений у меня больше нет

Последний раз редактировалось S.Kuskov; 29.07.2014 в 12:53.
Старый 29.07.2014, 12:51   #12  
ice is offline
ice
Участник
Аватар для ice
Лучший по профессии 2014
 
1,689 / 405 (17) +++++++
Регистрация: 23.03.2006
Такую конструкцию не только заморские программисты пишут, часто встречал при кодревью, но на вопрос "зачем?" так никто и не ответил, просто переделывали под второй вариант. видимо зашито где-то в подсознании
Старый 29.07.2014, 12:51   #13  
Logger is offline
Logger
Участник
Лучший по профессии 2015
Лучший по профессии 2014
 
3,873 / 3123 (112) ++++++++++
Регистрация: 12.10.2004
Адрес: Москва
Записей в блоге: 2
Делал такое иногда в оракле для длительных процедур.
Например делаем перебор номенклатурного справочника, а для каждой номенклатуры выполняется какая-о длительная операция.

Если перебор номенлктаур делать в цикле как предлагал Маззи, то может вылезти ошибка "Snapshot is too old" или что-то в этом духе. В общем сегмент повтора пришлось бы увеличить чтобы этого не было.

Для обхода такой особенности, обычно быстро начитывали перечень интересующих номенклатур в set, освобождали курсор и спокойно работали дальше начитывая номенклатуры из Set-а (контейнер - это кривизна !)

Возможно для SQL сервера тоже бывают аналогичные проблемы. Версионность забесплатно не дается. На неё БД все равно тратит ресурсы.
Старый 29.07.2014, 12:52   #14  
online
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,892 / 5650 (194) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Цитата:
Сообщение от mazzy Посмотреть сообщение
не, не, не.
это код из R3. это код, который наваяли уже в Майкрософте весной этого года.

унаследованный был в r2 раньше.
в унаследованном порядка и логичности было больше. Хотя и там были места для рефакторинга. Но хоть какое-то логичное объяснение тогдашнему коду было.
А что ты поповоду использования контейнера думаешь ? Он же immutable - каждый раз память захватывается и освобождается. По моему - правильнее было бы set использовать. (Даже если есть какой-то смысл в подходе в целом). Ну то есть - на мой взгляд - вероятность просто кривого программирования намного выше, чем вероятность хитрого подхода...
За это сообщение автора поблагодарили: mazzy (2), S.Kuskov (1).
Старый 29.07.2014, 12:57   #15  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от S.Kuskov Посмотреть сообщение
Таки да, очень похоже на то что они не знали о такой возможности
грустно, девушки...

Цитата:
Сообщение от Logger Посмотреть сообщение
Делал такое иногда в оракле для длительных процедур.
ключевое слово - "длительных".
для длительных я согласен, наверное. хотя все равно лучше объявить вторую переменную по методу из книжки Еременко.

но в данном то случае никаких "длительных" нет.
и не предполагается. вся байда в пределах одного метода.

==================
смотрю я на код retail'а в R3, хочется шипя материться.
такого накала идиотии давно не видел.
все таки остается надежда, что я чего то не понимаю.
Старый 29.07.2014, 13:04   #16  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от fed Посмотреть сообщение
А что ты поповоду использования контейнера думаешь ? Он же immutable - каждый раз память захватывается и освобождается.
В X++ контейнер плохо.
Однако на .net короткоживущие переменные - очень даже хорошо.

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

предполагается, что данный код будет исполняться в пакетном режиме и в CIL.
поэтому бог с ними, с контейнерами.

мне непонятно зачем надо было городить ТАКОЙ огород с двумя циклами.
при условии, что нет никаких длительных операций.

Цитата:
Виктор поразился, какое лицо сделалось у Дианы. Как на картине. Даже не на картине – на иконе. Странная неподвижность черт, и ты недоумеваешь, то ли это замысел мастера, то ли бессилие ремесленника.
(С) Гадкие лебеди. Стругацкие.
 

Похожие темы
Тема Автор Раздел Ответов Посл. сообщение
Какой смысл InventDim на строках заказа? IKA DAX: Функционал 7 08.04.2013 19:04
DAX 2009 RU5 - Ошибка компилятора в конструкции select sukhanchik DAX: Программирование 9 11.01.2013 08:43
Смысл проекта AxUpgradeRecIdConflicts trud DAX: Администрирование 1 23.08.2012 11:51
Объясните смысл проводки Eldar9x DAX: Функционал 8 24.11.2009 08:32
Деловое отношение – родитель и Разовый клиент. В чем их смысл? ShyGuy DAX: Прочие вопросы 4 10.03.2009 10:28
Опции темы Поиск в этой теме
Поиск в этой теме:

Расширенный поиск
Опции просмотра

Ваши права в разделе
Вы не можете создавать новые темы
Вы не можете отвечать в темах
Вы не можете прикреплять вложения
Вы не можете редактировать свои сообщения

BB коды Вкл.
Смайлы Вкл.
[IMG] код Вкл.
HTML код Выкл.
Быстрый переход

Рейтинг@Mail.ru
Часовой пояс GMT +3, время: 18:40.
Powered by vBulletin® v3.8.5. Перевод: zCarot
Контактная информация, Реклама.