OBI 2022 - Fase 3 - Programação Nível 1
Para se preparar para a OBI, confira o nosso Roteiro de Estudos, nele você vai encontrar um direcionamento completo de como estudar!
Para conferir a prova na íntegra, clique aqui.
Carro Elétrico
Escrito por Enzo Dantas
Conhecimento prévio necessário:
Parcial 1 (37 pontos):
Para essa subtask, devemos apenas calcular se é possível ir de uma cidade à outra usando o carro, ou seja, se a autonomia dos carros é maior ou igual à distância entre as duas cidades. A distância entre dois pontos é km, e devemos checar se . Para facilitar a compreensão, vamos nos referir a como se fosse e a como se fosse .
Parcial 2 (32 pontos):
(todas as cidades estão na mesma linha), e é garantido que as cidades são dadas em ordem crescente de (isto é, ).
Perceba que, ao simular imaginariamente um caso qualquer, vão se formar algumas "regiões" conectadas por carro.
Vamos chamar essas "regiões" de "componentes".
Perceba que, se a distância entre e é maior que A, eles fazem parte de componentes diferentes e é preciso usar o avião para conectá-las. Assim, para contar quantos aviões precisamos usar, vamos contar quantas vezes .
Parcial 3 (31 pontos):
Nenhuma restrição adicional
Nota: esse é um problema conhecido, no qual se deve contar o número de componentes conexas em um grafo.
Similarmente a subtask anterior, algumas componentes conectadas por carro vão se formar ao simular um caso qualquer.
Se temos N componentes, qual o número mínimo de aviões necessários para conectar todas elas?
Lema: o número mínimo de arestas necessárias para conectar N componentes = N-1
Imagine que temos uma componente só. Nesse caso, não precisamos de nenhuma nova aresta para conectar todas as componentes (1 componente, 0 arestas). Se adicionarmos mais uma componente, precisaremos adicionar uma aresta (2 componentes, 1 aresta), mais outra componente, mais outra aresta (3 componentes, 2 arestas), e assim em diante. Ou seja, para cada nova componente adicionada, precisaremos de mais uma aresta, então a subtração é constante. Como no caso inicial (componentes = 1) o número de arestas é , para qualquer número de componentes usaremos arestas.
Sendo assim, para construir o grafo e achar as componentes, vamos criar uma aresta entre todos os pares de pontos tal que a distância entre eles é menor ou igual a A. Feito isso, vamos realizar várias DFS's: vamos iterar pelos vértices e, se o vértice não tiver sido visitado, vamos adicionar 1 ao contador de componentes, vamos rodar uma DFS e vamos marcar todos os vértices da componente como visitado. Ao fim, devemos printar (quantidade de componentes - 1).
Complexidade:
Complexidade de memória: para cada vértice criaremos no máximo arestas. Como temos vértices, teremos arestas no total .
Complexidade de tempo: a complexidade de criar as arestas é , enquanto a complexidade da DFS é . A complexidade final é .
Clique aqui para conferir o código
Restaurante de Pizza
Escrito por Enzo Dantas
Conhecimento prévio necessário:
Para que a pizza caiba inteiramente na caixa, seu diametro deve ser menor ou igual a cada um dos lados da caixa. Além disso, para que todas as fatias sejam do mesmo tamanho, a soma de vários ângulos internos deve formar um círculo completo (uma pizza), ou seja, 360 deve ser um múltiplo do ângulo interno das fatias.
Clique aqui para conferir o código
Dígitos
Escrito por Vitor Veiga
Conhecimento prévio necessário:
Dada uma sequência de todos os dígitos presentes entre dois números quaisquer e , queremos encontrar o menor que satisfaz a sequência. A ideia para o problema será checar todos os possíveis , de até algarismos, até achar um termo inicial que satisfaz as restrições da sequência.
Para isso, utilizaremos as strings para nos auxiliar, pela maior facilidade em adicionar e remover algarismos. Começamos com duas strings e , que armazenam o número inicial e o próximo número que temos que achar para haver uma sequência válida, ou seja, , , , e assim por diante. Sempre que acharmos o , adicionamos uma unidade no número contido nele e continuamos nosso loop.
Para melhor entendimento da ideia, o passo a passo do primeiro caso teste:
e
O primeiro número inicial que testaremos será o :
(ok)
(ok)
(ok)
(absurdo)
Depois, testaremos :
(ok)
(absurdo)
Finalmente, testaremos :
(ok)
(ok)
fim da sequência
Note que caso continuássemos, também seria válida a sequência que inicia com o número , que só teria ele mesmo.
Segue o código:
Clique aqui para conferir o código
Pilha de Moedas
Escrito por Caique Paiva
Conteúdo Prévio Necessário:
Primeiro, vamos ordenar as pilhas por quantidade de moedas na ordem crescente. Por exemplo, a sequência de pilhas vai virar . Obviamente isso não vai mudar a resposta final, porém vai nos ajudar a achar a resposta.
Como foi spoilado nos conteúdos, vamos fazer programação dinâmica!! A ideia principal do problema é que os valores mais altos nunca vão ser alterados, pois isso é obviamente não optimal (Por exemplo, no exemplo , não faz sentido colocarmos uma moeda em um dos oitos, por que isso vai sempre deixar a quantidade de moedas diferente maior).
Com isso, podemos fazer a operação ser: Pegar um intervalo e fazer com que todos os valores sejam iguais a , e isso diminui a quantidade de valores distintos. Tal operação tem um custo de .
Certo, mas por que fizemos essa mudança de operação? A primeira vista parece ser inútil, mas o ponto todo é que nunca vamos fazer essa operação duas vezes no mesmo índice, o que nos ajuda muito a fazer a .
Com isso, seja a quantidade de operações que precisamos para o intervalo ter no máximo valores distintos. Então
E os casos bases vão ser e se .
Usando uma soma de prefixo para manter o valor de , o código vai rodar em , pois para calcular toda a dp leva de tempo, e a transição da dp é . Tal complexidade passa, pois .
Clique aqui para conferir o código
Fica de exercício para o leitor codar o problema com uma dp iterativa ao invés de recursiva.