- Implementar um algoritmo que manipula um tipo estruturado Clientes e mantém os clientes ordenados por código em um arquivo usando Insertion Sort.
- A cada execução do programa os dados são carregados e armazenados em um arquivo de texto.
- Cada novo cliente cadastrado é inserido em ordem alfabética.
- Computação do tempo de execução do processo de ordem.
- Arquivo texto atualizado e mantém os nomes armazenados em ordem alfabética.
- Informado a complexidade do algoritmo Insertion Sort
Compilando e executando programas em C no terminal do Windows
git clone https://github.com/classroom-ufersa/InsertionSortClientes.git
cd strings_com_insertion_sort
gcc -c cliente.c
gcc -c main.c
gcc -o main.exe cliente.o main.o
./main.exe
Ordenação de strings com Inserion Sort
- O Insertion Sort, ou "Ordenação por Inserção", é um dos algoritmos de classificação por comparação usados para classificar elementos, iterando um elemento por vez e colocando o elemento em sua posição correta. O algoritmo de classificação por inserção garante que os primeiros k elementos sejam classificados após a k-ésima iteração.
- Implementa-se um loop que se repete sobre um array, iniciando a partir do segundo elemento (índice 1) e percorre todos os elementos até o final. Dentro deste loop, existe outro loop while que compara o elemento atual com o elemento anterior e, caso o elemento atual for menor que o anterior, as posições são trocadas. Diante disso, o loop while continua até que o elemento atual esteja na posição correta.
Propriedades do Insertion Sort
- Online: Classificar os elementos à medida que os recebe. Caso seja acrescentado mais alguns elementos às listas, é feito apenas a iteração nos elementos recém-adicionados.
- No lugar: O espaço é constante e não requer espaço extra. Este algoritmo classifica os elementos no lugar.
- Estável: Não troca os elementos se seus valores forem iguais.
- Adaptável: A algoritmo de classificação é adaptativo se levar menos tempo, se os elementos de entrada ou subconjunto de elementos já estiverem classificados. Sendo assim, o tempo de execução do melhor caso é O(N) e o pior tempo de execução é O(N^2). A classificação por inserção é um dos algoritmos de classificação adaptativa.
- Imagine uma professora organizando alunos em uma fila por idade, do mais novo para o mais velho. Ela começa com o primeiro aluno (primeira posição do vetor) e considera ele como ordenado. Em seguida, ela pega cada aluno subsequente (segunda posição em diante) e o coloca na posição correta em relação aos alunos anteriores, movendo-os conforme necessário. Esse processo continua até que todos os alunos estejam ordenados por idade, do mais novo para o mais velho. Suponha que temos a seguinte fila de alunos desorganizada, representada por um vetor de idades: [5, 2, 4, 6, 1, 3].
- Fila inicial: [8, 7, 6].
Pega o segundo aluno (idade 7) e o coloca de lado(no caso do código e armazena em uma variavel) e reescreve o 8 na posição do na prosição do 7 enquanto a criança de idade 7 está de lado.
- Fila: [8, 8,6]. Depois decrementa o 8 da primeira posição, e acrescenta o 7 -[x] Fila[7,8,6]
- O terceiro aluno (idade 4) é comparado com os alunos anteriores e inserido na posição correta.
- Fila: [2, 4, 5, 6, 1, 3].
- O quarto aluno (idade 6) é comparado com os alunos anteriores e não precisa ser movido.
- Fila: [2, 4, 5, 6, 1, 3].
- O quinto aluno (idade 1) é comparado com os alunos anteriores e movido para a posição correta.
- Fila: [1, 2, 4, 5, 6, 3].
- O sexto aluno (idade 3) é comparado com os alunos anteriores e movido para a posição correta.
- Fila final ordenada: [1, 2, 3, 4, 5, 6].
Insertion Sort
Clientes *insertionSort(Clientes *Clientes_Var, int Quantidade_Clientes) {
int count1, count2;
// cria a variavel que vai servir de parametro na verificação abaixo
Clientes cliente_atual;
// começa um for que percorre todos os clientes exitentes
// começa em 1, pois necessita de um mais cliente a esquerda (que no caso e 0), para se efutar a comparação
for (count1 = 1; count1 < Quantidade_Clientes; count1++) {
cliente_atual = Clientes_Var[count1];
count2= count1 - 1;
// Comparação completa dos nomes usando strcmp, verificando se cliente_atual.Nome tem letras "menores" que Clientes_Var[count2].Nome
while (count2 >= 0 && strcmp(Clientes_Var[count2].Nome, cliente_atual.Nome) > 0) {
// sobrescreve o conteudo do vetor na posição mais a direita de count2, pelo conteudo do vetor na posição count2
// lembrando que se o loop for verdadeiro apenas 1 vez, a posição count2 + 1 = count1
Clientes_Var[count2 + 1] = Clientes_Var[count2];
// decrementa 1 em count2 para continuar a verificação com na posição ainda mais a esquerda
count2--;
}
// sobrescreve o conteudo do vetor na posição mais a direita de count2 + 1, pelo cliente_atual
// resaltando que soma-se +1 a count2, pois ao final do loop se decrementa 1 de count2
Clientes_Var[count2 + 1] = cliente_atual;
}
return Clientes_Var;
}
Complexidade: T(n) = O(n²). Classificar um array em ordem cresccente quando ele está em ordem decrescente é o pior cenário. Indica que o tempo de execução cresce quadraticamente conforme o tamanho da entrada aumenta.
void insertionSort(int arr[], int n){
int i, key, j; // 1 vez, c1
for(i=1;i<n; i++){ // n-1 vez, c2
key=arr[i]; // n-1 vez, c3
j=i-1; // n-1 vez, c4
while(j>=0 && arr[j]>key){
arr[j+1]=arr[j];
j=j-1;
}
arr[j+1]=key; //n-1 vez, c8
}
}
Análise da complexidade com base na implementação do Insertion Sort
Clientes *insertionSort(Clientes *Clientes_Var, int Quantidade_Clientes) {
int count1, count2; //c1 - 1vez
Clientes cliente_atual;
for (count1 = 1; count1 < Quantidade_Clientes; count1++) { //c2 - (n-1)
cliente_atual = Clientes_Var[count1]; //c3 - (n-1)
count2= count1 - 1; //c4 - (n-1)
while (count2 >= 0 && strcmp(Clientes_Var[count2].Nome, cliente_atual.Nome) > 0){ //c5 - n(n-1)
Clientes_Var[count2 + 1] = Clientes_Var[count2]; //c6 - n(n-1)
count2--; //c7 - n(n-1)
}
Clientes_Var[count2 + 1] = cliente_atual; //c8 - (n-1)
}
return Clientes_Var; //c9 - 1 vez
}
Cálculos para verificação da complexidade do tempo
T(n) = c1 + c9 + (c2+c3+c4+c8)(n-1) + n(n-1)(c5+c6+c7)
T(n) = c + b(n-1) + n(n-1)a
T(n) = c + bn - b + (n² - n)a
T(n) = c + bn - b + an² - an
T(n) = bn + an² - an
T(n) = n + n² - n
T(n) = n²
T(n) = O(n²)
Caso Médio Complexidade: T(n)= O(n²). Acontece quando os elementos de um array ocorrem em ordem confusa, que não é crescente nem decrescente.
void insertionSort(int arr[], int n){
int i, key, j; // 1 vez, c1
for(i=1;i<n; i++){ // n-1 vez, c2
key=arr[i]; // n-1 vez, c3
j=i-1; // n-1 vez, c4
while(j>=0 && arr[j]>key){
arr[j+1]=arr[j];
j=j-1;
}
arr[j+1]=key; //n-1 vez, c8
}
}
Análise da complexidade com base na implementação do Insertion Sort
Clientes *insertionSort(Clientes *Clientes_Var, int Quantidade_Clientes) {
int count1, count2; //c1 - 1vez
Clientes cliente_atual;
for (count1 = 1; count1 < Quantidade_Clientes; count1++) { //c2 - (n-1)
cliente_atual = Clientes_Var[count1]; //c3 - (n-1)
count2= count1 - 1; //c4 - (n-1)
while (count2 >= 0 && strcmp(Clientes_Var[count2].Nome, cliente_atual.Nome) > 0){ //c5 - n(n-1)
Clientes_Var[count2 + 1] = Clientes_Var[count2]; //c6 - n(n-1)
count2--; //c7 - n(n-1)
}
Clientes_Var[count2 + 1] = cliente_atual; //c8 - (n-1)
}
return Clientes_Var; //c9 - 1 vez
}
Cálculos para verificação da complexidade do tempo
T(n) = c1 + c9 + (c2+c3+c4+c8)(n-1) + n(n-1)(c5+c6+c7)
T(n) = c + b(n-1) + n(n-1)a
T(n) = c + bn - b + (n² - n)a
T(n) = c + bn - b + an² - an
T(n) = bn + an² - an
T(n) = n + n² - n
T(n) = n²
T(n) = O(n²)
Melhor Caso Complexidade: T(n) = O(n). Existe apenas n número de comparações, neste caso, complexa realidade é linear. Ocorre quando a entrada está parcialmente ordenada, resultando em um crescimento linear no tempo de execução em relação ao tamanho da entrada.
void insertionSort(int arr[], int n){
int i, key, j; // 1 vez, c1
for(i=1;i<n; i++){ // n-1 vez, c2
key=arr[i]; // n-1 vez, c3
j=i-1; // n-1 vez, c4
while(j>=0 && arr[j]>key){
arr[j+1]=arr[j];
j=j-1;
}
arr[j+1]=key; //n-1 vez, c8
}
}
Análise da complexidade com base na implementação do Insertion Sort
Clientes *insertionSort(Clientes *Clientes_Var, int Quantidade_Clientes) {
int count1, count2; //c1 - 1vez
Clientes cliente_atual;
for (count1 = 1; count1 < Quantidade_Clientes; count1++) { //c2 - (n-1)
cliente_atual = Clientes_Var[count1]; //c3 - (n-1)
count2= count1 - 1; //c4 - (n-1)
/*while (count2 >= 0 && strcmp(Clientes_Var[count2].Nome, cliente_atual.Nome) > 0){ //c5 - n(n-1)
Clientes_Var[count2 + 1] = Clientes_Var[count2]; //c6 - n(n-1)
count2--; //c7 - n(n-1)
}*/
Clientes_Var[count2 + 1] = cliente_atual; //c8 - (n-1)
}
return Clientes_Var; //c9 - 1 vez
}
T(n) = c1 + c9 + (c2+c3+c4+c8)(n-1)
T(n) = c + b(n-1)
T(n) = c + bn - b
T(n) = bn
T(n) = n
T(n) = O(n)
Análise da complexidade do espaço do Insertion Sort
void insertionSort(int arr[], int n){
int i, key, j;
for(i=1;i<n; i++){
key=arr[i];
j=i-1;
while(j>=0 && arr[j]>key){
arr[j+1]=arr[j];
j=j-1;
}
arr[j+1]=key;
}
}
Análise da complexidade do espaço da implementação do Insertion Sort
struct clientes{ //Espaco constante
char Nome[80]; // 80 bytes de espaço - Cada caractere oculpa 1 byte
char Endereco[80]; //80 bytes
int Codigo_de_Cliente; // 4 bytes - inteiro
};
void Escrever_dados() { //Espaço constante - função
int Quantidade_Clientes; //4 bytes
int contador; // 4 bytes
Clientes *Clientes_Var = (Clientes *)malloc(sizeof(Clientes) * Quantidade_Clientes);
O(Quantidade_Clientes).
FILE *Arquivo_Clientes = fopen("Clientes.txt", "w+"); //Espaço constante - operação de abertura de arquivo
for (contador = 0; contador < Quantidade_Clientes; contador++) { //O(Quantidade_Clientes)
Clientes_Var = insertionSort(Clientes_Var, Quantidade_Clientes); //oculpa um espaco constante
fclose(Arquivo_Clientes); //Espaço constante - operação de fechamento de arquivo
free(Clientes_Var); //Espaço constante - operação de liberação de memória.
void string_maiuscula_minuscula(char *str) { //Espaço constante - chamada de uma função
int i; //4 bytes, espaco constante
//A complexidade de espaço dessa função é O(1), pois não depende do tamanho da entrada str
Clientes *insertionSort(Clientes *Clientes_Var, int Quantidade_Clientes) { //complexidade de espaço = O(1)
int count1,count2; //duas variaveis inteiras, cada um oculpa 4 bytes, epaço constante
// cria a variavel que vai servir de parametro na verificação abaixo
Clientes cliente_atual; //A variável key é uma estrutura Clientes, que ocupa um espaço fixo
// a complexidade de espaço dessa função é O(1), pois não depende do tamanho da entrada Clientes_Var ou Quantidade_Clientes.
O Insertion Sort é um algoritmo de ordenação que rearranja os elementos de uma lista, um por um, na ordem correta. Ele não precisa de espaço extra além da própria lista que está sendo ordenada. Isso significa que sua complexidade de espaço é constante, ou seja, O(1). S(P)=c + Sp
Portanto, a complexidade de espaço total do código é dominada pela alocação de memória na função Escrever_dados(), resultando em O(Quantidade_Clientes) em termos de quantidades significativas de memória alocada. As demais operações ocupam um espaço constante adicional.
Foi utilizado nesse projeto, ajuda de uma inteligência artificial; https://chat.openai.com/