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

 
 
Опции темы Поиск в этой теме Опции просмотра
Старый 06.12.2012, 22:11   #1  
Blog bot is offline
Blog bot
Участник
 
25,644 / 848 (80) +++++++
Регистрация: 28.10.2006
axforum blogs: FizzBuzz
Источник: http://axforum.info/forums/blog.php?b=358
==============

Навеяно статьей FizzBuzz, или почему программисты не умеют программировать

Цитата:
Напишите программу, которая выводит на экран числа от 1 до 100. При этом вместо чисел, кратных трем, программа должна выводить слово «Fizz», а вместо чисел, кратных пяти — слово «Buzz». Если число кратно и 3, и 5, то программа должна выводить слово «FizzBuzz»
X++:
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    ;

    for (i = 1; i <=100; i++)
    {
        switch(true)
        {
            case ((i mod #FIZZ) == 0) && ((i mod #BUZZ) == 0):
                info(strfmt("FizzBuzz %1", i));
                break;
            case ((i mod #FIZZ) == 0):
                info(strfmt("Fizz (%1)", i));
                break;
            case ((i mod #BUZZ) == 0):
                info(strfmt("Buzz (%1)", i));
                break;
            default:
                info(strfmt("%1", i));
        }
    }
}
p.s. Кстати, switch(true) невероятно мощная конструкция, которую можно смело считать реализацией "Сопоставления с образцом"(pattern matching) в Х++.
__________________
Расскажите о новых и интересных блогах по Microsoft Dynamics, напишите личное сообщение администратору.
Старый 06.12.2012, 22:32   #2  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Если честно, то чудовищная реализация, на мой взгляд.
Избыточное вычисление mod и куча лишнего кода.

с точки зрения вычислений лучше уж так.

X++:
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    str fizz;
    str buzz;
    ;

    for (i = 1; i <=100; i++)
    {
        fizz = ((i mod #FIZZ) == 0) ? "Fizz" : "";
        buzz = ((i mod #BUZZ) == 0) ? "Buzz" : "";
        if( fizz || buzz )
        {
            info(strfmt("%1%2 %3", fizz, buzz, i));
        }
    }
}
можно было бы отказаться и от переменных fizz, buzz. Но код стал бы менее читаемым.
Старый 06.12.2012, 22:42   #3  
Lemming is offline
Lemming
Участник
Аватар для Lemming
 
1,144 / 343 (14) ++++++
Регистрация: 20.04.2004
Адрес: Москва, Чайнатаун в Люблино
Записей в блоге: 10
Хмм, что-то у меня не компилируется код, ругается на строчку "fizz = (i mod #FIZZ) == 0) ? "Fizz" : "";" - Типы операнда не совместимы с оператором
Старый 06.12.2012, 22:52   #4  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от Lemming Посмотреть сообщение
Хмм, что-то у меня не компилируется код, ругается на строчку "fizz = (i mod #FIZZ) == 0) ? "Fizz" : "";" - Типы операнда не совместимы с оператором
ага. спасибо. скобку открывающую забыл.
надо fizz = ((i mod #FIZZ) == 0) ? "Fizz" : ""

просто я пока без акспты
За это сообщение автора поблагодарили: Lemming (1).
Старый 06.12.2012, 22:48   #5  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
а вот объектно-ориентированная многословная реализация, если не лень заводить класс в AOT. Но зато тело цикла не содержит деталей реализации - детали в классе.
X++:
class TestState
{
   boolean state;
   str description;
}
void new(int i, int num, int _description)
{
    state = i mod num;
    if( state )
        description = _description;
}
boolean getState()
{
    return state;
}
str getDescription()
{
    return description;
}
/////////////////////////////////////////
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    TestState fizz, buzz;
    ;

    for (i = 1; i <=100; i++)
    {
        fizz = new TestState(i, #FIZZ, "Fizz");
        buzz = new TestState(i, #BUZZ, "Buzz");
        if( fizz.getState() || buzz.getState() )
        {
            info(strfmt("%1%2 %3", fizz.getDescription(), buzz.getDescription(), i));
        }
    }
}
вместо параметра num в методе new можно сделать пару классов-потомков
вместо простой конкатенации строк FizzBuzz можно сделать класс, который формирует вывод...

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

X++:
class TestState
{
   str description;
}
void new(boolean _state, int _description)
{
    if( _state )
        description = _description;
}
str getDescription()
{
    return description;
}
/////////////////////////////////////////
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    TestState fizz, buzz;
    ;

    for (i = 1; i <=100; i++)
    {
        fizz = new TestState(((i mod #FIZZ) == 0), "Fizz");
        buzz = new TestState(((i mod #BUZZ) == 0), "Buzz");
        info(strfmt("%1%2 %3", fizz.getDescription(), buzz.getDescription(), i));
    }
}
Старый 06.12.2012, 22:59   #7  
Lemming is offline
Lemming
Участник
Аватар для Lemming
 
1,144 / 343 (14) ++++++
Регистрация: 20.04.2004
Адрес: Москва, Чайнатаун в Люблино
Записей в блоге: 10
Cool
Все, я сдаюсь, тут опять две ошибки компиляции, одна в new, я ее вроде пофиксил, хотя не факт что правильно, т.к. не до конца понимаю задумку кода, а вторая в запускающем Job'е, снова путаница int/str...mazzy может все-таки не "на бумажке", а то очень интересно проверить результат, а оно работать не хочет... Прошу прощения, не сразу прочел что вы без аксы.

upd: Джобик из второго поста запустил, да, работает. Спасибо!

Последний раз редактировалось Lemming; 06.12.2012 в 23:04. Причина: upd
Старый 06.12.2012, 23:04   #8  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
можно сократить число внутренних переменных в необъектной реализации.
правда не уверен, что код станет быстрее из-за дополнительной сборки мусора.
X++:
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    str state;
    ;

    for (i = 1; i <=100; i++)
    {
        state = "";
        if ((i mod #FIZZ) == 0) { state += "Fizz"; }
        if ((i mod #BUZZ) == 0) { state += "Buzz"; }
        info(strfmt("%1 %2", state, i));
    }
}
switch - зло как правило. Поскольку по мере развития программы число вариантов в switch может меняться. А найти все switch - на редкость нетривиальное дело.

Но как пример использования неконстант в case - очень даже ничего.
Старый 06.12.2012, 23:09   #9  
Lemming is offline
Lemming
Участник
Аватар для Lemming
 
1,144 / 343 (14) ++++++
Регистрация: 20.04.2004
Адрес: Москва, Чайнатаун в Люблино
Записей в блоге: 10
Цитата:
Сообщение от mazzy Посмотреть сообщение
switch - зло как правило. Поскольку по мере развития программы число вариантов в switch может меняться. А найти все switch - на редкость нетривиальное дело.
Не очень понял что именно сложно найти, вроде все ветви в одном месте, хотя когда он становится большим, читать такой код не очень приятно, согласен.
Старый 06.12.2012, 23:11   #10  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от Lemming Посмотреть сообщение
Не очень понял что именно сложно найти
сложно найти все switch, которые относятся к данному случаю (в Аксапте к данному enum)

собственно, стоит почитать Страустрапа и других отцов объектно-ориентированного программирования по поводу switch.
Старый 07.12.2012, 09:06   #11  
gl00mie is offline
gl00mie
Участник
MCBMSS
Most Valuable Professional
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
3,684 / 5813 (201) ++++++++++
Регистрация: 28.11.2005
Адрес: Москва
Записей в блоге: 3
Цитата:
Сообщение от mazzy Посмотреть сообщение
сложно найти все switch, которые относятся к данному случаю (в Аксапте к данному enum)
Как раз для enum'ов switch очень даже прекрасно подходит, если использовать простые правила касаемо default:
  • всегда использовать блок default;
  • либо явно предусматривать в нем логику для "всех остальных" случаев, либо ставить там заглушку, чтобы код сразу валился при неизвестных на момент разработки значениях
Например:
X++:
switch (noYesValue)
{
    case NoYes::No :
        // ...
        break;
    case NoYes::Yes :
        // ...
        break;
    default :
        throw error(Error::unsupportedEnumValue(noYesValue));
}
Здесь Error::unsupportedEnumValue() - метод вида:
X++:
public static LabelType unsupportedEnumValue(anytype _enumValue)
{
    return strfmt(@"Значение %1 (%2) не поддерживается", _enumValue, any2int(_enumValue));
}
За это сообщение автора поблагодарили: Stitch_MS (2).
Старый 16.09.2013, 16:49   #12  
Napalm is offline
Napalm
Участник
 
80 / 88 (3) ++++
Регистрация: 23.05.2012
Цитата:
Сообщение от mazzy Посмотреть сообщение
Если честно, то чудовищная реализация, на мой взгляд.
Избыточное вычисление mod и куча лишнего кода.

с точки зрения вычислений лучше уж так.

X++:
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    str fizz;
    str buzz;
    ;

    for (i = 1; i <=100; i++)
    {
        fizz = ((i mod #FIZZ) == 0) ? "Fizz" : "";
        buzz = ((i mod #BUZZ) == 0) ? "Buzz" : "";
        if( fizz || buzz )
        {
            info(strfmt("%1%2 %3", fizz, buzz, i));
        }
    }
}
можно было бы отказаться и от переменных fizz, buzz. Но код стал бы менее читаемым.

Вот только результат не совпадает с указанным в задании. Считаю, что тест по выявлению "не умеющих писать код" работает как надо.

Задание

Цитата:
Напишите программу, которая выводит на экран числа от 1 до 100. При этом вместо чисел, кратных трем, программа должна выводить слово «Fizz», а вместо чисел, кратных пяти — слово «Buzz». Если число кратно и 3, и 5, то программа должна выводить слово «FizzBuzz»
Результат
Цитата:
Fizz 3
Buzz 5
Fizz 6
Fizz 9
Buzz 10
Fizz 12
FizzBuzz 15
Fizz 18
Buzz 20
Fizz 21
Fizz 24
Buzz 25
Fizz 27
FizzBuzz 30
Fizz 33
Buzz 35
Fizz 36
Fizz 39
Buzz 40
Fizz 42
FizzBuzz 45
Fizz 48
Buzz 50
Fizz 51
Fizz 54
Buzz 55
Fizz 57
FizzBuzz 60
Fizz 63
Buzz 65
Fizz 66
Fizz 69
Buzz 70
Fizz 72
FizzBuzz 75
Fizz 78
Buzz 80
Fizz 81
Fizz 84
Buzz 85
Fizz 87
FizzBuzz 90
Fizz 93
Buzz 95
Fizz 96
Fizz 99
Buzz 100
За это сообщение автора поблагодарили: mazzy (5).
Старый 16.09.2013, 17:56   #13  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от Napalm Посмотреть сообщение
Вот только результат не совпадает...
Результат
1. в чем именно не совпадает?
2. вы привели результат чего именно?
Старый 18.09.2013, 12:58   #14  
Stitch_MS is offline
Stitch_MS
Участник
Аватар для Stitch_MS
Соотечественники
 
397 / 483 (17) +++++++
Регистрация: 27.02.2006
Адрес: Дания
Цитата:
Сообщение от mazzy Посмотреть сообщение
1. в чем именно не совпадает?
"...При этом вместо чисел..."
За это сообщение автора поблагодарили: mazzy (5).
Старый 18.09.2013, 14:01   #15  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от Stitch_MS Посмотреть сообщение
"...При этом вместо чисел..."
Ай, хорошо. Ай, люблю...

Спасибо.
исправленная версия

X++:
#define.FIZZ(3)
#define.BUZZ(5)
static void FizzBuzzImpl(Args _args)
{
    int i;
    str fizz;
    str buzz;
    ;

    for (i = 1; i <=100; i++)
    {
        fizz = ((i mod #FIZZ) == 0) ? "Fizz" : "";
        buzz = ((i mod #BUZZ) == 0) ? "Buzz" : "";
        if( fizz || buzz )
        {
            info(strfmt("%1%2", fizz, buzz));
        }
        else
        {
            info(strfmt("%1", i));
        }
    }
}
За это сообщение автора поблагодарили: S.Kuskov (1).
Старый 18.09.2013, 13:14   #16  
ALES is offline
ALES
Участник
Злыдни
 
220 / 45 (2) +++
Регистрация: 11.08.2004
Цитата:
Сообщение от mazzy Посмотреть сообщение
1. в чем именно не совпадает?
судя по приведенному результату в том, что вывод с 3ки идет, вместо " выводит на экран числа от 1 до 100."
За это сообщение автора поблагодарили: mazzy (2).
Старый 07.12.2012, 09:43   #17  
S.Kuskov is offline
S.Kuskov
Участник
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
 
3,448 / 1792 (66) ++++++++
Регистрация: 28.04.2007
Адрес: Калуга
Цитата:
Сообщение от Blog bot Посмотреть сообщение
switch(true) невероятно мощная конструкция, которую можно смело считать реализацией "Сопоставления с образцом"(pattern matching) в Х++.
Только нужно понимать, что образцы в этом случае должны быть взаимоисключающими. Т.е. в каждый конкретный момент истинным должно быть только одно условие, иначе логика выполнения может стать отличной от ожидаемой.
Из-за этого добавление нового условия к уже имеющимся может потребовать изменить и все остальные.

Ещё как-то встречал извращенцев умельцев играющих на опускании break в определённых ветках case. Уследить за полётом мысли таких "авторов" бывает очень сложно.

P.S.: Для решения задачи из первого поста подходит паттерн цепочка декораторов
За это сообщение автора поблагодарили: mazzy (2).
Старый 07.12.2012, 11:35   #18  
Lemming is offline
Lemming
Участник
Аватар для Lemming
 
1,144 / 343 (14) ++++++
Регистрация: 20.04.2004
Адрес: Москва, Чайнатаун в Люблино
Записей в блоге: 10
Цитата:
Сообщение от S.Kuskov Посмотреть сообщение
Ещё как-то встречал извращенцев умельцев играющих на опускании break в определённых ветках case. Уследить за полётом мысли таких "авторов" бывает очень сложно.
Например code conventions в java(во всяком случае я там про это прочел) предлагает помечать пропуски break'ов комментарием fall-through, что бы программисты читающие код были предупреждены что это не ошибка, а задумка.
За это сообщение автора поблагодарили: S.Kuskov (1).
Старый 07.12.2012, 12:54   #19  
fed is offline
fed
Moderator
Аватар для fed
Ex AND Project
Соотечественники
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
2,913 / 5736 (197) ++++++++++
Регистрация: 13.03.2002
Адрес: Hüfingen,DE
Хотя я уже лет 15 не программирую на C/C++, мне конструкции вида (i mod #FIZZ) == 0 режут глаз.
Я бы написал !(i mod #FIZZ). Я понимаю что это вопрос стиля, вкуса и вообще для интерпретируемого языка разница в производительности отсутствует. Но глаз все равно режет...
За это сообщение автора поблагодарили: mazzy (2).
Старый 07.12.2012, 13:42   #20  
Stitch_MS is offline
Stitch_MS
Участник
Аватар для Stitch_MS
Соотечественники
 
397 / 483 (17) +++++++
Регистрация: 27.02.2006
Адрес: Дания
Можно и так (кода больше, но резь в глазах уменьшится )

X++:
class Number
{
    int number;
    public void new(int _number)
    {
        number = _number;
    }
    public boolean isDivisibleBy(int _denominator)
    {
        return (number mod _denominator) == 0;
    }
}
static void testFizzBuzz(Args _args)
{
    str fullWord;
    Number number;
    int i;
    for (i = 1; i <= 100; i++)
    {
        fullWord = '';
 
        number = new Number(i);
 
        if (number.isDivisibleBy(3))
        {
            fullWord += "Fizz";
        }
 
        if (number.isDivisibleBy(5))
        {
            fullWord += "Buzz";
        }        
 
        if (fullWord != '')
        {
            info(strFmt("%1 %2", i, fullWord));
        }
    }
}
 

Похожие темы
Тема Автор Раздел Ответов Посл. сообщение
emeadaxsupport: Budget entries import in AX2012 Blog bot DAX Blogs 0 21.11.2012 00:12
DynamicsAxSCM: Visualizing Security in Microsoft Dynamics AX 2012 Blog bot DAX Blogs 0 29.08.2011 13:11
dynamicsaxbi: Screenshots of Dynamics AX 2009 Role Centers Blog bot DAX Blogs 0 18.03.2011 20:11
DynamicsAxSCM: Personalization of Role Centers in Dynamics AX 2009 Blog bot DAX Blogs 0 21.06.2010 16:05
Опции темы Поиск в этой теме
Поиск в этой теме:

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

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

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

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