0549 - Epidemie

From Bitnami MediaWiki

Cerința

Într-o țară locuiesc n persoane. Anumite perechi de persoane se cunosc între ele și se cunosc aceste perechi. Relația de cunoaștere între două persoane este reciprocă.

În țară izbucnește o epidemie (nu este mortală, doar foarte contagioasă). Dacă persoana A este bolnavă și cunoaște persoana B, se va îmbolnăvi și aceasta, după o perioadă de incubație a bolii de 1 zi. Inițial sunt bolnave k persoane cunoscute. Se cere să se determine după câte zile sunt bolnave toate cele n persoane.

Date de intrare

Fișierul de intrare epidemieIN.txt conține pe prima linie numerele n m, reprezentând numărul persoanelor și numărul perechilor de persoane care se cunosc reciproc. Urmează m linii cu câte două numere i j, cu semnificația: persoanele i și j se cunosc. Pe următoarea linie se află numărul k de persoane infectate inițial, iar pe ultima linie se află k numere distincte cuprinse între 1 și n, reprezentând aceste persoane.

Date de ieșire

Fișierul de ieșire epidemieOUT.txt va conține pe prima linie numărul Z, reprezentând numărul de zile după care toate cele n persoane sunt infectate.

Restricții și precizări

  • 1 ≤ n ≤ 1000
  • 1 ≤ m ≤ n*(n-1)/2
  • 1 ≤ i , j ≤ n
  • 1 ≤ k ≤ n
  • ziua inițială va fi luată în considerare
  • graful asociat acestei populații este conex

Exemplul 1

epidemieIN.txt

10 12
1 2
1 7
1 10
2 4
2 5
2 6
3 4
4 5
5 6
7 8
8 10
9 10
2
4 8

epidemieOUT.txt

3

Explicație

În prima zi se îmbolnăvesc persoanele: 4 8. În ziua a doua se infectează 2 3 5 7 10. În ziua a treia se infectează 1 6 9.

Exemplul 2

epidemieIN.txt

4 5
1 2
2 3
3 4
4 1
1 3
2
1 5

epidemieOUT.txt

Datele nu corespund restrictiilor impuse

Rezolvare

<syntaxhighlight lang="python3" line="1"> from collections import defaultdict, deque

def citeste_input(file_path):

   with open(file_path, 'r') as file:
       n, m = map(int, file.readline().split())
       cunostinte = defaultdict(list)
       for _ in range(m):
           i, j = map(int, file.readline().split())
           cunostinte[i].append(j)
           cunostinte[j].append(i)
       k = int(file.readline())
       persoane_infectate = set(map(int, file.readline().split()))
   return n, m, cunostinte, k, persoane_infectate

def verifica_restrictii(n, m, cunostinte, k, persoane_infectate):

   if not (1 <= n <= 1000) or not (1 <= m <= n*(n-1)//2) or not (1 <= k <= n):
       return False
   for i, j in cunostinte.items():
       for vecin in j:
           if not (1 <= i <= n) or not (1 <= vecin <= n):
               return False
   if any(persoana <= 0 or persoana > n for persoana in persoane_infectate):
       return False
   return True

def rezolva(n, cunostinte, k, persoane_infectate):

   zile = 0
   infectate = set(persoane_infectate)
   queue = deque(persoane_infectate)
   while queue:
       zile += 1
       for _ in range(len(queue)):
           persoana = queue.popleft()
           for vecin in cunostinte[persoana]:
               if vecin not in infectate:
                   infectate.add(vecin)
                   queue.append(vecin)
   return zile

def scrie_output(file_path, zile):

   with open(file_path, 'w') as file:
       file.write(str(zile))

if __name__ == "__main__":

   input_file = "epidemieIN.txt"
   output_file = "epidemieOUT.txt"
   n, m, cunostinte, k, persoane_infectate = citeste_input(input_file)
   
   if verifica_restrictii(n, m, cunostinte, k, persoane_infectate):
       zile = rezolva(n, cunostinte, k, persoane_infectate)
       scrie_output(output_file, zile)
   else:
       with open(output_file, 'w') as file:
           file.write("Datele nu corespund restrictiilor impuse")

</syntaxhighlight>