Де-Snakify Строка

58

Обычная строка выглядит так:

Hello,IAmAStringSnake!

И струнная змея выглядит примерно так:

Hel         
  l      rin
  o,IAmASt g
           S
       !ekan

Твое задание

Строковые змеи опасны, поэтому вы должны создать программу, которая принимает строковую змею в качестве входных данных и выводит ее как обычную строку.

Характеристики

  • Ввод может быть многострочной строкой или массивом строк.
  • Каждая строка ввода будет дополнена пробелами для формирования прямоугольной сетки.
  • Персонажи в змее могут подключаться только к смежным персонажам сверху, снизу, слева или справа от них (как в игре Snake). Они не могут идти по диагонали.
  • Персонажи-змеи никогда не будут соседствовать с другой частью змеи, только с подключенными персонажами.
  • Первый символ строки - это конечный символ с самым коротким манхэттенским расстоянием от верхнего левого угла входной сетки (т. Е. Минимальное количество ходов, которое потребуется змее, чтобы перейти непосредственно от конечного символа к верхнему левому углу). угол). Оба конца никогда не будут иметь одинаковое расстояние.
  • Строка может содержать любой символ ASCII между точками кода 33 и 126 включительно (без пробелов и переносов).
  • Длина строки будет от 2 до 100 символов.
  • Самый короткий код в байтах побеждает.

Тестовые случаи

(Входная сетка с последующей выходной строкой)

Hel         
  l      rin
  o,IAmASt g
           S
       !ekan

Hello,IAmAStringSnake!

----------

Python

Python

----------

P  ngPu  Code 
r  i  z  d  G 
o  m  z  n  o 
gram  lesA  lf

ProgrammingPuzzlesAndCodeGolf

----------

   ~ zyx tsr XWVUTSR
   }|{ wvu q Y     Q
!          p Z `ab P
"#$ 6789:; o [ _ c O
  % 5    < n \]^ d N
('& 432  = m     e M
)     1  > lkjihgf L
*+,-./0  ?         K
         @ABCDEFGHIJ

!"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}~

----------

  tSyrep    
  r    p    
  in Sli    
   g    Sile
   Snakes  n
Ser      ylt
a eh   ilS  
fe w   t    
   emo h    
     Sre    

SlipperyStringSnakesSilentlySlitherSomewhereSafe
user81655
источник
20
Интересный факт: Согласно связанной статье в Википедии, расстояние Манхэттена также известно как расстояние змеи!
user81655
Будет ли правильный ввод, такой как hastebin.com/asugoropin.vbs ?
FliiFe
@FliiFe Нет, части змеи не могут быть рядом друг с другом, потому что не всегда возможно сказать, куда змея идет, когда это происходит (и это усложнит задачу). Я добавил строку в спецификации, чтобы объяснить это.
user81655
Может ли ввод быть двумерным массивом (т.е. каждый символ как элемент)?
FliiFe
29
Конечно, эта головоломка нуждается в ответе на Python?
abligh

Ответы:

11

APL, 55 байт

{⍵[1↓(⊂0 0){1<⍴⍵:⍺,∆[⊃⍋+/¨|⍺-∆]∇∆←⍵~⍺⋄⍺}(,⍵≠' ')/,⍳⍴⍵]}

Эта функция принимает символьную матрицу со строкой змея в ней.

Пример:

      s1 s2 s3
┌────────────┬──────────────┬────────────────────┐
│Hel         │P  ngPu  Code │   ~ zyx tsr XWVUTSR│
│  l      rin│r  i  z  d  G │   }|{ wvu q Y     Q│
│  o,IAmAst g│o  m  z  n  o │!          p Z `ab P│
│           S│gram  lesA  lf│"#$ 6789;: o [ _ c O│
│       !ekan│              │  % 5    < n \]^ d N│
│            │              │('& 432  = m     e M│
│            │              │)     1  > lkjighf L│
│            │              │*+,-./0  ?         K│
│            │              │         @ABCDEFGHIJ│
└────────────┴──────────────┴────────────────────┘
      ↑ {⍵[1↓(⊂0 0){1<⍴⍵:⍺,∆[⊃⍋+/¨|⍺-∆]∇∆←⍵~⍺⋄⍺}(,⍵≠' ')/,⍳⍴⍵]} ¨ s1 s2 s3 
Hello,IAmAstringSnake!                                                                        
ProgrammingPuzzlesAndCodeGolf                                                                 
!"#$%&'()*+,-./0123456789;:<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefhgijklmnopqrstuvwxyz{|}~

Объяснение:

  • (,⍵≠' ')/,⍳⍴⍵: получить координаты всех не пробелов
  • (⊂0 0): начинаются с (0,0) (что является недопустимой координатой)
  • {... }: следуй за змеей, учитывая положение и змею:
    • 1<⍴⍵:: если осталось более 1 элемента:
      • ∆←⍵~⍺: удалить текущую позицию из змеи и сохранить ее в .
      • +/¨|⍺-∆: найти расстояние между текущей позицией и каждой точкой в ​​остальной части змеи
      • ∆[⊃⍋...] `: получить ближайшую точку на змее
      • : запустите функцию снова, с ближайшей точкой в ​​качестве новой текущей точки и укороченной змеей в качестве новой змеи.
      • ⍺,: добавить текущую позицию к результату этого
    • ⋄⍺: в противном случае просто верните текущую позицию
  • 1↓: удалить первый элемент из результата (это позиция (0,0))
  • ⍵[... ]: получить эти элементы из ⍵, в том порядке
Мэринус
источник
24

JavaScript (ES6) + SnakeEx , 176 байт

a=b=>snakeEx.run("s{A}:<+>([^ ]<P>)+",b).reduce((c,d)=>(e=c.marks.length-d.marks.length)>0?c:e?d:c.x+c.y>d.x+d.y?d:c).marks.reduce((c,d,e,f)=>e%2?c+b.split`\n`[d][f[e-1]]:c,"")

Помните SnakeEx? Хорошо, потому что я тоже! Предложения по игре в гольф приветствуются.

LegionMammal978
источник
19

MATL , 80 байт

Спасибо @LevelRiverSt за исправление

32>2#fJ*+X:4Mt1Y6Z+l=*2#fJ*+ttXjwYj+K#X<)wt!"tbb6#Yk2#)yw]xxvtXjwYjqGZy1)*+Gw)1e

Ввод осуществляется в виде двумерного массива char с разделенными строками ;. Тестовые случаи в этом формате

['Hel         ';'  l      rin';'  o,IAmASt g';'           S';'       !ekan']
['Python']
['P  ngPu  Code ';'r  i  z  d  G ';'o  m  z  n  o ';'gram  lesA  lf']
['   ~ zyx tsr XWVUTSR';'   }|{ wvu q Y     Q';'!          p Z `ab P';'"#$ 6789:; o [ _ c O';'  % 5    < n \]^ d N';'(''& 432  = m     e M';')     1  > lkjihgf L';'*+,-./0  ?         K';'         @ABCDEFGHIJ']
['  tSyrep    ';'  r    p    ';'  in Sli    ';'   g    Sile';'   Snakes  n';'Ser      ylt';'a eh   ilS  ';'fe w   t    ';'   emo h    ';'     Sre    ']

Попробуйте онлайн!

объяснение

Координаты каждого непространственного символа представлены комплексным числом. Для каждого текущего символа следующий получается как ближайший (минимальная абсолютная разница их комплексных координат).

Чтобы определить начальный символ, необходимо найти две конечные точки. Это делается следующим образом. Конечная точка - это непространственный символ, имеющий ровно одного непространственного соседа. Количество соседей получается путем двухмерной свертки с подходящей маской. Начальная точка - это конечная точка, комплексная координата которой имеет наименьшую сумму действительной и мнимой частей; т.е. на Манхэттене самое близкое расстояние к комплексному числу 0 или, что эквивалентно 1 + 1j, которое является комплексной координатой верхнего левого угла.

32>      % Take input as 2D char array. Compute 2D array with true for nonspace,
         % false for space
2#f      % Arrays of row and column indices of nonspaces
J*+      % Convert to complex array. Real part is row, imaginary part is column
X:       % Convert to column array
4Mt      % Push arrays of zeros and ones again. Duplicate
1Y6      % Push array [0 1 0; 1 0 1; 0 1 0]. Used as convolution mask to detect
         % neighbours that are nonspace
Z+       % 2D convolution with same size as the input
1=       % True for chars that have only one neighbour (endpoints)
*        % Multiply (logical and): we want nonspaces that are endpoints
2#fJ*+   % Find complex coordinates (as before)
ttXjwYj  % Duplicate. Take real and imaginary parts
+        % Add: gives Manhattan distance to (0,0)
K#X<     % Arg min. Entry with minimum absolute value has least Manhattan
         % distance to (0,0), and thus to (1,1) (top left corner)
)        % Apply as an index, to yield complex coordinate of initial endpoint
wt!      % Swap, duplicate, transpose.
         % The stack contains, bottom to top: complex coordinates of initial
         % endpoint, column array with all complex coordinates, row array of all
         % coordinates. The latter is used (and consumed) by the next "for"
         % statement to generate that many iterations
"        % For loop. Number of iterations is number of nonspaces
  tbb    %   Duplicate, bubble up twice (rearrange is stack)
  6#Yk   %   Find which of the remaining points is closest to current point. This
         %   is the next char in the string
  2#)    %   Remove the point with that index from the array of complex
         %   coordinates. Push that point and the rest of the array
  yw     %   Duplicate extracted point, swap
]        % End for
xx       % Delete top two elements of the stack
v        % Concatenate all stack contents into a column array. This array
         % contains the complex coordinates of chars sorted to form the string
tXjwYj   % Extract real part and imaginary part
GZy1)    % Number of rows of input. Needed to convert to linear index
*+       % Convert rows and columns to linear index
Gw       % Push input below that
)        % Index to get the original chars with the computed order
1e       % Force the result to be a row array (string). Implicitly display
Луис Мендо
источник
Я предполагаю, что объяснение ожидается?
Мэтт
@Matt Конечно ... позже в тот же день :-)
Луис Мендо
Добавлено объяснение @Matt
Луис Мендо
The initial point is the endpoint whose complex coordinate has the least absolute valueОсторожно: Евклидово расстояние! = Манхэттенское расстояние. например, точка 7 + 7j имеет евклидово расстояние 9,8994 и манхэттенское расстояние 14. 10j дальше по евклидову расстоянию, но значительно ближе по манхэттенскому расстоянию. Кроме этого, отличная концепция!
Уровень Река St
@LevelRiverSt Спасибо! Исправлено
Луис Мендо
14

C 198 190 179 180 181 байт

Редактировать: Использовал подсказку user81655 и удалил скобки в тройном операторе, спасибо! Я также изменил громоздкий (S & 1) тест на равномерность для более подходящего (и более короткого!) S% 2.

Edit2: Использование * стиля адресации в значительной степени сделало меня слепым к очевидным оптимизациям в определении S, то есть замене * (a + m) на [m] и т. Д. Затем я заменил сам S на T, что по существу делает половина того, что делает С. Код также теперь использует возвращаемое значение putchar.

Edit3: исправлена ​​ошибка, которая возникала с самого начала, критерии остановки поиска на Манхэттене a <b + m верны, только если a уже уменьшено. Это добавляет 2 байта, но один восстанавливается, сделав определение m глобальным.

Edit4: мой гольф прошел минимум и сейчас идет не в ту сторону. Еще одно исправление, связанное с поиском на Манхэттене. Первоначально у меня были встроенные проверки на месте, и без них поиск продолжается для больших входных массивов (где-то около 50x50) за пределами массива b. Следовательно, этот массив должен быть расширен как минимум вдвое по сравнению с предыдущим размером, что добавляет еще один байт.

#define T(x)+x*((a[x]>32)-(a[-x]>32))
m=103;main(){char b[m<<8],*e=b,*a=e+m;while(gets(e+=m));for(e=a;(T(1)T(m))%2**a<33;a=a<b+m?e+=m:a)a-=m-1;for(;*a;a+=T(1)T(m))*a-=putchar(*a);}

Разгромил и объяснил:

/* T(1)T(m) (formerly S) is used in two ways (we implicitly assume that each cell has
   one or two neighbors to begin with):
   1. (in the first for-loop) T(1)T(m) returns an odd (even) number if cell a has one (two)
      neighbors with ASCII value > 32. For this to work m must be odd.
   2. (in the second for-loop) at this stage each cell a in the array at which T(1)T(m) is
      evaluated has at most one neighboring cell with ASCII value > 32. T(1)T(m) returns the
      offset in memory to reach this cell from a or 0 if there is no such cell.
      Putting the + in front of x here saves one byte (one + here replaces two
      + in the main part)

#define T(x)+x*((a[x]>32)-(a[-x]>32))

  /* A snake of length 100 together with the newlines (replaced by 0:s in memory) fits
     an array of size 100x101, but to avoid having to perform out-of-bounds checks we
     want to have an empty line above and the array width amount of empty lines below
     the input array. Hence an array of size 202x101 would suffice. However, we save
     a (few) bytes if we express the array size as m<<8, and since m must be odd
     (see 1. above), we put m = 103. Here b and e point to the beginning of the (now)
     256x103 array and a points to the beginning of the input array therein */

m=103;
main()
{

  char b[m<<8],*e=b,*a=e+m;

  /* This reads the input array into the 256x103 array */

  while(gets(e+=m));

  /* Here we traverse the cells in the input array one
     constant-Manhattan-distance-from-top-left diagonal at a time starting at the top-left
     singleton. Each diagonal is traversed from bottom-left to top-right since the starting point
     (memory location e) is easily obtained by moving one line downwards for each diagonal
     (+m) and the stopping point is found by comparing the present location a to the input array
     starting position (b+m). The traversal is continued as long as the cell has either
     an ASCII value < 33 or it has two neighbors with ASCII value > 32 each (T(1)T(m)
     is even so that (T(1)T(m))%2=0).
     Note that the pointer e will for wide input arrays stray way outside (below) the
     input array itself, so that for a 100 cell wide (the maximum width) input array
     with only two occupied cells in the bottom-right corner, the starting cell
     will be discovered 98 lines below the bottom line of the input array.
     Note also that in these cases the traversal of the diagonals will pass through the
     right-hand side of the 103-wide array and enter on the left-hand side. This, however,
     is not a problem since the cells that the traversal then passes have a lower
     Manhattan distance and have thereby already been analyzed and found not to contain
     the starting cell. */

  for(e=a;(T(1)T(m))%2**a<33;a=a<b+m?e+=m:a)a-=m-1;

  /* We traverse the snake and output each character as we find them, beginning at the
     previously found starting point. Here we utilize the function T(1)T(m), which
     gives the offset to the next cell in the snake (or 0 if none), provided that
     the current cell has at most one neighbor. This is automatically true for the
     first cell in the snake, and to ensure it for the rest of the cells we put the
     ASCII value of the current cell to 0 (*a-=putchar(*a)), where we use the fact
     that putchar returns its argument. The value 0 is convenient, since it makes the
     stopping condition (offset = 0, we stay in place) easy to test for (*a == 0). */

  for(;*a;a+=T(1)T(m))
    *a-=putchar(*a);
}
Зунга
источник
1
Этот ответ великолепен.
abligh
+1 Это здорово. Я хотел бы добавить пояснение к своему ответу, показывающее, где я перебрал байты по сравнению с вашим решением, это нормально?
Миллибайт
mIllIbyte - не стесняйтесь добавлять комментарии, это путь к новым идеям.
Зунга
user81655 - спасибо за подсказку, сбрил 6 байт. Фактически, я попробовал это вчера, а также использовал S% 2 вместо (S & 1), который сбрил еще 2 байта, для теста на равномерность, но по какой-то причине (мой код был глючил где-то еще), тогда не работало надлежащим образом. Теперь, однако, все кажется в порядке.
Зунга
Очень хорошо. Сохраните еще немного с a[1]и a[-m]т. Д., И сделайте mглобальным - m=103;main().
Угорен
9

C 272 байта

#define E j+(p/2*i+1)*(p%2*2-1)
#define X(Y) a[Y]&&a[Y]-32
char A[999],*a=A+99;j,p,t;i;main(c){for(;gets(++j+a);j+=i)i=strlen(a+j);for(c=j;j--;){for(t=p=4;p--;)t-=X(E);t==3&&X(j)?c=c%i+c/i<j%i+j/i?c:j:0;}for(j=c;c;){putchar(a[j]),a[j]=0;for(c=0,p=4;!c*p--;)X(E)?c=j=E:0;}}

Посмотрите на источник @ Zunga. Теперь посмотри на мою. Хотите знать, как я получил дополнительные 91 байт?
Ungolfed:

#define E j+(p/2*i+1)*(p%2*2-1)
#define X(Y) a[Y]&&a[Y]-32  //can be more concise, see @Zunga's
  //and then doesn't need the define
char A[999],*a=A+99;j,p,t;i;
main(c){for(;gets(++j+a);j+=i)
i=strlen(a+j); //we don't need to know the length of a line
  //in @Zunga's solution, lines are spaced a constant distance apart
for(c=j;j--;){
for(t=p=4;p--;)t-=X(E);  //a ton of bytes can be saved with determining 
  //the neighbors, see @Zunga's source
t==3&&X(j)?
c=c%i+c/i<j%i+j/i?c:j:0;}  //we search for ends of the snake, 
  //and compute the Manhattan distance
for(j=c;c;){putchar(a[j]),a[j]=0;
for(c=0,p=4;!c*p--;)  //determining the neighbors again
X(E)?c=j=E:0;}}
mIllIbyte
источник
5

Python (2 и 3), 640 624 604 583 575 561 546 538 байт

Я все еще играю в гольф n00b, так что это немного больше.

Изменить: Спасибо @porglezomp за предложения! Я не удалил все операторы 'и', так как это сломало бы Python 3.

Edit2: Спасибо @Aleksi Torhamo за комментарий о isspace (). Результирующее сокращение компенсирует исправление, которое я вставил. Также спасибо anonymous за подсветку синтаксиса!

Edit3: Спасибо @ mbomb007 за несколько дополнительных байтов.

import sys;s=sys.stdin.read().split('\n');m={};q=[];o=len;j=o(s);r=range;g='!'
for y in r(j):
 v=s[y];f=o(v);d=y+1
 for x in r(f):
  n=[];a=n.append;U=s[y-1]
  if v[x]>=g:j>=y>0==(U[x]<g)<=x<o(U)and a((x,y-1));j>y>=0==(v[x-1]<g)<x<=f and a((x-1,y));j>y>-1<x+1<f>(v[x+1]<g)<1and a((x+1,y));j>d>-1<x<o(s[d])>(s[d][x]<g)<1and a((x,d));m[x,y]=[v[x],n];o(n)-1or q.append((x,y))
t=min(q,key=sum);w=sys.stdout.write;w(m[t][0]);c=m[t][1][0]
while o(m[c][1])>1:
 b=m[c][1];w(m[c][0])
 for k in r(o(b)):
  if b[k]!=t:t=c;c=b[k];break
print(m[c][0])

А вот и моя предварительная версия для гольфа

import sys

lines = sys.stdin.read().split('\n')
startend = []
mydict = {}
for y in range( 0, len(lines)):
  for x in range( 0, len(lines[y])):
    if not lines[y][x].isspace():
      neighbors = []
      if x>=0 and x<len(lines[y-1]) and y-1>=0 and y-1<len(lines):
        if not lines[y-1][x].isspace():
          neighbors.append( (x,y-1) )
      if x-1>=0 and x-1<len(lines[y]) and y>=0 and y<len(lines):
        if not lines[y][x-1].isspace():
          neighbors.append( (x-1,y) )
      if x+1>=0 and x+1<len(lines[y]) and y>=0 and y<len(lines):
        if not lines[y][x+1].isspace():
          neighbors.append( (x+1,y) )
      if x>=0 and x<len(lines[y+1]) and y+1>=0 and y+1<len(lines):
        if not lines[y+1][x].isspace():
          neighbors.append( (x,y+1) )
      mydict[(x,y)] = [ lines[y][x], neighbors ]

      if len( neighbors ) == 1:
        startend.append( (x,y) )

startend.sort( key=lambda x : x[0]*x[0] + x[1]*x[1] )

last = startend[0]
sys.stdout.write( mydict[ last ][0] )
current = mydict[last][1][0]
while len( mydict[current][1] ) > 1:
  sys.stdout.write( mydict[current][0] )
  for k in range( 0, len( mydict[current][1] ) ):
    if mydict[current][1][k] != last:
      last = current
      current = mydict[current][1][k]
      break

print(mydict[current][0])
ceilingcat
источник
1
Я сэкономил 12 байтов, введя S=lambda s:s.isspace()и затем сделав S(s)вместо s.isspace().
porglezomp
1
Я думаю, что вы также можете изменить все and на <, поскольку f() < g() < h()это то же самое, что и g = g(); f() < g and g < h()с точки зрения побочных эффектов (короткое замыкание цепей сравнения), и вы все равно игнорируете результат сравнений.
porglezomp
1
m[(x,y)]=такой же, как и более короткийm[x,y]=
porglezomp
2
@porglezomp: это еще короче сказатьS=str.isspace
Алекси Торхамо
1
Удаление Sи использование <'!'в каждом случае может иметь одинаковую длину, открывая возможность сэкономить больше. Изменение if 1-S(v[x]):к if(v[x]<'!')<1:, например. И, возможно, вы могли бы удалить некоторые скобки в последующих сравнениях, сделав это таким образом.
mbomb007
4

JavaScript (ES6), 195

Смотрите объяснение внутри тестового фрагмента.

s=>[...s].map((c,i)=>{if(c>' '&([n=-1,1,o=-~s.search('\n'),-o].map(d=>n+=s[i+d]>' '&&!!(e=d)),n<1)&m>(w=i/o+i%o|0))for(m=w,r=c,p=i+e;r+=s[i=p],[e,o/e,-o/e].some(d=>s[p=i+(e=d)]>' '););},m=1/0)&&r

Контрольная работа

f=s=>[...s].map((c,i)=>{if(c>' '&([n=-1,1,o=-~s.search('\n'),-o].map(d=>n+=s[i+d]>' '&&!!(e=d)),n<1)&m>(w=i/o+i%o|0))for(m=w,r=c,p=i+e;r+=s[i=p],[e,o/e,-o/e].some(d=>s[p=i+(e=d)]>' '););},m=1/0)&&r

// Less golfed

u=s=>(
  o = -~s.search('\n'), // offset between lines
  m = 1/0, // current min manhattan distance, init at infinity
  // scan input looking for the 2 ends of the string
  [...s].map((c,i)=>{ // for each char c at position i
     if(c > ' ' // check if part of the string
        & ( [-1,1,o,-o] // scan in 4 directions and count neighbors
             .map(d=> n+=s[i+d]>' '&&!!(e=d), n=0), // remember direction in e
          n < 2) // if at end of string will have exactly 1 neighbor
        & (w = i/o + i%o |0) < m) // manhattan distance in w, must be less than current min
       // found one of the ends, follow the path and build the string in r
       for(m = w, r = c, p = i+e; 
           r += s[i=p], 
           [e,o/e,-o/e] // check 3 directions, avoiding to go back
           .some(d=>s[p=i+(e=d)]>' '); // save candidate position and direction in p and e
          ); // empty for body, all the work is inside the condition
  }),
  r
)  

console.log=x=>O.textContent+=x+'\n'

;[`Hel         
  l      rin
  o,IAmASt g
           S
       !ekan`,
 `   ~ zyx tsr XWVUTSR
   }|{ wvu q Y     Q
!          p Z \`ab P
"#$ 6789:; o [ _ c O
  % 5    < n \\]^ d N
('& 432  = m     e M
)     1  > lkjihgf L
*+,-./0  ?         K
         @ABCDEFGHIJ`].forEach(t=>{
  console.log(t+'\n\n'+f(t)+'\n')
  })
<pre id=O></pre>

edc65
источник
Нужны ли точки с запятой ););}?
Сис Тиммерман
1
@CeesTimmerman да. Первый находится внутри forзаголовка, где нужны 2 двоеточия. Вторым является разделитель для forтела
edc65
3

Lua, 562 535 529 513 507 504 466 458 байт

Безусловно, мой самый большой гольф на данный момент, я думаю , что я все еще могу отрезать 100 байтов, к которым я буду стремиться, но разместив его как ответ, поскольку это уже заняло некоторое время :). Я был прав, я вырубил более 100 байтов! Я не думаю, что есть много возможностей для улучшения.

эта функция должна вызываться с двумерным массивом, содержащим один символ на ячейку.

Благодаря ему удалось сэкономить 40 байт при работе с @KennyLau !

Woohoo! До 500!

function f(m)t=2u=1i=1j=1s=" "::a::if s~=m[i][j]and(i<#m and m[i+1][j]~=s)~=(j<#m[i]and m[i][j+1]~=s)~=(i>1 and m[i-1][j]~=s)~=(j>1 and m[i][j-1]~=s)then goto b end
i,t=i%t+1,#m>t and t==i and t+1or t j=j>1 and j-1or u u=u<#m[1]and j==1 and u+1or u goto a::b::io.write(m[i][j])m[i][j]=s
i,j=i<#m and s~=m[i+1][j]and i+1or i>1 and s~=m[i-1][j]and i-1or i,j<#m[i]and s~=m[i][j+1]and j+1or j>1 and s~=m[i][j-1]and j-1or j
if s==m[i][j]then return end goto b end

Ungolfed

Пояснения придут, как только я закончу играть в гольф, на данный момент я одолжу вам читаемую версию этого исходного кода: D Вот и объяснения!

Изменить: не обновляется с последней модификацией, все еще игра в гольф перед обновлением. То же самое касается объяснений

function f(m)                    -- declare the function f which takes a matrix of characters
  t=2                            -- initialise the treshold for i
                                 -- when looking for the first end of the snake
  u=1                            -- same thing for j
  i,j=1,1                        -- initialise i and j,our position in the matrix
  s=" "                          -- shorthand for a space
  ::a::                          -- label a, start of an infinite loop
    if m[i][j]~=s                -- check if the current character isn't a space
      and(i<#m                   -- and weither it is surrounded by exactly
          and m[i+1][j]~=s)      -- 3 spaces or not
      ~=(j<#m[i]
          and m[i][j+1]~=s)      -- (more explanations below)
      ~=(i>1 
          and m[i-1][j]~=s)
      ~=(j>1
          and m[i][j-1]~=s)
      then goto b end            -- if it is, go to the label b, we found the head
    i,t=                         -- at the same time
      i%t+1,                     -- increment i
      #m>t and t==i and t+1or t  -- if we checked all chars in the current range, t++
    j=j>1 and j-1or u            -- decrement j
    u=u>#m[1]and j==1 and u+1or u-- if we checked all chars in the current range, u++
  goto a                         -- loop back to label a

  ::b::                          -- label b, start of infinite loop
  io.write(m[i][j])                    -- output the current char
    m[i][j]=s                    -- and set it to a space
    i,j=i<#m                     -- change i and j to find the next character in the snake
          and m[i+1][j]~=s       -- this nested ternary is also explained below
            and i+1              -- as it takes a lot of lines in comment ^^'
          or i>1 
            and m[i-1][j]~=s
            and i-1
          or i,
       j<#m[i] 
         and m[i][j+1]~=s
           and j+1
         or j>1 
           and m[i][j-1]~=s 
           and j-1
         or j
    if m[i][j]==s                -- if the new char is a space
    then                         -- it means we finished
      return                  -- exit properly to avoid infinite
    end                          -- printing of spaces
  goto b                         -- else, loop back to label b
end

Итак, вот некоторые подробные объяснения о том, как работает эта программа.

Прежде всего, давайте рассмотрим цикл с меткой a, он позволяет нам найти ближайший конец к верхнему левому углу. Это будет цикл навсегда, если нет конца, но это не проблема: D.

На сетке 4x4 здесь указаны расстояния между змеями (слева) и порядок их просмотра (справа)

1  2  3  4    |     1  2  4  7
2  3  4  5    |     3  5  8 11
3  4  5  6    |     6  9 12 14
4  5  6  7    |    10 13 15 16

Для каждого этого символа, чтобы быть в конце, он должен проверить два условия: - не быть пробелом - быть окруженным ровно 3 пробелами (или ровно 1 не пробел)

Эти условия проверяются следующим фрагментом кода

r=m[i][j]~=s
    and(i<#m and m[i+1][j]~=s)
    ==not(j<#m[i] and m[i][j+1]~=s)
    ==not(i-1>0 and m[i-1][j]~=s)
    ==not(j-1>0 and m[i][j-1]~=s)
    and m[i][j]
    or r
  -- special note: "==not" is used as an equivalent to xor
  -- as Lua doesn't know what is a xor...

Проверка, если символ не пробел, достигается выражением m[i][j]~=s.

Проверка того, что мы окружены только одним незаполненным пространством, достигается путем изменения указанных выше условий для нашего окружения, это можно записать как

m[i+1][j]~=" "  m[i][j+1]~=" "  m[i-1][j]~=" "  m[i][j-1]~=" "

И, наконец, если все вышеперечисленное оценено как true, троичный вернет то, что в последнем and-> m[i][j]. Остальное мы дадим rнеустановленным :)

Теперь, когда у нас есть голова змеи, давайте пройдем весь путь до другого конца! Итерации змеи в основном достигаются следующими вложенными троицами:

i,j=i<#m and m[i+1][j]~=s and i+1or i-1>0 and m[i-1][j]~=s and i-1or i,
    j<#m[i]and m[i][j+1]~=s and j+1or j-1>0 and m[i][j-1]~=s and j-1or j

Мы переустанавливаем iи jв то же время избегаем использования манекенов для хранения старых значений. Они оба имеют одинаковую структуру и используют простые условия, поэтому я представлю их в виде вложенных if, это должно позволить вам прочитать их легче. :)

i=i<#m and m[i+1][j]~=s and i+1or i-1>0 and m[i-1][j]~=s and i-1or i

Можно перевести на:

if(i<#m)
then
  if(m[i+1][j]~=" ")
  then
    i=i+1
  end
elseif(i-1>0)
then
  if(m[i-1][j]~=" ")
  then
    i=i-1
  end
end

Проверь это!

Вот код, который я использую для запуска, вы можете проверить его онлайн , скопировав его.

function f(m)t=2u=1i=1j=1s=" "::a::if s~=m[i][j]and(i<#m and m[i+1][j]~=s)~=(j<#m[i]and m[i][j+1]~=s)~=(i>1 and m[i-1][j]~=s)~=(j>1 and m[i][j-1]~=s)then goto b end
i,t=i%t+1,#m>t and t==i and t+1or t j=j>1 and j-1or u u=u<#m[1]and j==1 and u+1or u goto a::b::io.write(m[i][j])m[i][j]=s
i,j=i<#m and s~=m[i+1][j]and i+1or i>1 and s~=m[i-1][j]and i-1or i,j<#m[i]and s~=m[i][j+1]and j+1or j>1 and s~=m[i][j-1]and j-1or j
if s==m[i][j]then return end goto b end


test1={}
s1={
"  tSyrep    ",
"  r    p    ",
"  in Sli    ",
"   g    Sile",
"   Snakes  n",
"Ser      ylt",
"a eh   ilS  ",
"fe w   t    ",
"   emo h    ",
"     Sre    ",
     }
for i=1,#s1
do
  test1[i]={}
  s1[i]:gsub(".",function(c)test1[i][#test1[i]+1]=c end)
end
f(test1)
Katenkyo
источник
1
У меня есть слабость к более длинным ответам, которые практически не имеют выбора из-за выбора языка.
Мэтт
@Matt большое спасибо за поддержку! на самом деле, я все еще нахожу способы сыграть в эту игру, но все сложнее и сложнее!
Катенкё
2

Луа, 267 байт

Требуется Lua 5.3.

e=" "w=#arg[1]+1i=1/0q=0s=table.concat(arg,e)s=e:rep(#s)..s
m,n=i,{}for p in s:gmatch"()%g"do u=-1
for _,d in ipairs{-1,1,-w,w}do u=u+(s:find("^%S",d+p)or 0)end
n[p]=u+1m=math.min(m,p*i^(u//#s)+(p%w*w+p)*#s)end
p=m%#s repeat q,p=p,n[p]-q io.write(s:sub(q,q))until p<1

Использование:

$ lua desnakify.lua \
>    "  tSyrep    " \
>    "  r    p    " \
>    "  in Sli    " \
>    "   g    Sile" \
>    "   Snakes  n" \
>    "Ser      ylt" \
>    "a eh   ilS  " \
>    "fe w   t    " \
>    "   emo h    " \
>    "     Sre    "
SlipperyStringSnakesSilentlySlitherSomewhereSafe
Егор Скриптунов
источник
2

Python 3, 245 243 241 236 байт

sявляется входной строкой, nвыводится ли вывод на стандартный вывод:

f=s.find
w,l=f('\n')+1,len(s)
t=1,w,-1,-w
y=z=f(s.strip()[0]);n=s[y];v={y}
for i in range(l*l):
 i%=l;c=s[i]
 if c>' 'and i not in v:
  if i-y in t:y=i;n=c+n;v|={i}
  elif i-z in t:z=i;n+=c;v|={i}
if y%w+y//w>z%w+z//w:n=n[::-1]
print(n)

Редактировать: Спасибо @Cees Timmerman за сохранение 5 байтов!

pgks
источник
c>' 'andи print nв Python 2.
Сис Тиммерман
Вы не можете сделать ifвместо elif?
clismique
@ Qwerp-Derp, к сожалению, нет, я пробовал это раньше, но он печатает, например, "! EkanSgnirtSAmAI, olleHello, IAmAStringSnake!" и "SlipperyStSyreppilS".
PGN
Какой формат ввода?
clismique
@ Qwerp-Derp sпеременная - многострочная строка; последний символ строки должен быть символом новой строки (это необходимо для прохождения Pythonконтрольного примера)
pgks
1

Python, 537

Мое первоначальное решение:

from itertools import chain, product, ifilter
from operator import add
moves = ((1,0),(0,1),(-1,0),(0,-1))
h = dict(ifilter(lambda (p,v):not v.isspace(),chain(*map(lambda (i,r):map(lambda (j,c):((i,j),c),enumerate(r)),enumerate(s)))))
n = defaultdict(list)
for m,p in product(moves, h):
    np = tuple(map(add,m,p))
    if np in h:
        n[p].append(np)
def pr(nx):
    return(lambda l:(h[l[0]], h.pop(l[0]))[0] + pr(l[0]) if l else '')([x for x in n[nx] if x in h])
(lambda y: h[y]+ pr(y))(next(x for x in n if len(n[x])==1))

Немного сжал, но оставил как метод:

from itertools import chain, product
from operator import add
def unsnake(s):
    (h,n) = (dict(filter(lambda (p,v):not v.isspace(),chain(*map(lambda (i,r):map(lambda (j,c):((i,j),c),enumerate(r)),enumerate(s))))),defaultdict(list))
    for m,p in product(((1,0),(0,1),(-1,0),(0,-1)), h):(lambda np: n[p].append(np) if np in h else 0)(tuple(map(add,m,p)))
    def pr(nx):return(lambda l:(h[l[0]], h.pop(l[0]))[0] + pr(l[0]) if l else '')([x for x in n[nx] if x in h])
    return(lambda y: h[y]+ pr(y))(next(x for x in n if len(n[x])==1))
topkara
источник
1

Java 7, 927 924 923 байта

import java.util.*;int l,k;char[][]z;Set p=new HashSet();String c(String[]a){int x=0,y=0,n,t,u,v,w=t=u=v=-1;l=a.length;k=a[0].length();z=new char[l][k];for(String s:a){for(char c:s.toCharArray())z[x][y++]=c;}x++;y=0;}for(x=0;x<l;x++)for(y=0;y<k;y++){n=0;if(z[x][y]>32){if(x<1|(x>0&&z[x-1][y]<33))n++;if(y<1|(y>0&&z[x][y-1]<33))n++;if(x>l-2|(x<l-1&&z[x+1][y]<33))n++;if(y>k-2|(y<k-1&&z[x][y+1]<33))n++;}if(n>2&t<0){t=x;u=y;}if(n>2&t>v){v=x;w=y;}}if(v+w>t+u){p(t,u);return n(""+z[t][u],t,u);}p(v,w);return n(""+z[v][w],v,w);}String n(String r,int x,int y){int a,b;if(x>0&&z[a=x-1][b=y]>32&q(a,b)){p(a,b);return n(r+z[a][b],a,b);}if(y>0&&z[a=x][b=y-1]>32&q(a,b)){p(a,b);return n(r+z[a][b],a,b);}if(x<l-1&&z[a=x+1][b=y]>32&q(a,b)){p(a,b);return n(r+z[a][b],a,b);}if(y<k-1&&z[a=x][b=y+1]>32&q(a,b)){p(a,b);return n(r+z[a][b],a,b);}return r;}boolean q(int x,int y){return!p.contains(x+","+y);}void p(int x,int y){p.add(x+","+y);}

Хорошо, это заняло некоторое время. В некоторых языках программирования не имеет значения, находятся ли ваши массивы x и y за пределами 2D-массива, но с Java это вызовет ArrayIndexOutOfBoundsExceptions, так что все должно быть проверено ...

Сначала я определяю начальную точку, а затем использую рекурсивный метод для построения строки оттуда. Кроме того, я использую список, чтобы отслеживать координаты, с которыми я уже столкнулся, поэтому он не будет переходить в цикл «вперед-назад-назад» (в результате возникает исключение StackOverflowException).

Это, пожалуй, самый длинный ответ, который я написал до сих пор, но, хотя некоторые части можно сыграть в гольф, я не думаю, что эта проблема может быть намного короче в Java. Java просто не подходит для того, чтобы следовать по пути в сетке. Тем не менее, это было забавное испытание. :)

Ungolfed и тестовые случаи:

Попробуй это здесь.

import java.util.*;
class M{
  static int l,
             k;
  static char[][] z;
  static Set p = new HashSet();

  static String c(String[] a){
    int x=0,
        y=0,
        n,
        t,
        u,
        v,
        w = t = u = v = -1;
    l = a.length;
    k = a[0].length();
    z = new char[l][k];
    for(String s:a){
      for(char c:s.toCharArray()){
        z[x][y++] = c;
      }
      x++;
      y = 0;
    }
    for(x=0; x<l; x++){
      for(y=0; y<k; y++){
        n = 0;
        if(z[x][y] > 32){ // [x,y] is not a space
          if(x < 1 | (x > 0 && z[x-1][y] < 33)){
            n++;
          }
          if(y < 1 | (y > 0 && z[x][y-1] < 33)){
            n++;
          }
          if(x > l-2 | (x < l-1 && z[x+1][y] < 33)){
            n++;
          }
          if(y > k-2 | (y < k-1 && z[x][y+1] < 33)){
            n++;
          }
        }
        if(n > 2 & t < 0){
          t = x;
          u = y;
        }
        if(n > 2 & t > v){
          v = x;
          w = y;
        }
      }
    }
    if(v+w > t+u){
      p(t, u);
      return n(""+z[t][u], t, u);
    }
    p(v, w);
    return n(""+z[v][w], v, w);
  }

  static String n(String r, int x, int y){
    int a,b;
    if(x > 0 && z[a=x-1][b=y] > 32 & q(a,b)){
      p(a, b);
      return n(r+z[a][b], a, b);
    }
    if(y > 0 && z[a=x][b=y-1] > 32 & q(a,b)){
      p(a, b);
      return n(r+z[a][b], a, b);
    }
    if(x < l-1 && z[a=x+1][b=y] > 32 & q(a,b)){
      p(a, b);
      return n(r+z[a][b], a, b);
    }
    if(y < k-1 && z[a=x][b=y+1] > 32 & q(a, b)){
      p(a, b);
      return n(r+z[a][b], a, b);
    }
    return r;
  }

  static boolean q(int x, int y){
    return !p.contains(x+","+y);
  }

  static void p(int x, int y){
    p.add(x+","+y);
  }

  public static void main(String[] a){
    System.out.println(c(new String[]{ "Hel         ",
      "  l      rin",
      "  o,IAmASt g",
      "           S",
      "       !ekan" }));
    p = new HashSet();
    System.out.println(c(new String[]{ "Python" }));
    p = new HashSet();
    System.out.println(c(new String[]{ "P  ngPu  Code ",
      "r  i  z  d  G",
      "o  m  z  n  o",
      "gram  lesA  lf" }));
    p = new HashSet();
    System.out.println(c(new String[]{ "   ~ zyx tsr XWVUTSR",
      "   }|{ wvu q Y     Q",
      "!          p Z `ab P",
      "\"#$ 6789:; o [ _ c O",
      "  % 5    < n \\]^ d N",
      "('& 432  = m     e M",
      ")     1  > lkjihgf L",
      "*+,-./0  ?         K",
      "         @ABCDEFGHIJ" }));
    p = new HashSet();
    System.out.println(c(new String[]{ "  tSyrep    ",
      "  r    p   ",
      "  in Sli   ",
      "   g    Sile",
      "   Snakes  n",
      "Ser      ylt",
      "a eh   ilS ",
      "fe w   t   ",
      "   emo h   ",
      "     Sre    " }));
  }
}

Выход:

Hello,IAmAStringSnake!
Python
ProgrammingPuzzlesAndCodeGolf
!"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}~
SlipperyStringSnakesSilentlySlitherSomewhereSafe
Кевин Круйссен
источник
924 байта, иисус Христос ... LOL
Шон Уайлд
@BasicallyAlanTuring Хе-хе. Я просто выполнил задачу, выполнил код, а затем посмотрел на количество байтов. Было действительно намного выше, чем ожидалось, но, ну, по крайней мере, это ниже 1 КБ ... Если вы видите что-то для улучшения, дайте мне знать, и если у вас есть альтернативный подход с (намного) меньше байтов, не стесняйтесь сделать отдельный Почта; Мне было бы интересно увидеть это. PS: сейчас 923 байта. XD
Кевин Круйссен
Не нужно проверять все , просто добавьте несколько отступов к вашей строке. Вероятно, использование одной многострочной строки делает это проще. Взгляните на мой ответ C #, портированный из javascript
edc65
1

PHP, 199 184 182 байта

может еще иметь небольшой потенциал для игры в гольф

for($w=strpos($i=$argv[1],"
");;)for($y=++$n;$y--;)if($i[$p=$y*$w+$n-1]>" ")break 2;for($p-=$d=$w++;$d&&print$i[$p+=$e=$d];)foreach([-$w,-1,1,$w,0]as$d)if($d+$e&&" "<$i[$d+$p])break;

принимает ввод как многострочную строку из командной строки, ожидает перевод строки в стиле Linux.
Бегать php -r '<code>' '<string>'; избежать разрывов строк.

сломать

for(
    // find width
    $w=strpos($i=$argv[1],"\n")
    ;
    // find first character: initialize $p(osition)
    ;
)
    for($y=++$n             // increase distance
        ;$y--;)             // loop $y from (old)$n to 0
        if(" "<$i[$p=$y*$w+$n   // if character at $y*($width+1)+$x(=$d-$y) is no space
            -1                  // (adjust for the premature increment)
        ])
            break 2;                    // break loops

for(
    $p-=            // b) reverse the increment that follows in the pre-condition
    $d=             // a) initialize $d to anything!=0 to enable the first iteration
    $w++;           // c) increase $w for easier directions
    $d              // loop while direction is not 0 (cursor has moved)
    &&
    print$i[$p+=$e=$d]              // remember direction, move cursor, print character
    ;
)
    foreach([-$w,-1,1,$w,0]as$d)// loop through directions
        if($d+$e                    // if not opposite previous direction
            &&" "<$i[$d+$p]         // and character in that direction is not space
        )break;                     // break this loop
Titus
источник
1

C #, 310

(Изменить: исправление ошибки)

Функция с многострочным строковым параметром, возвращающая строку.

Включая запрошенный usingв подсчет байтов.

Это портирование моего ответа javascript.

using System.Linq;
string f(string s){int o=-~s.IndexOf('\n'),m=99;var r=new string(' ',o);(s=r+s+r).Select((c,i)=>{int n=2,e=0,p,w=i%o+i/o;if(c>' '&w<m&&new[]{-1,1,o,-o}.All(d=>(s[i+d]>' '?(e=d)*--n:n)>0))for(m=w,r=""+c+s[p=i+e];new[]{e,o/e,-o/e}.Any(d=>s[p+(e=d)]>' ');)r+=s[p+=e];return i;}).Max();return r;}

Тест на идеоне

С пробелами

    string f(string s)
    {
        int o = -~s.IndexOf('\n');
        var r = new string(' ', o);
        var m = 99;
        (s = r + s + r).Select((c, i) =>
        {
            int n = 2, e = 0, p, w = i % o + i / o;
            if (c > ' ' & w < m & new[] { -1, 1, o, -o }.All(d => (s[i + d] > ' ' ? (e = d) * --n : n) > 0))
                for (m = w, r = "" + c + s[p = i + e]; 
                     new[] { e, o / e, -o / e }.Any(d => s[p + (e = d)] > ' '); 
                     ) 
                   r += s[p += e];
            return i;
        }
        ).Max();
        return r;
    }
edc65
источник
1

Python 2, 251 байт

w=s.find('\n')+1;q=' ';p=q*w+'\n';s=list(p+s+p);d=-w,1,w,-1
def r(x):c=s[x];s[x]=q;v=[c+r(x+o)for o in d if s[x+o]>q];return v[0]if v else c
e=[x for x in range(len(s))if s[x]>q and sum([s[x+o]>q for o in d])<2]
print r(e[e[0]/w+e[0]%w>e[1]/w+e[1]%w])

Или, если вам нужны ведущие символы новой строки в ваших тестовых случаях, 257 байт:

w=s.find('\n',1);q=' ';p=q*-~w+'\n';s=list(p+s[1:]+p);d=-w,1,w,-1
def r(x):c=s[x];s[x]=q;v=[c+r(x+o)for o in d if s[x+o]>q];return v[0]if v else c
e=[x for x in range(len(s))if s[x]>q and sum([s[x+o]>q for o in d])<2]
print r(e[e[0]/w+e[0]%w>e[1]/w+e[1]%w])

Проходит все тестовые случаи.

s="""
  tSyrep    
  r    p    
  in Sli    
   g    Sile
   Snakes  n
Ser      ylt
a eh   ilS  
fe w   t    
   emo h    
     Sre    
"""

Результаты в:

SlipperyStringSnakesSilentlySlitherSomewhereSafe
Сис Тиммерман
источник
3
Я думаю, что вы можете заменить b.append(...)на b+=[...]и def n(x,y):return ...сn=lambda x,y:...
Акролит
1
Создайте переменную для ' '.
pacholik
1
и использовать ~-xвместо x-1, вам не придется использовать скобки.
pacholik
0

Japt -P , 106 байт

K=U·ÌÊÄ ç iU ¬mx T=[-KJ1K]
ËÊ*Tm+E è@gX
[]V£YÃf@gXÃrQ@WpQ Tm+Q kW fZ Ì}V£[XYuK YzK]ÃkÈv ÉÃñx v rÈ+Y*K
W£gX

Попробуйте онлайн!

Это ... хм ... мерзость

Распаковано и как это работает

K=UqR gJ l +1 ç iU q mx
  UqR gJ l +1            Split the input by newline, take last item's length +1
K=                       Assign to K
              ç iU       Generate a string of K spaces, and append to U
                   q mx  Split into chars, and trim whitespaces on each item
                         Implicit assign to U

T=[-KJ1K]  Assign an array [-K, -1, 1, K] to T (this represents 4-way movement)
           I could use implicit assignment, but then 4-argument function below is broken

UmDEF{Dl *Tm+E èXYZ{UgX
UmDEF{                   Map over the list of one- or zero-length strings...
      Dl *                 If the length is zero, return zero
          Tm+E             Add the index to each element of T
               èXYZ{UgX    Count truthy elements at these indices
                         The result is an array of 0(space/newline), 1(start/end), or 2(body)
                         Implicit assign to V

[]  Implicit assign to W

VmXYZ{Y} fXYZ{UgX} rQXYZ{WpQ Tm+Q kW fZ gJ }
VmXYZ{Y}                                      Map V into indices
         fXYZ{UgX}                            Filter the indices by truthiness of U's element
                   rQXYZ{                     Reduce on the indices... (Q=last item, Z=array)
                         WpQ                    Push Q to W
                             Tm+Q               Take 4-way movements from Q
                                  kW fZ gJ }    Exclude visited ones, take last one in Z

VmXYZ{[XYuK YzK]} kXYZ{Xv -1} ñx v rXYZ{X+Y*K  Starting point of reduce
VmXYZ{[XYuK YzK]}                              Convert elements of V to [elem, col, row]
                  kXYZ{Xv -1}                  Take the ones where elem(popped)=1
                              ñx v             Sort by row+col and take first one
                                   rXYZ{X+Y*K  Convert [row,col] back to the index

WmXYZ{UgX  Map indices back to chars

-P  Join with empty string

Примечательным моментом является то, что я использовал приоритет оператора между операторами присваивания и запятой в JS, чтобы упаковать несколько строк и сохранить возможность использования ярлыка @( XYZ{).

фонтанчик для питья
источник