土地がフェンスで完全に囲まれているかどうかを判断する


19

ブール値の2次元配列を想像してください。0は土地の長方形のプロット上の草の正方形を表し、1はフェンスを表します。

2D配列を入力として受け入れ、フェンスにぶつかることなく、北/東/西/南の動きのみを使用して、草のある領域から他の草の領域に移動できるかどうかを決定する関数を記述します。

配列内の草のいずれかの領域がフェンスで完全に囲まれている場合(配列内の草の他のすべての領域に到達するためにN / E / W / Sを移動できないことを意味します)、関数はfalseを返します。それ以外の場合は、trueを返します。

以下は入力として使用できる2つのサンプル配列です。ただし、関数はこれらだけでなく、ブール値の2D配列も処理できるはずです。

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

(should return true)

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

(should return false, since the middle 0 in the top row is fully enclosed)

最短の作業コードが勝ちます。1週間が経過したか、24時間以内に新しい投稿がなかった場合に勝者を選びます。


二項演算子も禁止できますか?私は思い愛する人々が思い付くだろうか見て。
ピエールアラード

これは、昨年(2012/2013シーズン)のUSACO問題と非常に似ていると思います。そこにアクセスすることができますいくつかの巨大なテストケースがあります...
apnorton

配列のサイズは常に5 * 5ですか?
ProgramFOX 14年

1
@ProgramFOX配列は、任意の高さ、幅にできると仮定します。そして、必ずブール値を出力してください。
jawns317 14年

1
3X3マト​​リックスについてはどうですか1 1 11 0 1; 1 1 1?中央に1つの草細胞があります。視覚的には、中央の芝生のセルはフェンスで完全に囲まれていますが、定義ではそうではありません。
エモリー14年

回答:


1

Matlab 45

input('');c=bwconncomp(~ans,4);c.NumObjects<2

1
@ jawns317なぜこれが受け入れられた答えなのかわかりません。これは機能ではありません。関数ではない他の唯一の答えは、stdinから受け入れます。これはそれさえしません。
ティムセギーン14年

1
標準入力をそのまま受け入れることができますinput('');c=bwconncomp(~ans,4);c.NumObjects<2。これにより、45文字になります。
デニスJaheruddin 14年

7

APL(39)

{∧/,⊃{⍺∨⊖⍵}/{⍵∨(∧\~⍵)∨⌽∧\⌽~⍵}¨s,⊖¨s←⊂⍵}

使用法:

      board1 board2
 0 0 0 0 0  0 1 0 1 0 
 0 1 0 0 0  0 1 1 0 0 
 0 1 1 1 1  0 0 0 0 0 
 0 0 0 0 0  0 0 0 1 0 
 0 0 0 1 1  1 1 1 1 0 
      {∧/,⊃{⍺∨⊖⍵}/{⍵∨(∧\~⍵)∨⌽∧\⌽~⍵}¨s,⊖¨s←⊂⍵} ¨ board1 board2
1 0

9
APLの利点は、ラインノイズに非常に似ているため、誰もそれが正しいことを確認したくないことです。
ティムセギーン14年

@Tim誰でも通訳をダウンロードして実行および確認できます。
ガレス14年

3
@Garethええ、コメントは頬に舌であることになっています。
ティムセギーン14年

@ティムああすみません。それを逃した。:-(
ガレス14年

4

Mathematica、60 58文字

f=Max@MorphologicalComponents[1-#,CornerNeighbors->1>2]<2&

使用法:

f[{{0, 0, 0, 0, 0}, {0, 1, 0, 0, 0}, {0, 1, 1, 1, 1}, {0, 0, 0, 0, 0}, {0, 0, 0, 1, 1}}]

本当

f[{{0, 1, 0, 1, 0}, {0, 1, 1, 0, 0}, {0, 0, 0, 0, 0}, {0, 0, 0, 1, 0}, {1, 1, 1, 1, 0}}]


2
同じ長さf=Max@WatershedComponents[Image@#,CornerNeighbors->1>2]<2&
ベリサリウス博士14年

3

ルビー、202 198 193

a=$<.read.split('
').map(&:split)
f=->x,y{a[x][y]=1
[[-1,0],[1,0],[0,-1],[0,1]].map{|o|d,e=x+o[0],y+o[1]
f[d,e]if a[d]&&a[d][e]==?0}}
f[i=a.index{|x|x.index ?0},a[i].index(?0)]
p !(a.join=~/0/)

フラッドフィルを実行してから、0が残っているかどうかを確認します。


くそー!最初に自分のコードをテストしていなかったら、もっと速くできただろう。;)
ティムセギーン14年

@Timしかし、それは間違っていたでしょう!:P
ドアノブ

3

PHP 147 202 177 165 149バイト

編集 gzipハックを実際のphpソリューションで打ち負かしました。

少し長い....テキスト文字列として入力、スペースなし、改行で区切られた行。csで塗りつぶし、ゼロが残っているかどうかを確認します。ループexpでは、必要な反復回数の大まかな上限として使用します。対称性を利用して、より少ないコードで重複したケースを処理します

function f($s){$r=strpos;$n=$r($s,'
');$s[$r($s,'0')]=c;for(;$i++<1<<$n;)$s=strrev(ereg_replace('0(.{'.$n.'})?c','c\1c',$s));return!1==$r($s,'0');}

以下に、テストされていないテストケースを示します。

<?php
$s1="00000
01000
01111
00000
00011";

$s2="01010
01100
00000
00010
11110";

function f($s){
    $n=strpos($s,"\n");
    $s[strpos($s,'0')]=c;
    for(;$i<strlen($s);++$i)
        $s=strrev(ereg_replace(
            '0(.{'.$n.'})?c',
            'c\1c'
            ,$s));
    return!1===strpos($s,'0');
}

var_dump(f($s1));
var_dump(f($s2));

3

Excel VBA、305 215バイト

はい、VBAハハが、問題のマトリックス性質は、Excelの実用的なソリューションは面白いかもしれませんsuggets(プラス誰かがすでに私の他の言語での回答を提出しています!)。明らかに、VBAは最も簡潔ではありませんが、合理的だと思います。

この洪水は任意の開始点からいっぱいになり、「草」が残っているかどうかをチェックします

Rは、問題で定義されているフェンスと草を表す1と0のワークシート範囲です。ボーナス、競技場は長方形である必要はなく、隣接している必要さえありません。

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

たとえば、Falseを返します。左側のゼロから右側のゼロに到達することはできません。不規則なフィールドはそれを壊しません。

Function F(R)
L R, R.Find(0)
F = Not IsNumeric(R.Find(0))
End Function

Sub L(R, S As Range)
If S Or IsEmpty(S) Then Exit Sub
S = 1
L R, S.Offset(1, 0)
L R, S.Offset(-1, 0)
L R, S.Offset(0, 1)
L R, S.Offset(0, -1)
End Sub

ゴルフに関するいくつかのメモ。

要件が1と0に関して反転している場合、一部の文字はトリミングできると思いますが、反転するだけの価値があるとは言えません。

VBAは、束に空白(a = b対a = b)を要求しますが、これは文字カウントを助けません。

Sは、範囲として明示的に宣言する必要があります。バリアントのままにすると、範囲ではなく範囲値に変わります。

洪水を分岐させるより良い方法でしょうか?N / E / S / Wを送信するために文字を保存するループを思い付くことができませんでした

編集:塗りつぶしのベースケースを再調整し、再帰を防止するのではなく、再帰後のベースケースにあるかどうかを確認することで、かなりトリムしました。


2

Python(219バイト)

間違いなく最短ではありませんが、ここでの最初の試みなので、誇りに思っています。

def f(n):
 m=[int(c) for c in n if c!='\n']
 for i in range(len(m)):
  if m[i]<1:m[i]=2;break
 g(m,n.find('\n'),i);return not 0in m
def g(n,w,i):
 for x in i-w,i-1,i+1,i+w:
  if 0<=x<len(n):
   if n[x]<1:n[x]=2;g(n,w,x)

入力は、行が改行(\ n)文字で区切られた0と1の文字列でなければなりません。

使用例:

>>> f("00000\n01000\n01111\n00000\n00011")
True
>>> f("01010\n01100\n00000\n00010\n11110")
False

最後の2つのifステートメントを1つに結合できます。andいくつかの文字を節約できると思います
Tim Seguine

タブレーターは8つのスペースとして使用できます。
コンラッドボロスキー14年

2

パイソン(196)

標準的な浸水。

g=raw_input()
m=g.find(' ')
g=g.replace(' ','')
V={}
def D(V,x):
 if V.get(x,0)or g[x]=='1':return
 V[x]=1;[D(V,x+i)for i in 1,-1,m,-m if 0<=x+i<len(g)]
D(V,g.find('0'))
print len(V)==g.count('0')

各行を単一のスペースで区切って、STDINを介してマトリックスを取得します。たとえば、「01010 01100 00000 00010 11110」。


2

Mathematica 53

f=Max@(Symbol@@Names@"I*`i*B*l")[Image[1-#],0,1>2]<2&

内部関数を呼び出します Image`MorphologicalOperationsDump`imageBinaryLabel、これはに似ていMorphologicalComponentsます。

f[{{0, 0, 0, 0, 0}, {0, 1, 0, 0, 0}, {0, 1, 1, 1, 1}, {0, 0, 0, 0, 0}, {0, 0, 0, 1, 1}}]
f[{{0, 1, 0, 1, 0}, {0, 1, 1, 0, 0}, {0, 0, 0, 0, 0}, {0, 0, 0, 1, 0}, {1, 1, 1, 1, 0}}]

本当


1

PHP(286文字)

あまりにも長い間、私はおそらく長い道のりを行きました。

function D($a){$x=count($a);$y=count($a[0]);for($i=0;$i<$x;$i++)$a[$i][-1]=$a[$i][$y]=1;for($j=0;$j<$y;$j++)$a[-1][$j]=$a[$x][$j]=1;for($i=0;$i<$x;$i++){for($j=0;$j<$y;$j++){if($a[$i][$j]!=1){if($a[$i][$j-1]==1&&$a[$i][$j+1]==1&&$a[$i-1][$j]==1&&$a[$i+1][$j]==1)return 0;}}}return 1;}

ゴルフ以外:

function D($a)
{
$x=count($a);
$y=count($a[0]);
for ($i=0;$i<$x;$i++)
    $a[$i][-1]=$a[$i][$y]=1;
for ($j=0;$j<$y;$j++)
    $a[-1][$j]=$a[$x][$j]=1;
for ($i=0;$i<$x;$i++)
{
    for ($j=0;$j<$y;$j++)
    {
        if ($a[$i][$j] != 1)
        {
            if ($a[$i][$j-1] == 1 && $a[$i][$j+1] == 1 && $a[$i-1][$j] == 1 && $a[$i+1][$j] == 1)
                return 0;
        }
    }
}
return 1;
}

これは正しくありません。1で囲まれた単一のゼロがないかどうかを確認するだけです。ゼロを遮断するより複雑な方法があります。
ティムセギーン14年

もちろん、あなたは正しいです。私はこれを洪水で埋めることなく解決する別の方法を探していました、私の検索は続くと思います!
ヴェレオス14年

これは単なるグラフの到達可能性の問題であり、この場合のフラッドフィルは、基本的に、明示的に到達可能性グラフを作成または表現することなく、floyd-warshallです。グラフを抽出して推移的な閉包を自分で行うこともできますが、もっと長くなると思います。
ティムセギーン14年

1

C#、235バイト

int[][]D;int w,h,n;bool Q(int x,int y){var r=x>=0&&x<w&&y>=0&&y<h&&(D[x][y]++==0);if(r){Q(x-1,y);Q(x+1,y);Q(x,y+1);Q(x,y-1);}return r;}
bool P(int[][]B){D=B;w=D[0].Length;h=D.Length; for(int i=0;i<w*h;i++)if(Q(i%w,i/w))n++;return n==1;}

ボード内のすべてのセルを塗りつぶそうとし、1つの塗りつぶしのみがtrueを返します。

bool R( int x, int y)
{
    var r = (x >= 0 && x < w && y >= 0 && y < h && D[x, y]++ == 0);
    if (r)
    {
        R(x-1, y);
        R(x+1, y);
        R(x, y+1);
        R(x, y-1);
    }
    return r;
}

public bool Do()
{
    D = Board1;
    w = D.GetLength(0);
    h = D.GetLength(1);
    for (int x = 0; x < w; x++) for (int y = 0; y< h; y++) if (R(x, y)) n++;
    return n == 1;
}

0

Python 2.X + 3.X:335文字

ゴルフ:

def f(n):
 x,y=0,0
 z=lambda x,y:y<len(n)and x<len(n[0])and n[x][y]!=1
 while not z(x,y):
  y+=1
  if y==len(n):
   y=0
   x+=1
  if x==len(n[0]):
   return False
 t=set([(x,y)])
 v=set()
 while t:
  (x,y)=t.pop()
  v|=set([(x,y)])
  if z(x+1,y):
   t|=set([(x+1, y)])
  if z(x,y+1):
   t|=set([(x, y+1)])
 return len(v)+sum(map(sum,n))==len(n)*len(n[0])

ゴルフをしていない:

def f(n):
    """In the following filed, starting from a 0: is it possible to
       get to every other 0?

        >>> f([[0,0,0,0,0],\
               [0,1,0,0,0],\
               [0,1,1,1,1],\
               [0,0,0,0,0],\
               [0,0,0,1,1]])
        True
        >>> f([[0,1,0,1,0],\
               [0,1,1,0,0],\
               [0,0,0,0,0],\
               [0,0,0,1,0],\
               [1,1,1,1,0]])
        False
        >>> f([[1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,1]])
        False
        >>> f([[1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,0,1,1,1],\
               [1,1,1,1,1]])
        True
        >>> f([[1,1,1,1,1],\
               [1,1,1,1,1],\
               [0,1,1,1,1],\
               [1,0,1,1,1],\
               [1,1,0,1,1]])
        False
        >>> f([[1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,1],\
               [1,1,1,1,0]])
        True
    """
    x, y = 0,0
    isValid = lambda x,y: y<len(n) and x<len(n[0]) and n[x][y] != 1
    for i in range(len(n)*len(n[0])):
        x = i%len(n)
        y = i/len(n)
        if isValid(x,y):
            break

    while not isValid(x,y):
        y += 1
        if y == len(n):
            y = 0
            x += 1
        if x == len(n[0]):
            return False # Problem is not clearly defined here
    toGo=set([(x,y)])
    visited=set()
    while toGo:
        (x,y) = toGo.pop()
        visited |= set([(x,y)])
        if isValid(x+1,y):
            toGo |= set([(x+1, y)])
        if isValid(x,y+1):
            toGo |= set([(x, y+1)])
    return len(visited)+sum(map(sum,n)) == len(n)*len(n[0])

if __name__ == "__main__":
    import doctest
    doctest.testmod()

ゴルフバージョンをトップに移動できますか?一部の人々は、このサイトのユーザースクリプトを使用して、コードの最初のブロックの文字をカウントしています。
ガレス14年

@Gareth:完了...
マーティン・トーマ
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.