Faire un peu de continent

11

Imaginons que nous ayons une matrice de bits (qui en contient au moins un 1):

0 1 0 1 1 0 1 0 0 1 0
0 1 0 1 0 0 1 0 1 1 0
0 0 1 0 1 1 0 1 0 1 0
1 1 0 0 1 0 0 1 1 0 1
0 0 0 1 0 1 1 0 0 1 0

Nous voulons définir certains des bits de cette matrice de telle sorte qu'elle forme une goutte contiguë de 1s, dans laquelle chacun 1est directement ou indirectement connecté les uns aux autres 1par un mouvement orthogonal:

0 1 1 1 1 1 1 0 0 1 0
0 1 0 1 0 0 1 0 1 1 0
0 1 1 0 1 1 1 1 0 1 0
1 1 0 0 1 0 0 1 1 1 1
0 0 0 1 1 1 1 0 0 1 0

(Vous pouvez le voir plus clairement en recherchant 1avec la fonction "trouver" de votre navigateur.)

Cependant, nous voulons également minimiser le nombre de bits que nous définissons.

La tâche

Étant donné une matrice (ou un tableau de tableaux) de bits ou de booléens, retournez le nombre minimum de bits qui doivent être définis pour créer un continent contigu de 1s. Il devrait être possible de passer d'un bit défini dans la matrice à un autre en se déplaçant uniquement dans une direction orthogonale vers d'autres bits définis.

Il s'agit de , donc la soumission valide la plus courte (mesurée en octets) l'emporte.

Cas de test

0 1 0 1 1 0 1 0 0 1 0
0 1 0 1 0 0 1 0 1 1 0
0 0 1 0 1 1 0 1 0 1 0
1 1 0 0 1 0 0 1 1 0 1
0 0 0 1 0 1 1 0 0 1 0
=> 6

1 0 0 0 0 0 1 0 0
1 1 0 0 1 1 1 0 0
1 1 1 0 1 1 1 1 1
0 1 0 0 0 0 0 0 0
0 0 0 0 0 1 1 1 1
0 1 0 0 0 0 1 1 0
1 0 0 0 0 0 1 0 0
=> 4

0 0 0 1 1 1 0 1 1
0 0 1 0 0 0 0 1 0
0 0 1 1 1 1 1 1 0
1 1 0 0 1 1 0 0 0
0 0 1 1 1 0 0 1 1
0 1 1 1 0 0 0 0 0
1 1 1 0 0 1 1 1 0
1 1 1 0 1 1 0 1 1
0 0 0 0 1 0 0 0 1
1 1 0 0 1 1 0 1 1
0 0 0 0 0 0 0 1 0
0 1 1 1 1 0 0 0 0
0 0 0 1 1 0 0 0 1
0 1 0 0 1 0 1 1 0
0 1 1 1 0 0 0 0 1
=> 8

1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
=> 0
Esolanging Fruit
la source
1
Cela nécessite un peu plus d'explication. Qu'est-ce qu'un "blob contigu" dans une matrice?
NoOneIsHere
11
Comme le problème est connu pour être NP-difficile, ce n'est pas un bon problème pour l' algorithme le plus rapide .
Peter Taylor
1
@Peter Taylor et esolangingfruit NP-
Hardness
1
À la lumière des commentaires de Peter Taylor et HyperNeutrino, et du fait que la question n'a actuellement aucune réponse, je change la méthode de notation en code-golf .
Esolanging Fruit
1
Que devons-nous faire s'il n'y en a pas 1dans la matrice?
Colera Su

Réponses:

1

C (gcc), 308 306 octets

La fonction freçoit (height, width, flattened array, pointer to ans)et renvoie une réponse par pointeur.

S'il n'y a pas 1dans la matrice, il reviendra 0.

#define v A[i]
N,M,K,R,C,T,i,*A;s(x,y){i=x*M+y;if(!(x<0|y<0|x>=N|y>=M|v^1))v=2,s(x,y+1),s(x,y-1),s(x+1,y),s(x-1,y);}g(i){if(C<R){if(i^K){g(i+1);if(!v)C+=v=1,g(i+1),v=0,C--;}else{T=1;for(i=0;i<K&&!v;i++);s(i/M,i%M);for(i=0;i<K;i++)T&=v^1,v=!!v;if(T)R=C;}}}f(n,m,a,b)int*a,*b;{K=R=(N=n)*(M=m),A=a;g(0);*b=R;}

Essayez-le en ligne!

Non golfé:

N,M,R,C,T,i,*A; // height, width, result, recursion depth

s(x,y)
{ // depth first search: replace all 1 in the same connected component with 2
    i=x*M+y;
    if(!(x<0|y<0|x>=N|y>=M|A[i]^1)) { // check if out of boundary
        A[i]=2;
        s(x, y+1),s(x, y-1),s(x+1, y),s(x-1, y);
    }
}

g(i)
{ // enumerate all posible solutions
    if(C<R) {
        if(i!=N*M) {
            g(i+1);      // nothing change for this entry
            if (!A[i]) { // set the entry to 1
                C++, A[i]=1;
                g(i+1);
                C--, A[i]=0;
            }
        }
        else {
            T=1;
            for (i=0; i<N*M && !A[i]; i++); // find first non-zero entry
            s(i/M, i%M);     // replace the connected component
            for (i=0; i<N*M; i++) {
                T&=A[i]!=1;   // check if no other components
                A[i]=!!A[i]; // change 2s back to 1
            }
            if (T) R=C;      // update answer
        }
    }
}

f(n,m,a,b)int*a,*b;{
    R=(N=n)*(M=m), A=a;
    g(0);
    *b=R;
}
Colera Su
la source
0

Python 2 , 611 octets

Un programme complet qui prend une liste de listes à travers l'entrée utilisateur. Les fonctions Iet dcompter le nombre d'îles dans le tableau. La boucle for à la fin énumère toutes les possibilités où vous pouvez changer 0s en 1s puis s'il reste un îlot stocke le nombre de 1s ajoutés à la liste C. Le minimum de cette liste est le nombre minimum de retournements de bits requis pour connecter les îles. C'est un algorithme très lent, donc il n'exécute pas les cas de test donnés en moins de 60 ans (je n'ai pas essayé plus longtemps) mais j'ai essayé quelques cas de test plus petits (~ 5x5) et il semble fonctionner correctement. J'ai obtenu l'algorithme de comptage des îles sur cette page.

from itertools import*
def d(g,i,j,v):
 v[i][j],R,C=1,[-1,1,0,0],[0,0,-1,1]
 for k in range(4):
	if len(g)>i+R[k]>=0<=j+C[k]<len(g[0]):
	 if v[i+R[k]][j+C[k]]<1and g[i+R[k]][j+C[k]]:v=d(g,i+R[k],j+C[k],v)
 return v
def I(g):
 w=len(g[0])
 v,c=[w*[0]for r in g],0
 for i in range(len(g)*w):
	if v[i/w][i%w]<1and g[i/w][i%w]>0:v=d(g,i/w,i%w,v);c+=1
 return c           
g=input()
C=[]
for p in [list(t)for t in product([0,1],repeat=sum(r.count(0)for r in g))]:
 h,G,x=0,[r[:]for r in g],len(g[0])
 for i in range(x*len(G)):
	if G[i/x][i%x]<1:h+=p[0];G[i/x][i%x]=p[0];del p[0]
 if I(G)<2:
	C.append(h)
print min(C)

Essayez-le en ligne!

Version prégolfée avant d'optimiser certaines choses:

from itertools import*
def d(g,i,j,v):
    v[i][j]=1
    R=[-1,1,0,0]
    C=[0,0,-1,1]
    for k in range(4):
        if len(g)>i+R[k]>=0<=j+C[k]<len(g[0]):
            if v[i+R[k]][j+C[k]]<1:
                if g[i+R[k]][j+C[k]]:
                    v=d(g,i+R[k],j+C[k],v)
    return v
def I(g):
    w=len(g[0])
    v=[[0]*w for r in g]
    c=0
    for i in range(len(g)):
        for j in range(w):
            if v[i][j]<1and g[i][j]>0:
                v=d(g,i,j,v)
                c+=1
    return c           
g=input()
z=sum(r.count(0)for r in g)
f=[list(t)for t in product('01',repeat=z)]
C=[]
for p in f:
    h=0
    G=[r[:]for r in g]
    x=len(G[0])
    for i in range(x*len(G)):
        exec('h+=int(p[0]);G[i/x][i%x]=int(p[0]);del p[0]'*(G[i/x][i%x]<1))
    if I(G)<2:
        C.append(h)
print min(C)
dylnan
la source