Gerar ID’s sequenciais sem perder a sequência – c# sql-server entity-framework

Pergunta:


Por Identity não quero, pois ele quebra a sequência se alguma coisa acontecer. Max() + 1, acho muito vulnerável e lento, quando tiver muitos registros. Logo pergunto, há uma forma que alguém conheça, que posso gerar números sequenciais e se houver algum problema na geração, ele não dispara o número, como o identity faz? Dizem que existem algorítmos que fazem isso no Sql Server, mas não gostaria de usar triggers não. Uso Sql Server 2008, Entity 6 e C#. Será gerado pelo C# com Entity.

Autor da pergunta pnet

Resposta :

A melhor maneira de resolver isso é:

Não usando Identities.

Para isso você terá que abandonar as chaves sequenciais e partir pra chaves aleatórias. No Entity Framework, o tipo de campo é o Guid.

A chance de colisão de Guids (ou seja, de você gerar dois Guids iguais) é de uma em 2.600.000.000.000.000.000.000.

Para usar, declare a chave do seu Model da seguinte forma:

public Guid MeuModeloId { get; set; }

Antes de aplicar SaveChanges(), é necessário gerar um Guid para a sua propriedade Id:

meuObjeto.MeuModeloId = Guid.NewGuid();

O SQL Server trabalha perfeitamente com Guids, mas lá o tipo de dado é chamado de UNIQUEIDENTIFIER (leia mais em http://msdn.microsoft.com/pt-br/library/ms190348.aspx).

O recurso que você está procurando existe a partir do SQL Server 2012 e é chamado SEQUENCES.

As SEQUENCES são objetos usado para gerar valores de auto numeração, onde podemos definir os valores iniciais e seus incrementos. Eles são objetos independentes das tabelas e triggers e foram criados exatamente para contornar limitações do Identity, como o citado Max() + 1 por exemplo.

É possível criar uma SEQUENCE com o seguinte T-SQL

CREATE SEQUENCE dbo.NomeSeq AS INT MINVALUE 1 CYCLE;

Será criada uma sequencia do tipo INT iniciando do número 1, e ao chegar ao limite do tipo (2147483648 no caso de INT) do campo, a contagem será reiniciada a partir do CYCLE.

Determinar o valor seguinte da sequencia pode ser feito com o comando NEXT VALUE FOR

SELECT NEXT VALUE FOR dbo.NomeSeq;

Mais informações referentes a sintaxe e uso do SEQUENCE podem ser encontrados aqui.

Simplesmente crie uma tabela paralela de uma linha e um campo só para guardar seu incremento.

Assim, você dá um update nesta tabela com contador = contador + 1 toda vez que precisar de um novo ID, e o usa na transação seguinte.

O preço é a query extra para obter o ID, mas pode ser mais interessante que o MAX()+1, com a vantagem de não voltar o contador caso se delete o registro de número maior. (seria o único caso de “registro banguela”, mas é fácil controlar se a tabela auxiliar vai incrementar ou não).

Esta solução está entre o limiar de uma solução real e um reparo técnico 1, mas funciona bem se for aplicada corretamente.

1. gambiarra

Adendo: se você tiver index na coluna, não muda nada um MAX() com 1 ou com 10000000 registros. Não imagino que qualquer DB razoável iria verificar linha a linha um campo indexado para obter um simples MAX().

Criando um sequencial usando uma procedure

USE [DBBANCO]
GO

/****** Object:  StoredProcedure [dbo].[Gerador_Seq]  15:35:53 ******/

SET ANSI_NULLS ON
GO

SET QUOTED_IDENTIFIER ON
GO

Create procedure [dbo].[Gerador_Seq]  @Total int as

DECLARE @intFlag INT

SET @intFlag = 1

WHILE (@intFlag <= @Total )

Begin

    SET @intFlag = @intFlag + 1
    INSERT INTO dbo.TEMP_SEQ (my_name) values ('na') ;

    IF @intFlag = @Total
       begin 
         BREAK;
       end
end
  return ''



GO

/** O campo my_id int IDENTITY =>> Completa o Serviço **/

Fonte

Related Posts:

Qual a diferença entre AppCompatActivity e Activity? – android android-activity
Pergunta: Qual a diferença da AppCompatActivity para Activity ? A partir de qual versão a AppCompatActivity foi adicionada ao Android? Autor da pergunta Luhhh A diferença reside ...
Como abreviar palavras em PHP? – php string
Pergunta: Possuo informações comuns como nome de pessoas e endereços, e preciso que elas contenham no máximo 30 caracteres sem cortar palavras. Exemplo: 'Avenida Natalino João Brescansin' ...
Qual é a finalidade de um parêntese vazio numa declaração Lambda? – c# expressões-lambda característica-linguagem
Pergunta: Criei um exemplo de uma declaração Lambda sem argumentos, entretanto, estou com duvidas referente a omissão do parêntese vazio () na declaração. Veja o exemplo: class ...
Boas práticas para URI em API RESTful – api rest restful
Pergunta: Estou com dúvida em relação às URIs de alguns recursos da api que estou desenvolvendo. Tenho os recursos projetos e atividades com relação 1-N, ...
Dúvidas sobre a integração do MySQL com Java – java mysql netbeans
Pergunta: Estou criando um sistema no NetBeans, utilizando a linguagem Java e o banco de dados MySQL. Escrevi o seguinte código para realizar a conexão ...
Qual é a finalidade da pasta Model do framework Inphinit? – php inphinit
Pergunta: No Inphinit micro-framework existe a pasta Model que fica dentro da pasta application, e nela é onde ficam as classes, mas eu estou muito ...
Uso do ‘@’ em variáveis – javascript typescript coffeescript
Pergunta: Vejo em algumas linguagens que compilam para javascript, como TypeScript e CoffeeScript, o uso do @ em variáveis, como também, casos em que o ...
Qual tamanho máximo um arquivo JSON pode ter? – json arquivo
Pergunta: Vou dar um exemplo para conseguir explicar minha duvida: Preciso recuperar informação de imagens vindas de uma API, esse banco de imagens me retorna JSON's ...
O que é Teste de Regressão? – terminologia engenharia-de-software testes
Pergunta: Na matéria de Teste de Software o professor abordou um termo chamado Teste de Regressão, isto dentro da disciplina de teste de software. Sendo ...
O que é um construtor da linguagem? – php característica-linguagem
Pergunta: Em PHP, já li e ouvi várias vezes a respeito dos Construtores da Linguagem. Os casos que sempre ouvi falar deles foi em casos ...
Função intrínseca para converter numérico para string – cobol
Pergunta: Estou a tentar saber se existe alguma função intrínseca do COBOL para converter um data numérico para string sem precisar usar a cláusula REDEFINES: ( ...
Porque usar implements? – java android
Pergunta: Qual a diferença entre usar btn.setOnClickListener(new OnClickListener() { e public class MainActivity extends Activity implements OnClickListener{ Estive fazendo um curso de Android e meu professor falou que ...
O que é XHTML e quando deve ser usado? – html xml xhtml
Pergunta: O que eu sei é que o XHTML precisa ser XML válido. Isso implica, por exemplo, que todas as tags precisam ser fechadas. Por ...
Uma placa aceleradora de vídeo pode melhorar o desempenho não-gráfico? [fechada] – desempenho
Pergunta: Para desenvolver em Ruby on Rails, eu utilizo aqui uma máquina virtual do VirtualBox com Ubuntu Server 14.04 sem interface gráfica instalada. Recentemente descobri uma ...
Concat() VS Union() – c# .net
Pergunta: Qual a diferença entre Concat() e Union() ? Quando usar Concat() e quando usar Union() ? Somente pode ser usado em list ? ...

Deixe uma resposta

O seu endereço de email não será publicado. Campos obrigatórios marcados com *