Hamburguesquín

El éxito y expansión de la conocida marca de restaurantes de comida rápida Hamburguesquín no tiene precedentes. En poco tiempo ha llenado la ciudad con locales por todas partes. Ahora sus dueños se están planteando reducir el número de restaurantes, pero obviamente sin perder clientes.

Han detectado que por ejemplo en la calle Gran Vía hay opciones de eliminar algún restaurante. Para hacerlo con garantías, han estudiado el área de influencia de cada restaurante en esa calle, es decir, los puntos de la calle tales que si alguien con hambre se encontrara ahí acudiría a dicho restaurante. Formalmente, si x es la localización del restaurante en la calle (0 ≤ xL, donde L es la longitud de la calle), el área de influencia es el intervalo [x − r, x + r], donde r es el radio de influjo del restaurante (algo que depende de cada restaurante).

Ahora te han contratado para que, con esa información, les digas cuántos restaurantes como máximo podrían cerrar sin dejar de cubrir ningún punto de la calle. Por el mismo precio quieren también que les digas si hay puntos en la calle no cubiertos, porque en ese caso se plantearían mover algún restaurante de sitio.

Entrada

La entrada consiste en varios casos de prueba. La primera línea de cada caso contiene dos enteros: L es la longitud de la calle (1 ≤ L ≤ 109) y N es el número de restaurantes (0 ≤ N ≤ 200.000). A continuación aparecen N líneas, cada una con dos enteros: la posición xi de un restaurante (0 ≤ xiL) y su radio de influjo ri (0 < riL).

Salida

Para cada caso de prueba se escribirá una línea con el máximo número de restaurantes que se pueden cerrar de tal forma que todo punto de la calle siga bajo la influencia de algún restaurante no cerrado. En caso de que haya puntos de la calle no cubiertos por ningún restaurante, se escribirá -1 para indicar la situación (en vez del número de restaurantes a cerrar).

Entrada de ejemplo

10 4
5 3
3 3
9 2
8 2
5 2
1 1
4 1

Salida de ejemplo

2
-1

Solución propuesta

class Intervalo:

    def __init__(self, centro, radio):
        self.centro = centro
        self.radio = radio

    def extremo_izquierdo(self):
        return self.centro - self.radio

    def extremo_derecho(self):
        return self.centro + self.radio

    def __eq__(self, intervalo):
        return self.centro == intervalo.centro and self.radio == intervalo.radio

    def __ne__(self, intervalo):
        return self.centro != intervalo.centro and self.radio != intervalo.radio

    def __le__(self, intervalo):
        return self.extremo_izquierdo() <= intervalo.extremo_izquierdo()

    def __lt__(self, intervalo):
        return self.extremo_izquierdo() < intervalo.extremo_izquierdo()

    def __ge__(self, intervalo):
        return self.extremo_izquierdo() >= intervalo.extremo_izquierdo()

    def __gt__(self, intervalo):
        return self.extremo_izquierdo() > intervalo.extremo_izquierdo()

    def __str__(self):
        extremo_izquierdo = self.extremo_izquierdo()
        extremo_derecho = self.extremo_derecho()
        return f'[{extremo_izquierdo}, {extremo_derecho}]'

    def __repr__(self):
        return f'{self.__class__.__name__}({self.centro}, {self.radio})'


if __name__ == '__main__':
    soluciones = []
    longitud, restaurantes = [int(x) for x in input().split()]
    while longitud and restaurantes:
        lista_restaurantes = []
        for _ in range(restaurantes):
            centro, radio = [int(x) for x in input().split()]
            lista_restaurantes.append(Intervalo(centro, radio))
        lista_restaurantes.sort()
        if lista_restaurantes[0].extremo_izquierdo() > 1:
            soluciones.append(-1)
        else:
            eliminables = 0
            posición = 1
            while posición < len(lista_restaurantes) and \
                    lista_restaurantes[posición - 1].extremo_derecho() >= lista_restaurantes[posición].extremo_izquierdo():
                if posición < len(lista_restaurantes) - 1 and \
                        lista_restaurantes[posición].extremo_derecho() >= lista_restaurantes[posición + 1].extremo_izquierdo():
                    eliminables += 1
                posición += 1
            if posición == len(lista_restaurantes) and lista_restaurantes[posición - 1].extremo_derecho() >= longitud:
                soluciones.append(eliminables)
            else:
                soluciones.append(-1)
        longitud, restaurantes = [int(x) for x in input().split()]

    for solucion in soluciones:
        print(solucion)

Enlace del código

Enlace en aceptaelreto.com

Publicado el 1 de enero de 2018