Preenchendo uma tabela hierárquica utilizando métodos hierárquicos
O AdventureWorks2008R2 tem 8 funcionários que trabalham no departamento de marketing. A hierarquia dos funcionários é assim:
David, EmployeeID 6, é o gerente de marketing. Três especialistas em marketing são subordinados a David:
Sariya, EmployeeID 46
John, EmployeeID 271
Jill, EmployeeID 119
Wanida, assistente de marketing (EmployeeID 269), é subordinada a Sariya, e a assistente de marketing Mary (EmployeeID 272), é subordinada a John.
Para inserir a raiz da árvore de hierarquia
O exemplo a seguir insere David, o gerente de marketing, na tabela na raiz da hierarquia. A coluna OrdLevel é uma coluna computada. Portanto, não faz parte da instrução INSERT. Este primeiro registro usa o método GetRoot () para popular o primeiro registro como a raiz da hierarquia.
INSERT HumanResources.EmployeeOrg (OrgNode, EmployeeID, EmpName, Title) VALUES (hierarchyid::GetRoot(), 6, 'David', 'Marketing Manager') ; GO
Execute o seguinte código para examinar a linha inicial na tabela:
SELECT OrgNode.ToString() AS Text_OrgNode, OrgNode, OrgLevel, EmployeeID, EmpName, Title FROM HumanResources.EmployeeOrg ;
Aqui está o conjunto de resultados.
Text_OrgNode OrgNode OrgLevel EmployeeID EmpName Title ------------ ------- -------- ---------- ------- ----------------- / Ox 0 6 David Marketing Manager
Como na lição anterior, usamos o método ToString() para converter o tipo de dados hierarchyid em um formato mais facilmente entendido.
Para inserir um funcionário subordinado
Sariya é subordinada a David. Para inserir o nó de Sariya, você deve criar um valor OrgNode apropriado do tipo de dados hierarchyid. O código a seguir cria uma variável do tipo de dados hierarchyid e a popula com o valor OrgNode de raiz da tabela. Depois, usa essa variável com o método GetDescendant () para inserir a linha que é um nó subordinado. O GetDescendant obtém dois argumentos. Analise as seguintes opções para os valores de argumento:
Se o pai for o NULL, o GetDescendant retornará NULL.
Se o pai não for NULL, e child1 e child2 forem NULL, o GetDescendant retornará um filho de pai.
Se o pai e child1 forem NULL e child2 for NULL, o GetDescendant retornará um filho de pai maior que child1.
Se o pai e child2 não forem NULL e child1 for NULL, o GetDescendant retornará um filho de pai menor que child2.
Se o pai, child 1 e child 2 não forem NULL, o GetDescendant retornará um filho de pai maior que child1 e menor que child2.
O código a seguir usa os argumentos (NULL, NULL) do pai da raiz, pois ainda não há linhas na tabela exceto a raiz. Execute o seguinte código para inserir Sariya:
DECLARE @Manager hierarchyid SELECT @Manager = hierarchyid::GetRoot() FROM HumanResources.EmployeeOrg ; INSERT HumanResources.EmployeeOrg (OrgNode, EmployeeID, EmpName, Title) VALUES (@Manager.GetDescendant(NULL, NULL), 46, 'Sariya', 'Marketing Specialist') ;
Repita a consulta do primeiro procedimento para consultar a tabela e verificar como as entradas são exibidas:
SELECT OrgNode.ToString() AS Text_OrgNode, OrgNode, OrgLevel, EmployeeID, EmpName, Title FROM HumanResources.EmployeeOrg ;
Aqui está o conjunto de resultados.
Text_OrgNode OrgNode OrgLevel EmployeeID EmpName Title ------------ ------- -------- ---------- ------- ----------------- / Ox 0 6 David Marketing Manager /1/ 0x58 1 46 Sariya Marketing Specialist
Para criar um procedimento para inserir novos nós
Para simplificar a inserção de dados, crie o procedimento armazenado a seguir para adicionar funcionários à tabela EmployeeOrg. O procedimento aceita valores de entrada sobre o empregado sendo adicionado. Isso inclui o EmployeeID do gerente do novo funcionário, o número EmployeeID do novo funcionário, além de seu nome e cargo. O procedimento usa o método GetDescendant() e GetAncestor(). Execute o seguinte código para criar o procedimento:
CREATE PROC AddEmp(@mgrid int, @empid int, @e_name varchar(20), @title varchar(20)) AS BEGIN DECLARE @mOrgNode hierarchyid, @lc hierarchyid SELECT @mOrgNode = OrgNode FROM HumanResources.EmployeeOrg WHERE EmployeeID = @mgrid SET TRANSACTION ISOLATION LEVEL SERIALIZABLE BEGIN TRANSACTION SELECT @lc = max(OrgNode) FROM HumanResources.EmployeeOrg WHERE OrgNode.GetAncestor(1) =@mOrgNode ; INSERT HumanResources.EmployeeOrg (OrgNode, EmployeeID, EmpName, Title) VALUES(@mOrgNode.GetDescendant(@lc, NULL), @empid, @e_name, @title) COMMIT END ; GO
O exemplo a seguir adiciona os 4 funcionários restantes que são subordinados direta ou indiretamente a David.
EXEC AddEmp 6, 271, 'John', 'Marketing Specialist' ; EXEC AddEmp 6, 119, 'Jill', 'Marketing Specialist' ; EXEC AddEmp 46, 269, 'Wanida', 'Marketing Assistant' ; EXEC AddEmp 271, 272, 'Mary', 'Marketing Assistant' ;
Novamente, execute a seguinte consulta para examinar as linhas na tabela EmployeeOrg:
SELECT OrgNode.ToString() AS Text_OrgNode, OrgNode, OrgLevel, EmployeeID, EmpName, Title FROM HumanResources.EmployeeOrg ; GO
Aqui está o conjunto de resultados.
Text_OrgNode OrgNode OrgLevel EmployeeID EmpName Title ------------ ------- -------- ---------- ------- ----------------- / Ox 0 6 David Marketing Manager /1/ 0x58 1 46 Sariya Marketing Specialist /1/1/ 0x5AC0 2 269 Wanida Marketing Assistant /2/ 0x68 1 271 John Marketing Specialist /2/1/ 0x6AC0 2 272 Mary Marketing Assistant /3/ 0x78 1 119 Jill Marketing Specialist
Agora a tabela está totalmente populada com a organização de marketing.