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

 
 
Опции темы Поиск в этой теме Опции просмотра
Старый 13.06.2008, 12:31   #1  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Foreign Key в Ax
Вероятно для большинства нижеизложенное не представляет интереса, но для начинающих, как я, это не так. Речь идет о том, как правильно сделать Foreign Key в MS DAX 4.0.

Хочу сделать пару таблиц master-detail.
Для этого в Ах 4.0 делаю следующее:
1.Создаю EDT TestId.
2.Создаю таблицы TestMaster и TestDetail. В каждую из них кидаю TestId и Descripnion.
3.Создаю для таблицы TestMaster уникальный индекс TestIdx по полю TestId. Делаю его PrimaryIndex для этой таблицы.
4.Для таблицы TestDetail делаю не уникальный индекс по полю TestId.
5.В EDT TestId делаю relation TestId == TestMaster.TestId
6.В свойствах полей TestId в обеих таблицах ставлю Mandatory = Yes.
7.Для таблицы TestMaster делаю DeleteAction = Restricted по отношению к таблице TestDetail.

Вроде бы все. Можно смотреть результат .
Открываю Обозреватель таблиц для TestMaster и TestDetail.
Пытаюсь вставить значение в TestDetail. Не получается . Хорошо .
Вставляю пару строк в TestMaster (1 и 2). Вставляю пару строк в TestDetail для каждого значения из Master.
Пытаюсь удалить из TestMaster. Не получается . Хорошо .
Удаляю из TestDetail и снова пытаюсь удалить из TestMaster. Получается .
Захожу на TestMaster и пытаюсь поменять TestId с 2 на 3. Получается .
Ничего себе, строки в TestDetail стали сиротами!

Отпив и бутылочки корвалола продолжаю тестировать.
Создаю job такого содержания:
Код:
static void TestJobMD(Args _args)
{
    TestDetail testDetail;
; 
    ttsbegin;
    TestDetail.TestId = "100";
    TestDetail.Description = "Test 100";
 
    if (TestDetail.validateWrite())
        TestDetail.insert();
    ttscommit; 
}
Выполняю. Проходит!!! Не верю! Открываю Обозреватель таблиц - таки да!

Допиваю остатки корвалола продолжаю. Создаю еще job:
Код:
static void TestDeleteMD(Args _args)
{
    TestMaster testMaster;
;
    ttsbegin;
    select forupdate testMaster
        where testMaster.TestId == "2";

    if ( testMaster.validateDelete())
         testMaster.delete();
    ttscommit;
}
Предварительно в TestMaster возвращаю строку с TestId = 2.
Запускаю. Не проходит. Ну хоть тут Слава Богу.

А теперь вопрос к знатокам. Я что-то не так сделал?

Как человек, долго работавший с Oracle добавлю. Там (да и в MS SQL) таких проблем просто нет!
Пару строк в скрипте создания таблиц.(Primary Key + Foreign Key) и вы получаете надежную ссылочную целостность, выдерживающую прямое попадание любой ядерной бомбы, запущенной умелой рукой пользователя .
А в Ах что получается: спасение утопающих - дело рук самих утопающих? Т.е. программист должен сам все проверять?

Еще раз повторю. Может я что-то не так сделал? Тогда готов посыпать голову пеплом.
Старый 13.06.2008, 12:58   #2  
Михаил Андреев is offline
Михаил Андреев
Участник
Компания АМАНД
Лучший по профессии 2009
 
1,295 / 239 (10) ++++++
Регистрация: 09.11.2001
Адрес: Химки, Московская область
Цитата:
Сообщение от SergeK Посмотреть сообщение
Открываю Обозреватель таблиц для TestMaster и TestDetail.
У Вас каждый пользователь имеет доступ к обозревателю таблиц? Обозреватель - инструмент "хирурга", а не простого пользователя. У Вас все пользователи Oracle имели права администратора?

Кроме того, нужно было хотя бы настроить Relation таблицы Detail на Master по ключевому полю со свойством Validate = Yes, а лучше сделать это в EDT.

P.S. Всё описанное подробно рассматривается на тренингах по разработке и кратко описано в документации "Справка разработчика".
__________________
Михаил Андреев
https://www.amand.ru
Старый 13.06.2008, 13:30   #3  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Цитата:
Сообщение от Михаил Андреев Посмотреть сообщение
У Вас каждый пользователь имеет доступ к обозревателю таблиц? Обозреватель - инструмент "хирурга", а не простого пользователя.
Ну положим я мог бы сделать простую форму, ориентированную на пользователя и получить тот же результат.

Цитата:
Сообщение от Михаил Андреев Посмотреть сообщение
Кроме того, нужно было хотя бы настроить Relation таблицы Detail на Master по ключевому полю со свойством Validate = Yes, а лучше сделать это в EDT.
Вот это может и поможет, хотя есть сомнения .
А в EDT свойства Validate нет, это на таблицах.
Кстати, кто-нибудь может объяснить, что означает отношение
CustTrans.AccountNum == CustTrans.AccountNum

Цитата:
Сообщение от Михаил Андреев Посмотреть сообщение
P.S. Всё описанное подробно рассматривается на тренингах по разработке и кратко описано в документации "Справка разработчика".
Да уж больно кратко

Вообще, на мой взгляд самая неприятная ситуация - это то, что таблица-detail может заполняться автоматически в коде и программист может ошибиться (все мы люди), причем не грубо, а только в некоторых специфических ситуациях могут появиться сироты, причем система шума не поднимет, и выяснится это нескоро и починить будет непросто.
Старый 13.06.2008, 14:39   #4  
Михаил Андреев is offline
Михаил Андреев
Участник
Компания АМАНД
Лучший по профессии 2009
 
1,295 / 239 (10) ++++++
Регистрация: 09.11.2001
Адрес: Химки, Московская область
Цитата:
Сообщение от SergeK Посмотреть сообщение
Кстати, кто-нибудь может объяснить, что означает отношение
CustTrans.AccountNum == CustTrans.AccountNum
Видимо, документ "Справка разработчика" Вы не дочитали. Ответ в разделе "Best Practices for Table Relations".
Цитата:
Сообщение от SergeK Посмотреть сообщение
Да уж больно кратко
Более подробно могут разъяснить наставники или на тренингах узнать. Но хотя бы этот документ обязательно нужно прочесть, чтобы понимать принципы разработки в Аксапте.
__________________
Михаил Андреев
https://www.amand.ru

Последний раз редактировалось Михаил Андреев; 13.06.2008 в 14:48.
Старый 13.06.2008, 15:20   #5  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Цитата:
Сообщение от Михаил Андреев Посмотреть сообщение
Видимо, документ "Справка разработчика" Вы не дочитали.
Да, каюсь, не дочитал Хотя подозреваю, что мало кто его прочел от корки до корки, особенно начинающие. Читают то, что нужно сейчас. Если найдут

Цитата:
Сообщение от Михаил Андреев Посмотреть сообщение
Ответ в разделе "Best Practices for Table Relations".
Вот за это большое спасибо. Счас буду разбираться.
Старый 15.06.2008, 14:48   #6  
glibs is offline
glibs
Member
Сотрудники компании It Box
Most Valuable Professional
Лучший по профессии 2011
Лучший по профессии 2009
 
4,942 / 911 (40) +++++++
Регистрация: 10.06.2002
Адрес: I am from Kyiv, Ukraine. Now I am in Moscow. For private contacts: glibs@hotmail.com
Цитата:
Сообщение от SergeK
...
Делаю его PrimaryIndex для этой таблицы.
...
Это мало на что влияет. Работает не так, как нечто созвучное в MS SQL, по крайней мере.
Цитата:
Сообщение от SergeK
...
Захожу на TestMaster и пытаюсь поменять TestId с 2 на 3. Получается .
Ничего себе, строки в TestDetail стали сиротами!
...
В ключевом поле AllowEdit ставится в false, а AllowEditOnCreate в true. Переименование только через паспорт записи. А если пишете код — то думать головой, и писать примерно так (писал по памяти и не в компиляторе, так что проверяйте и не забудьте транзакцию еще дописать):

select forupdate table
where ...
table.field = "new value";
table.renamePrimaryKey();
table.update();
__________________
С уважением,
glibs®
За это сообщение автора поблагодарили: mazzy (2).
Старый 16.06.2008, 11:04   #7  
belugin is offline
belugin
Участник
Аватар для belugin
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,622 / 2925 (107) +++++++++
Регистрация: 16.01.2004
Записей в блоге: 5
Цитата:
Сообщение от glibs Посмотреть сообщение

X++:
select forupdate table
where ...
table.field = "new value";
table.renamePrimaryKey();
table.update();
update не нужен - renamePrimaryKey вместо update
За это сообщение автора поблагодарили: glibs (1).
Старый 16.06.2008, 11:40   #8  
glibs is offline
glibs
Member
Сотрудники компании It Box
Most Valuable Professional
Лучший по профессии 2011
Лучший по профессии 2009
 
4,942 / 911 (40) +++++++
Регистрация: 10.06.2002
Адрес: I am from Kyiv, Ukraine. Now I am in Moscow. For private contacts: glibs@hotmail.com
Ага, спасибо. Перепутал с этим

Проблема изменения номера партии в складской аналитике
__________________
С уважением,
glibs®
Старый 17.06.2008, 12:54   #9  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Спасибо всем откликнувшимся!
К сожалению, подтверждаются самые худшие мои опасения .
Т.е. поддержка ограничения внешний ключ весьма убогая.
Может быть гуру Ах могут как то прокомментировать, почему не задействована эта возможность на уровне СУБД?
Кстати в Best Practices for Table Relations написано следующее:
Цитата:
The system guarantees that data entered in the database fulfills the specified relations.
Т.е. неправда.

И еще
Цитата:
Сообщение от Михаил Андреев Посмотреть сообщение
Цитата:
Сообщение от SergeK Посмотреть сообщение
Кстати, кто-нибудь может объяснить, что означает отношение
CustTrans.AccountNum == CustTrans.AccountNum
Видимо, документ "Справка разработчика" Вы не дочитали. Ответ в разделе "Best Practices for Table Relations".
Я извиняюсь за назойливость, но тут написано, что это нужно сделать, если таблица имеет ключ. НО! Ничего не написано, почему это нужно сделать
Старый 17.06.2008, 13:57   #10  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Цитата:
Сообщение от SergeK Посмотреть сообщение
Может быть гуру Ах могут как то прокомментировать, почему не задействована эта возможность на уровне СУБД?
Когда эта Аксапта разрабатывалась, она поддерживала несколько СУБД. В том числе и такие, где ключи на уровне СУБД отсутствовали. С тех пор прошло время и осталось две СУБД в качестве формата.

Похоже, что через несколько версий останется только одна СУБД. Только тогда, наверное, и можно будет говорить о необходимости поддержки всех возможностей СУБД.

Цитата:
Сообщение от SergeK Посмотреть сообщение
Кстати в Best Practices for Table Relations написано следующее:

Т.е. неправда.
Правда. Только под "system" подразумевается сама Аксапта, а не СУБД.


Цитата:
Сообщение от SergeK Посмотреть сообщение
Я извиняюсь за назойливость, но тут написано, что это нужно сделать, если таблица имеет ключ. НО! Ничего не написано, почему это нужно сделать
Ищите. Здесь правильный ответ давал EVGL: Это нужно для процедуры переименования ключей, если ядро видит неоднозначности.
__________________
полезное на axForum, github, vk, coub.
Старый 17.06.2008, 15:39   #11  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Цитата:
Сообщение от mazzy Посмотреть сообщение
Правда. Только под "system" подразумевается сама Аксапта, а не СУБД.
Согласен, если работа ведется через формы. Но как только мы пишем код, даже если пользуемся стандартным API Аксапты, это становится неправдой. Мы легко вставим в detail записи-сироты.
Или я неправ?
Старый 17.06.2008, 15:54   #12  
kashperuk is offline
kashperuk
Участник
Аватар для kashperuk
MCBMSS
Соотечественники
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,361 / 2084 (78) +++++++++
Регистрация: 30.05.2004
Адрес: Atlanta, GA, USA
Это "фича" системы, я бы сказал, гибкость.
Если разработчик понимает, что делает, то все будет написано правильно.
Но вдруг ему таки нужно запихнуть строки не ссылающиеся на что-то?
К примеру, создать строки заказа, не привязанные к конкретному заказу (простой способ создать проводки, кстати, без добавления нового типа проводок)
Старый 17.06.2008, 16:18   #13  
mazzy is offline
mazzy
Участник
Аватар для mazzy
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии AXAWARD 2013
Лучший по профессии 2011
Лучший по профессии 2009
 
29,472 / 4494 (208) ++++++++++
Регистрация: 29.11.2001
Адрес: Москва
Записей в блоге: 10
Согласен с kashperuk'ом.

Цитата:
Сообщение от SergeK Посмотреть сообщение
Согласен, если работа ведется через формы. Но как только мы пишем код, даже если пользуемся стандартным API Аксапты, это становится неправдой. Мы легко вставим в detail записи-сироты.
Или я неправ?
Вставите, если сознательно будете обходить стандартные модули.
Программисту дается такая возможность.
Но ответственность за неправильное функционирование в этом случае лежит на нем, на программисте.
__________________
полезное на axForum, github, vk, coub.
Старый 17.06.2008, 16:33   #14  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Всем спасибо! Туман рассеялся
Отдельное спасибо EVGL за неоднократное объяснение, зачем нужны Self-Relations.
Поскольку этот вопрос регулярно задают начинающие, считаю, что ему самое место в FAQ.
Старый 17.06.2008, 16:49   #15  
kashperuk is offline
kashperuk
Участник
Аватар для kashperuk
MCBMSS
Соотечественники
Сотрудники Microsoft Dynamics
Лучший по профессии 2017
Лучший по профессии 2015
Лучший по профессии 2014
Лучший по профессии 2011
Лучший по профессии 2009
 
4,361 / 2084 (78) +++++++++
Регистрация: 30.05.2004
Адрес: Atlanta, GA, USA
Цитата:
Сообщение от SergeK Посмотреть сообщение
Отдельное спасибо EVGL за неоднократное объяснение, зачем нужны Self-Relations.
Поскольку этот вопрос регулярно задают начинающие, считаю, что ему самое место в FAQ.
Дай прямую ссылку на конкретный ответ - и думаю, добавят
Старый 17.06.2008, 17:06   #16  
SergeK is offline
SergeK
Участник
 
16 / 10 (1) +
Регистрация: 10.06.2008
Цитата:
Сообщение от kashperuk Посмотреть сообщение
Дай прямую ссылку на конкретный ответ - и думаю, добавят
Да, не вопрос!
Цитата:
Сообщение от EVGL Посмотреть сообщение
Сам поставил опыт: вы не поверите, но поля в индексе, который выбран как PrimaryIndex на таблице, действительно становятся членами первичного ключа в SQL!
Relations не при чем, а то, что написано в документации, относится к первичному ключу в понимании Аксапты, т.е. к полю, которое можно переименовывать. Именно для глобального переименования кодов справочников и нужны self-relatations.
Цитата:
Сообщение от EVGL Посмотреть сообщение
Запуск BestPractice дает ответ на этот вопрос. Self-relation нужны для переименования первичных ключей.
Если расширенный тип не имеет такой ссылки, то BestPractice выдает соответствущее предупреждение и ждет, что ссылка будет создана на самой таблице. Однако в ряде таблиц (RAssetMoveHist, скорее всего, в их числе) первичного ключа либо нет, либо он состоит из нескольких полей. Тут попытка создать Self-relation ни к чему не приводит: BestPractice продолжает ошибочно выдавать все то же предупреждение.

Есть и еще одна причина: в ряде случаев программист хочет, чтобы при вызове формы создалась динамическая ссылка с источником данных вызываеющего объекта. Пример: вызов проводок по ГК из плана счетов. А если вызывающая и вызываемая таблицы одинаковые? Если таблицу рабочих центров вызываем из формы рабочих центров и хотим видеть, как содержимое второй формы фильтруется по номеру из первой? Автоматически наложить правильный DynaLink на таблицу (особенно с первичным ключом из двух и более полей) в таком случае помогает Self-relation.
Цитата:
Сообщение от EVGL Посмотреть сообщение
Не в первый раз поднимался этот вопрос. Эта relation сделана для того, чтобы Best Practices не ругался. Вот и все. А Best Practices ругается тогда, когда переименование первичного ключа невозможно, т.е. нет уникального индекса. Поэксперементируйте, и все станет понятно.
Теги
ax4.0

 

Похожие темы
Тема Автор Раздел Ответов Посл. сообщение
Dynamics AX: Managing Your Supply Chain Using Microsoft Dynamics AX 2009 - Book Review Blog bot DAX Blogs 0 31.03.2009 23:06
axStart: Microsoft Dynamics AX 2009 Hot Topics Web Seminar Series Blog bot DAX Blogs 0 06.08.2008 12:05
Arijit Basu: AX 2009 - Quick Overview Blog bot DAX Blogs 4 19.05.2008 14:47
Arijit Basu: Reporting & BI in AX: An Overview [Level 100] Blog bot DAX Blogs 0 07.01.2008 16:01
при построении перекрёстных ссылок выдаётся сообщение об ошибках mmmax DAX: Программирование 10 21.01.2005 12:42

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

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

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