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

 
 
Опции темы Поиск в этой теме Опции просмотра
Старый 23.05.2012, 10:23   #1  
fed is offline
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,907 / 5717 (196) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
if (a == true)
Всегда когда вижу в коде (на любом языке), попытки сравнения логических переменных, у меня возникают сильные сомнения в минимальном профессионализме человека, который писал код. (Неужели автор кода смог научится программировать, но ни разу не прочитал пятистраничную статейку про логические операции ?)
Сравнивать логическую переменную с константой - бессмысленно.
Сравнивать две логических переменных можно, но стилистически не красиво. То есть - если возникло желание сравнить две логических переменных, скорее всего есть некая ошибка в подходе к программированию, поскольку при правильном построении алгоритма, эти две переменные надо было бы как-то по другому обрабатывать (типа if (a&&b) или if(a&&!b).

Тем не менее, каждая новая версия Аксапты содержит все больше и больше попыток сравнения переменной с логической константой. Например в 2009ой поиск в AOD по строке "== true" дает 461 совпадение. А в 2012ой - уже 700.
Причем, что самое любопытное, такие сравнения случаются не в каких-то второстепенных отчетах или формах, которые могут на откуп аутсорсерам и контрактерам отдавать. Такие сравнения можно увидеть, например, в классах ReqCalc, inventAdjustPost, LedgerJournalCheckPost. Я уверен что эти ключевые классы пишутся полноценными штатными разработчиками, причем не в Штатах, а в MDCC.

Я вот много раз слышал, что мол при наборе разработчиков в Микрософт, проверяют не знание аксапты, а типа общую программистскую культуру и знание технологий разработки. Я не верю, что человек смог прочитать книжки про OOD и Agile Development, но так и не научился оператор if нормально писать. Возникает подозрение, что авторы этого кода не только Аксапту не знают, но и вообще программировать не умеют...

Вот я и пытаюсь понять - это типа теперь такой best practice ?

Последний раз редактировалось fed; 23.05.2012 в 11:01.
За это сообщение автора поблагодарили: macklakov (1), b_nosoff (1), lev (2), S.Kuskov (2), makbeth (0), kornix (2).
Старый 23.05.2012, 10:42   #2  
macklakov is offline
macklakov
NavAx
Аватар для macklakov
 
2,236 / 974 (37) +++++++
Регистрация: 03.04.2002
Да в 2012 есть масса перлов. Есть новые, а есть и бережно переносимые из версии в версию.
Цитата:
Сообщение от fed Посмотреть сообщение
Неужели автор кода смог научится программировать, но ни разу не прочитал пятистраничную статейку про логические операции ?
Можете кинуть ссылку на статейку? Хочется почитать, для общего развития.
__________________
Isn't it nice when things just work?
Старый 23.05.2012, 10:48   #3  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,438 / 1775 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
Ещё очень умиляют конструкции типа
X++:
a = (< >) ? true : false;
Старый 23.05.2012, 10:52   #4  
fed is offline
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,907 / 5717 (196) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Цитата:
Сообщение от macklakov Посмотреть сообщение
Можете кинуть ссылку на статейку? Хочется почитать, для общего развития.
Ну я кажется такую статейку читал в Науке и Жизни году в 86ом. Но я думаю можно что-то популярное на хабре или в той же википедии найти.
Старый 23.05.2012, 10:57   #5  
oip is offline
oip
Axapta
Лучший по профессии 2014
 
2,564 / 1416 (53) ++++++++
Регистрация: 28.11.2005
Записей в блоге: 1
Если я вижу у кого-то код вида
X++:
boolean bool;
...
if (bool == true)
...
то в таких случаях говорю, что тип выражения "bool == true" - тоже boolean, а значит если следовать такой логике, то надо писать "if ((bool == true) == true)"... и так далее.

PS А в конце, для надежности, можно добавить "!= false".

Последний раз редактировалось oip; 23.05.2012 в 11:02. Причина: Добавил PS
За это сообщение автора поблагодарили: AlGol (1), lev (1), jeky (1), Dark Light (1).
Старый 23.05.2012, 11:03   #6  
macklakov is offline
macklakov
NavAx
Аватар для macklakov
 
2,236 / 974 (37) +++++++
Регистрация: 03.04.2002
Цитата:
Сообщение от fed Посмотреть сообщение
Ну я кажется такую статейку читал в Науке и Жизни году в 86ом. Но я думаю можно что-то популярное на хабре или в той же википедии найти.
А, ну букварь я уже читал.
__________________
Isn't it nice when things just work?
Старый 23.05.2012, 11:20   #7  
driller is offline
driller
Сам.AX
Аватар для driller
Самостоятельные клиенты AX
SAP
 
78 / 54 (2) ++++
Регистрация: 11.04.2007
Адрес: Санк-Петербург
Извиняюсь за оффтоп, немного иная история.
Но как бы вы отреагировали, увидев такой код.
PHP код:
 IF 1 = 2.
    MESSAGE e044(fmup).
 ENDIF
И ничего, повсеместно используемая фича ABAP
__________________
"Считать метафору доказательством, поток праздных слов источником истины, а себя оракулом - это заблуждение, свойственное всем нам."
Поль Валери
Старый 23.05.2012, 11:48   #8  
db is offline
db
Роман Долгополов (RDOL)
Лучший по профессии 2015
Лучший по профессии AXAWARD 2013
 
393 / 692 (24) +++++++
Регистрация: 01.04.2004
Адрес: Москва
Цитата:
Сообщение от S.Kuskov Посмотреть сообщение
Ещё очень умиляют конструкции типа
X++:
a = (< >) ? true : false;
тут сразу вспомнилось вот это обсуждение в похожей теме Проверка на RecId

да, тогда проблема реально была. уже давно всё исправлено, но код никто чистил и он остается примером для подражания для неокрепших мозгов
Старый 23.05.2012, 12:00   #9  
Владимир Максимов is offline
Владимир Максимов
Участник
КОРУС Консалтинг
 
1,692 / 1195 (43) ++++++++
Регистрация: 13.01.2004
Записей в блоге: 3
Честно говоря, не вижу особых проблем. Ну, сравнивают с логической константой и что? Для меня, например, так нагляднее, чем использование восклицательного знака, да еще без пробела. Попробуй еще разгляди его! А так, никакой неоднозначности и не надо "с лупой" искать этот восклицательный знак.

В конструкции вида

a = (<логическое условие>) ? true : false;

тоже не вижу никакого "криминала". Подобную структуру легче модифицировать, если вдруг надо изменить результат на противоположный. Ну, сложно заметить "слившийся" ведущий восклицательный знак. Не видно его!
__________________
- Может, я как-то неправильно живу?!
- Отчего же? Правильно. Только зря...
За это сообщение автора поблагодарили: ivas (2).
Старый 23.05.2012, 12:01   #10  
SRF is offline
SRF
Участник
MCBMSS
Axapta Retail User
 
375 / 562 (19) +++++++
Регистрация: 08.08.2007
Записей в блоге: 1
ну да, для типизированных языков, конструкция сравнения с константами избыточна, но не смертельна, лишь бы логически верный результат был и ожидаемое поведение.

Но к примеру на JavaScript, можно просто завести переменную, не инициализировать ее и фактически будет три значения true, false, null поэтому для упрощения можно использовать сравнение с какой либо константой, отметая два других значения, например, != true.

X++:
var flagTest;
//..
flagTest=$("CheckBoxTest").selected;
//...

if(flagTest !=true) {

}
И еще в C# существует хитрый тип bool? - логический с null(bool + null), так вот такая рода конструкция, в C# просто не откомпилируется, потребует явного приведения типа

X++:
bool? test = null;

if (test)
{ 
            
}
тогда как
X++:
bool? test = null;

if (test == true)
{ 
            
}
не вызовет никаких нареканий, примечательно, что такое неявное преобразование работает насколько я понимаю только для типа bool?, например если во втором случае написать int, то ничего не прокатит
__________________
Sergey Nefedov

Последний раз редактировалось SRF; 23.05.2012 в 12:04.
За это сообщение автора поблагодарили: fed (2).
Старый 23.05.2012, 13:35   #11  
kornix is offline
kornix
MCP
MCBMSS
Злыдни
Ex AND Project
 
414 / 146 (5) +++++
Регистрация: 24.02.2009
Адрес: Санкт-Петербург
В PriceDisc.findPrice():
X++:
return ((findAll     && inventDimAllActivated.RecId     && this.findPriceAgreement(_priceGroupId, inventDimAllActivated.inventDimId))       ||
            (findItemDim && inventDimItemDimActivated.recId && this.findPriceAgreement(_priceGroupId, inventDimItemDimActivated.inventDimId))   ||
                                                               this.findPriceAgreement(_priceGroupId, InventDim::inventDimIdBlank())            ||
             this.findItemPrice());
На мой взгляд это еще хуже чем написал fed.
В return неудобно анализировать такой код.

В классе sysDatabaseList метод new:
X++:
...
while (1 == 1 && listOfNames)
{
}
...

Последний раз редактировалось kornix; 23.05.2012 в 15:11.
За это сообщение автора поблагодарили: macklakov (1).
Старый 23.05.2012, 14:53   #12  
Napalm is offline
Napalm
Участник
 
80 / 88 (3) ++++
Регистрация: 23.05.2012
Общеизвестно, что для работы разработчиком Dynamics AX не обязательно знать X++ или уметь программировать.

Пример из LedgerJournalEngine\projQtyModified (AX 2012):

X++:
// This switch handles the normal(default) case and 2 other special cases
switch(true)
{
    case _ledgerJournalTrans_Project.Qty == 0:
        _ledgerJournalTrans.AmountCurCredit = 0;
        _ledgerJournalTrans.AmountCurDebit = 0;
        break;

    // If Credit/Debit Amounts are both 0, the Cost Price would be incorrectly set to 0 in the normal case
    // Handle this case separately
    case !_ledgerJournalTrans.AmountCurDebit && !_ledgerJournalTrans.AmountCurCredit && _ledgerJournalTrans_Project.CostPrice:
        this.projRecalcAmountCurDebitCredit(_ledgerJournalTrans, _ledgerJournalTrans_Project);
        break;

    default:
        // this.amount gets the debit amount or the negative credit amount
        offsetFactor = this.projOffsetFactor(_ledgerJournalTrans, _ledgerJournalTrans_Project);
        _ledgerJournalTrans_Project.CostPrice = Currency::price((offsetFactor * _ledgerJournalTrans.amount())/_ledgerJournalTrans_Project.Qty);
        // Cost Price should always be positive
        if (_ledgerJournalTrans_Project.CostPrice < 0)
            _ledgerJournalTrans_Project.CostPrice = - _ledgerJournalTrans_Project.CostPrice;
        // Cost Price have changed, recalc debit and credit amounts to avoid rounding diff
        this.projRecalcAmountCurDebitCredit(_ledgerJournalTrans, _ledgerJournalTrans_Project);
        break;
}
PS: Рекомендовано Microsoft! http://msdn.microsoft.com/en-us/libr...(v=ax.50).aspx
За это сообщение автора поблагодарили: macklakov (1), Zabr (12), db (0).
Старый 23.05.2012, 15:21   #13  
fed is offline
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,907 / 5717 (196) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Цитата:
Сообщение от Napalm Посмотреть сообщение
Общеизвестно, что для работы разработчиком Dynamics AX не обязательно знать X++ или уметь программировать.

Пример из LedgerJournalEngine\projQtyModified (AX 2012):

X++:
// This switch handles the normal(default) case and 2 other special cases
switch(true)
{
    case _ledgerJournalTrans_Project.Qty == 0:
        _ledgerJournalTrans.AmountCurCredit = 0;
        _ledgerJournalTrans.AmountCurDebit = 0;
        break;

    // If Credit/Debit Amounts are both 0, the Cost Price would be incorrectly set to 0 in the normal case
    // Handle this case separately
    case !_ledgerJournalTrans.AmountCurDebit && !_ledgerJournalTrans.AmountCurCredit && _ledgerJournalTrans_Project.CostPrice:
        this.projRecalcAmountCurDebitCredit(_ledgerJournalTrans, _ledgerJournalTrans_Project);
        break;

    default:
        // this.amount gets the debit amount or the negative credit amount
        offsetFactor = this.projOffsetFactor(_ledgerJournalTrans, _ledgerJournalTrans_Project);
        _ledgerJournalTrans_Project.CostPrice = Currency::price((offsetFactor * _ledgerJournalTrans.amount())/_ledgerJournalTrans_Project.Qty);
        // Cost Price should always be positive
        if (_ledgerJournalTrans_Project.CostPrice < 0)
            _ledgerJournalTrans_Project.CostPrice = - _ledgerJournalTrans_Project.CostPrice;
        // Cost Price have changed, recalc debit and credit amounts to avoid rounding diff
        this.projRecalcAmountCurDebitCredit(_ledgerJournalTrans, _ledgerJournalTrans_Project);
        break;
}
PS: Рекомендовано Microsoft! http://msdn.microsoft.com/en-us/libr...(v=ax.50).aspx
Ну это такой любопытный побочный эффект от использования оператора switch(). Сама по себе конструкция абсолютно легальна - пробегает по веткам case и заходит в первую ветку с выполненным условием. Теоретически может быть заменено на серию вложенных if() {} else{ if() {} else {if () {}}, однако вариант с switch() превосходит их по читаемости. На мой взгляд все легально.
Для раздумья о нелегкой судьбе оператора switch() в различных языках программирования, рекомендую прочитать: Duff's device
Старый 23.05.2012, 15:30   #14  
Napalm is offline
Napalm
Участник
 
80 / 88 (3) ++++
Регистрация: 23.05.2012
Цитата:
Сообщение от fed Посмотреть сообщение
Теоретически может быть заменено на серию вложенных if() {} else{ if() {} else {if () {}}, однако вариант с switch() превосходит их по читаемости. На мой взгляд все легально.
"if (a == true)" - тоже легально.
Старый 23.05.2012, 15:37   #15  
fed is offline
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,907 / 5717 (196) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Ну я бы сказал, что if(a==true) пишут от непонимания и не умения программировать, а switch(true) пишут как раз таки от хорошего понимания того как оператор switch работает и приличного опыта разработки Вариант switch(true) изрядно повышает читаемость кода из за отсутствия большого количества вложенных if else...
Старый 23.05.2012, 16:08   #16  
Владимир Максимов is offline
Владимир Максимов
Участник
КОРУС Консалтинг
 
1,692 / 1195 (43) ++++++++
Регистрация: 13.01.2004
Записей в блоге: 3
Цитата:
Сообщение от fed Посмотреть сообщение
Ну я бы сказал, что if(a==true) пишут от непонимания и не умения программировать, а switch(true) пишут как раз таки от хорошего понимания того как оператор switch работает и приличного опыта разработки Вариант switch(true) изрядно повышает читаемость кода из за отсутствия большого количества вложенных if else...
Довольно спорное утверждение. Можно ведь и так написать

X++:
ok = true;

if (ok && ...)
{
   ok = false;
   ...
}

if (ok && ...)
{
   ok = false;
   ...
}
Никаких вложенных if(), конструкция довольно "читабельна" и нет опоры на "фичу" со switch(true). Фича, она фича и есть. Лично я считаю что использование подобных фич - это, скорее, показатель НЕ профессионализма. Банальная лень искать штатные инструменты без опоры на фичи.
__________________
- Может, я как-то неправильно живу?!
- Отчего же? Правильно. Только зря...
За это сообщение автора поблагодарили: macklakov (1).
Старый 23.05.2012, 16:13   #17  
Raven Melancholic is offline
Raven Melancholic
Участник
Аватар для Raven Melancholic
Самостоятельные клиенты AX
Лучший по профессии 2015
 
2,164 / 1293 (48) ++++++++
Регистрация: 21.03.2005
Адрес: Москва-Петушки
Мне тоже кажется, что switch(true), в некоторых случаях, повышает удобочитаемость кода.
Только в данном примере других неизящностей хватает:
X++:
if (_ledgerJournalTrans_Project.CostPrice < 0)
            _ledgerJournalTrans_Project.CostPrice = - _ledgerJournalTrans_Project.CostPrice;
Вместо abs(...)
В одних случаях с нулем сравнивается прямо (что, на мой взгляд более понятно), в других полагаются на приведение числа к логическому типу:
X++:
_ledgerJournalTrans_Project.Qty == 0
X++:
!_ledgerJournalTrans.AmountCurDebit
Старый 23.05.2012, 16:27   #18  
Raven Melancholic is offline
Raven Melancholic
Участник
Аватар для Raven Melancholic
Самостоятельные клиенты AX
Лучший по профессии 2015
 
2,164 / 1293 (48) ++++++++
Регистрация: 21.03.2005
Адрес: Москва-Петушки
Если уж быть совсем занудой, то можно вспомнить теорему Джекопини (точное название не помню), смысл которой в том, что любой алгоритм можно представить всего тремя структурами.
Старый 23.05.2012, 16:33   #19  
fed is offline
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,907 / 5717 (196) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Цитата:
Сообщение от Владимир Максимов Посмотреть сообщение
Довольно спорное утверждение. Можно ведь и так написать

X++:
ok = true;

if (ok && ...)
{
   ok = false;
   ...
}

if (ok && ...)
{
   ok = false;
   ...
}
Ну лично я, протрассировав такой код решил бы что случилась какая-то засада, поскольку Ok = false. Можно конечно переменную поаккуратнее назвать - типа TryNextBranch. Но тогда как-то уж очень громоздко выглядит. Не лучше чем switch(true)
В идеале - нужно было бы добавить в X++ оператор elif или elseif, который бы позволял такие конструкции без вложености писать. Но раз такого оператора нету, остается использовать разные по степени кривости заменители...
Старый 23.05.2012, 16:43   #20  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,438 / 1775 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
Цитата:
Сообщение от Владимир Максимов Посмотреть сообщение
это, скорее, показатель НЕ профессионализма.
Не согласен. "Фича со switch(true)" не является какой-то лазейкой или брешью в компиляторе. Эта "фича" является прямым следствием возможности использовать в операторе Сase не только константы уровня компиляции, но и динамические структуры. Это преимущество, о котором нужно знать и использовать, а не стыдиться его
За это сообщение автора поблагодарили: db (2).
 

Похожие темы
Тема Автор Раздел Ответов Посл. сообщение
Бага в Query update(true) Alexius DAX: Программирование 5 14.09.2011 14:09
Не срабатывает skipDatabaseLog(true) jaran DAX: Программирование 14 09.04.2011 13:22
visible(true) и курсор mvf DAX: Программирование 6 20.07.2005 10:09
recordLevelSecurity(true) sassas DAX: Программирование 12 23.12.2004 16:44

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

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

Рейтинг@Mail.ru
Часовой пояс GMT +3, время: 00:10.