Nota
O acesso a esta página requer autorização. Podes tentar iniciar sessão ou mudar de diretório.
O acesso a esta página requer autorização. Podes tentar mudar de diretório.
Observação
Os grupos de interesse da comunidade passaram do Yammer para o Microsoft Viva Engage. Para participar de uma comunidade Viva Engage e participar das últimas discussões, preencha o formulário Solicitar acesso ao Finance and Operations Viva Engage Community e escolha a comunidade à qual deseja participar.
Este artigo descreve o uso de atributos no X++.
Um atributo é uma classe não abstrata que estende (herda de) a classe SysAttribute . Os atributos representam ou armazenam metadados sobre tipos e métodos. Um atributo pode ser anexado a uma classe, um campo de classe, um método de classe, uma interface ou uma tabela.
Os atributos são aplicados aos manipuladores de delegados e métodos, para mapear os manipuladores para esses destinos.
Criando uma classe de atributo
Uma classe de atributo pode estender a classe SysAttribute diretamente, ou pode estender qualquer descendente da classe SysAttribute . A classe SysAttribute não pode ser usada como um atributo porque é declarada abstrata. O exemplo a seguir mostra a declaração e o design de uma classe de atributo comum que você pode criar.
public class PracticeAttribute extends SysAttribute
{
// Fields in the classDeclaration.
StartEnd startEndEnum;
str reason;
// Constructor.
public void new(StartEnd _startEndEnum, str _reason)
{
startEndEnum = _startEndEnum;
reason = _reason;
}
// Other methods can go here.
}
Decorando uma classe com um atributo
O exemplo a seguir mostra uma classe e um método que são decorados com o PracticeAttribute dado no exemplo anterior. Se o construtor do atributo não usa parâmetros, os parênteses para os parâmetros são opcionais. A decoração do atributo poderia ser [AnotherAttribute] sem parênteses.
[PracticeAttribute(StartEnd::End, "Use the RegularClass class at the end.")]
public class RegularClass
{
[PracticeAttribute(Startend::Start, "Use the rehearse method at the start.")]
public int rehearse()
{
// Logic goes here.
}
// More fields and methods belong here.
}
Você pode omitir o sufixo do nome do atributo se o sufixo for Attribute. Por exemplo, você pode usar [Practice] em vez disso [PracticeAttribute] no exemplo anterior.
Construtores de atributos
Você pode habilitar sua classe de atributo para armazenar metadados personalizados cada vez que ela é usada para decorar uma classe, fazendo com que seu construtor tome parâmetros. Os parâmetros para o construtor devem ser literais dos tipos primitivos, como int,enum ou str. O compilador não constrói uma instância da classe de atributo. Ele armazena o nome da classe de atributo, mais os valores literais para seu construtor. Portanto, se a lógica em um construtor de atributo lançasse uma exceção, a exceção não seria encontrada decorando uma classe com o atributo. A exceção seria encontrada mais tarde, quando um processo examina uma classe para ver o atributo com o qual ela está decorada. É quando o atributo é construído.
Convenções de nomenclatura
Todas as classes de atributos têm o sufixo Attribute em seu nome. O sufixo Atributo é a convenção de nome que recomendamos, mas não é um requisito do sistema. Você pode determinar se uma classe se estende diretamente de SysAttribute selecionando a classe no Application Explorer e examinando a propriedade Extends na janela Properties .
SysObsoleteAttribute
O sistema fornece vários atributos, incluindo a classe SysObsoleteAttribute . Um uso da classe SysObsoleteAttribute é notificar o compilador de que a compilação deve falhar se um determinado método for chamado no código-fonte. O compilador rejeita a compilação e exibe a mensagem específica armazenada neste uso do atributo. A classe SysObsoleteAttribute também pode ser usada para notificar o compilador para emitir mensagens de aviso em vez de erros.
Exemplo de código SysObsoleteAttribute
[SysObsoleteAttribute("The Automobile class might have faster performance.", false)]
class Bicycle
{
// Members of the Bicycle class go here.
}
Reflexão de metadados
Use a reflexão para localizar os metadados de atributo anexados a uma classe. As classes a serem usadas para reflexão de atributos são as seguintes:
- Classe DictClass – Para classes e interfaces.
- Classe DictMethod – Para métodos em classes, interfaces ou tabelas.
Nas classes de reflexão anteriores, os métodos para refletir sobre metadados de atributos são os seguintes:
- Método getAllAttributes
- Método getAttribute
- Método getAttributedClasses
- Método getAttributes
Observação
Não há nenhum mecanismo para listar todos os métodos ou classes que são adornados com um atributo específico do código X++. No entanto, como o compilador X++ registra essas informações no banco de dados de referência cruzada, as informações podem ser extraídas a partir daí.
Exemplo de código de reflexão de metadados
Use a classe DictMethod para localizar o valor de metadados de um atributo que é decoração em um método. O exemplo de código a seguir usa o SysEntryPointAttribute classe como o atributo. Ele aceita seus valores de parâmetro para o nome do método e para o nome da classe que contém o método. O método parmChecked é particular para a classe SysEntryPointAttribute e não é herdado de sua classe base SysAttribute. Cada classe de atributo pode ter seu próprio nome de método para seus metadados.
static public int MetadataOfSysEntryPointAttributeOnMethod
(
str _sNameOfClass,
str _sNameOfMethod
)
{
// Return Values:
// 0 == Has the attribute, its metadata value is false;
// 1 == Has the attribute, its metadata value is true;
// 2 == The method lacks the SysEntryPointAttribute.
int nReturnValue = -1,
nClassId;
boolean boolParmChecked;
DictMethod dm;
Object attributeAsObject;
SysEntryPointAttribute sepAttribute;
Global::info("Starting AttributeReflection"
+ " ::MetadataOfSysEntryPointAttributeOnMethod ....");
Global::info(strFmt
("Parameters are: _sNameOfClass = %1 , _sNameOfMethod = %2 .",
_sNameOfClass, _sNameOfMethod)
);
nClassId = Global::className2Id(_sNameOfClass);
dm = new DictMethod
(UtilElementType::ClassInstanceMethod,
nClassId,
_sNameOfMethod
);
attributeAsObject = dm.getAttribute("SysEntryPointAttribute");
if (attributeAsObject is SysEntryPointAttribute)
{
sepAttribute = attributeAsObject as SysEntryPointAttribute;
boolParmChecked = sepAttribute.parmChecked();
if (boolParmChecked)
nReturnValue = 1;
else
nReturnValue = 0;
Global::info(
strFmt("Return value is %1.",
nReturnValue)
);
}
else
{
nReturnValue = 2;
Global::error("Object is not a SysEntryPointAttribute??");
}
return nReturnValue;
}
/*** Output displayed in the Infolog.
Message (05:03:22 pm)
Starting AttributeReflection ::MetadataOfSysEntryPointAttributeOnMethod ....
Parameters are: _sNameOfClass = CustCustomerService , _sNameOfMethod = create .
Return value is 1.
***/
/**************
// Simple AOT > Jobs job to run the method.
static void AttributeReflection33Job(Args _args)
{
AttributeReflection::MetadataOfSysEntryPointAttributeOnMethod
("CustCustomerService", "create");
}
**************/