AXForum  
Zurück   AXForum > Microsoft Dynamics AX > DAX: Программирование
All
Kennwort vergessen?
Registrieren Forum Rules Hilfe Benutzerliste Heutige Beiträge Suchen

 
 
Themen-Optionen Thema durchsuchen Ansicht
Alt 19.12.2017, 11:14   #1  
SuperStar88 ist offline
SuperStar88
Участник
 
82 / 10 (1) +
Registriert seit: 11.08.2017
? Помогите с кодом (ax 2009)
Как правильно написать код?
Надо добавлять данные в таблицу по определённому условию:
- пока "Разница дат" в строке >= "Периоду", то добавляем новую строку со значением "Дата С" = "Дата С + Период"
- иначе переходим на следующую строку

Пока вышло что-то страшное
X++:
//rowQuota - временная таблица

    while select forupdate rowQuota
    {
        monthDiff = intvNo(rowQuota.DateTo, rowQuota.DateFrom, IntvScale::Month);
        dayDiff = intvNo(rowQuota.DateTo, rowQuota.DateFrom, IntvScale::Day);

        if(rowQuota.UsagePeriod <= monthDiff && dayDiff > 0)
        {
            buf2buf(rowQuota, rowQuotaTmp);

            do
            {
                rowQuotaTmp.DateFrom = dateMthFwd(rowQuotaTmp.DateFrom, rowQuotaTmp.UsagePeriod);
                rowQuotaTmp.DatePotreb = rowQuotaTmp.DateFrom;
                rowQuotaTmp.insert();

                monthDiff = intvNo(rowQuotaTmp.DateTo, rowQuotaTmp.DateFrom, IntvScale::Month);
                dayDiff = intvNo(rowQuotaTmp.DateTo, rowQuotaTmp.DateFrom, IntvScale::Day);
            }
            while(rowQuotaTmp.UsagePeriod <= monthDiff && dayDiff > 0);
        }
    }
Alt 19.12.2017, 11:38   #2  
SuperStar88 ist offline
SuperStar88
Участник
 
82 / 10 (1) +
Registriert seit: 11.08.2017
Переписал так, без промежуточного буфера, всё равно долго
X++:
ttsbegin;
    while select forupdate rowQuota
    {
        monthDiff = intvNo(rowQuota.DateTo, rowQuota.DateFrom, IntvScale::Month);
        dayDiff = intvNo(rowQuota.DateTo, rowQuota.DateFrom, IntvScale::Day);

        if(rowQuota.UsagePeriod <= monthDiff && dayDiff > 0)
        {
            do
            {
                rowQuota.DateFrom = dateMthFwd(rowQuota.DateFrom, rowQuota.UsagePeriod);
                rowQuota.DatePotreb = rowQuota.DateFrom;
                rowQuota.insert();

                monthDiff = intvNo(rowQuota.DateTo, rowQuota.DateFrom, IntvScale::Month);
                dayDiff = intvNo(rowQuota.DateTo, rowQuota.DateFrom, IntvScale::Day);
            }
            while(rowQuota.UsagePeriod <= monthDiff && dayDiff > 0);
        }
    }
ttscommit;
Alt 19.12.2017, 14:56   #3  
dech ist offline
dech
Участник
Benutzerbild von dech
Самостоятельные клиенты AX
 
650 / 352 (13) ++++++
Registriert seit: 25.06.2009
Ort: Омск
Blog-Einträge: 3
Во-первых, if и do-while содержат одинаковые условия, превратите это в один цикл while.
Во-вторых, зачем пишете forupdate, когда не применяете ни udpate(), ни delete()?
В-третьих, разделите rowQuota на две разные переменные для выборки и вставки, тогда понятнее будет код.
В-четвертых, можно заменить monthDiff и dayDiff вызовом функций и сделать один вызов вместо двух.
__________________
// no comments
Alt 20.12.2017, 08:29   #4  
terraByteG ist offline
terraByteG
Участник
 
15 / 17 (1) ++
Registriert seit: 10.03.2016
Zitat:
Zitat von SuperStar88 Beitrag anzeigen
Переписал так, без промежуточного буфера, всё равно долго
Для начала, объясните, чего именно вы хотите добиться?
Там точно должен быть insert, а не update.

ПС: Долго? Так цикл в цикле.
Одна переменная для селекта и инсерта - жестоко...
Alt 20.12.2017, 09:16   #5  
belugin ist offline
belugin
Участник
Benutzerbild von belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4.622 / 2925 (107) +++++++++
Registriert seit: 16.01.2004
Blog-Einträge: 5
Нельзя оптимизировать, не зная статистики. Вот этот rowQuota - это большая таблица или маленькая? Что делает intvNo? Нельзя ли сделать, чтобы выбор записей был по индексу, если таблица большая. Нельзя ли использовать insert_recordset
This post has been rated by: MikeR (2).
Alt 20.12.2017, 10:50   #6  
SuperStar88 ist offline
SuperStar88
Участник
 
82 / 10 (1) +
Registriert seit: 11.08.2017
Есть временная таблица rowQuota с ДатаС, ДатаПо и Периодом. Если Период меньше, чем кол-во месяцев между датами, то создаём новую строку, где ДатаС = ДатаС+месяц(Период)

Таблица не очень большая, около 100к записей. Да, с индексом в 2 раза быстрее получилось. И решил использовать RecordInsertList
X++:
    ttsbegin;
    list = new RecordInsertList(rowQuotaTmp.TableId);
    while select rowQuota
    index hint Idx
    {
        buf2buf(rowQuota, rowQuotaTmp);

        monthDiff = intvNo(rowQuotaTmp.DateTo, rowQuotaTmp.DateFrom, IntvScale::Month);
        dayDiff = intvNo(rowQuotaTmp.DateTo, rowQuotaTmp.DateFrom, IntvScale::Day);

        while(rowQuotaTmp.UsagePeriod <= monthDiff && dayDiff > 0)
        {
            rowQuotaTmp.DateFrom = dateMthFwd(rowQuotaTmp.DateFrom, rowQuotaTmp.UsagePeriod);
            rowQuotaTmp.DatePotreb = rowQuotaTmp.DateFrom;
            //rowQuota.insert();
            list.add(rowQuotaTmp);

            monthDiff = intvNo(rowQuotaTmp.DateTo, rowQuotaTmp.DateFrom, IntvScale::Month);
            dayDiff = intvNo(rowQuotaTmp.DateTo, rowQuotaTmp.DateFrom, IntvScale::Day);
        }
    }
    list.insertDatabase();
    ttscommit;
Miniaturansicht angehängter Grafiken
Klicken Sie auf die Grafik für eine größere Ansicht

Name:	1.png
Hits:	415
Größe:	10,0 KB
ID:	11793  
Alt 20.12.2017, 10:53   #7  
Pustik ist offline
Pustik
Участник
 
807 / 372 (14) ++++++
Registriert seit: 04.06.2004
Автору поста. Вот джоб, который уходит в бесконечный цикл при использовании одной переменной для выборки и вставки
X++:
static void JobCycle(Args _args)
{
    TmpConfigId TmpConfigId;
    int i;
    ;
    TmpConfigId.ItemNum = 0;
    TmpConfigId.insert();

    while select TmpConfigId 
    {
        i++;
        TmpConfigId.ItemNum = i;
        TmpConfigId.insert();
    }
}
__________________
-Ты в гномиков веришь?
-Нет.
-А они в тебя верят, смотри, не подведи их.
Alt 20.12.2017, 12:04   #8  
dmn ist offline
dmn
Участник
 
9 / 18 (1) ++
Registriert seit: 25.09.2014
Zitat:
Zitat von SuperStar88 Beitrag anzeigen
Есть временная таблица rowQuota с ДатаС, ДатаПо и Периодом. Если Период меньше, чем кол-во месяцев между датами, то создаём новую строку, где ДатаС = ДатаС+месяц(Период)
А не проще анализировать записи при insert'е во временную таблицу?
Т.е. при добавлении если ДатаС + Период < ДатаПо, тогда добавляем еще 1 запись, где ДатаС = ДатаС + Период и т.д.
Alt 20.12.2017, 12:40   #9  
belugin ist offline
belugin
Участник
Benutzerbild von belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4.622 / 2925 (107) +++++++++
Registriert seit: 16.01.2004
Blog-Einträge: 5
Либо сохранить признак для условия отбора и проиндекировать его. Чтобы while select мог быстро отобрать только те записи, которые надо обрабатывать.

И прочитайте Best Practice (наименование индексов, например)
Alt 20.12.2017, 18:06   #10  
Alexius ist offline
Alexius
Участник
Benutzerbild von Alexius
 
461 / 248 (9) ++++++
Registriert seit: 13.12.2001
Рекомендую почитать старый топик Временные таблицы и скорость работы
Alt 21.12.2017, 18:17   #11  
Владимир Максимов ist offline
Владимир Максимов
Участник
КОРУС Консалтинг
 
1.720 / 1207 (44) ++++++++
Registriert seit: 13.01.2004
Blog-Einträge: 3
Если используется цикл с фиксированным шагом приращения, то имеет смысл подумать об использовании цикла for()

X++:
static void Test(Args _args)
{
    FromDate    fromDate = 01\01\2017;
    ToDate      toDate = 31\07\2017;
    int         period = 3;
    TransDate   transDate;
    ;

   // Исходная запись
    info(strFmt('с %1 по %2', fromDate, toDate));
    for (transDate = global::dateMthFwd(fromDate, period); transDate < toDate; transDate = global::dateMthFwd(transDate, period))
    {
       // новые записи
        info(strFmt('с %1 по %2', transDate, toDate));
    }

    info('end');

}
__________________
- Может, я как-то неправильно живу?!
- Отчего же? Правильно. Только зря...
This post has been rated by: SuperStar88 (1).
Alt 21.12.2017, 19:03   #12  
Raven Melancholic ist offline
Raven Melancholic
Участник
Benutzerbild von Raven Melancholic
Самостоятельные клиенты AX
Лучший по профессии 2015
 
2.164 / 1296 (48) ++++++++
Registriert seit: 21.03.2005
Ort: Москва-Петушки
Zitat:
Zitat von SuperStar88 Beitrag anzeigen
Есть временная таблица rowQuota
...
И решил использовать RecordInsertList
Если таблица rowQuota временная, то использовать RecordInsertList совершенно бессмысленно. Он помогает тогда, когда таблица расположена в базе данных и сокращает количество пакетов на вставку именно в базу данных (при некоторых других условиях).
This post has been rated by: SuperStar88 (1).
 

Ähnliche Themen
Thema Autor Forum Antworten Letzter Beitrag
dynamicsaxse: November 2017 Release – Dynamics AX 2012 R3 Blog bot DAX Blogs 0 08.12.2017 10:51
stephenmann: Technical History of Dynamics AX - From Axapta 3.0 to AX2012 Blog bot DAX Blogs 5 03.03.2017 10:22
DAX: Official Dynamics AX 2012 R2 Content (update) - Where is it, and how can you find out about updates? Blog bot DAX Blogs 0 03.12.2012 11:11
Sample Design Patterns: Upgrade to Microsoft Dynamics AX 2009 and issues with the global address book Blog bot DAX Blogs 0 21.12.2010 11:11

Forumregeln
Es ist Ihnen nicht erlaubt, neue Themen zu verfassen.
Es ist Ihnen nicht erlaubt, auf Beiträge zu antworten.
Es ist Ihnen nicht erlaubt, Anhänge hochzuladen.
Es ist Ihnen nicht erlaubt, Ihre Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Gehe zu

Рейтинг@Mail.ru
Alle Zeitangaben in WEZ +3. Es ist jetzt 13:26 Uhr.
Powered by vBulletin® Version 3.8.5 (Deutsch)
Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.