Virus vs Antidotes code golf [закрыт]

12

Существует прямоугольный двумерный массив, содержащий вирусы, обозначенные «v», антидот1, обозначенный «а», и антидот2, обозначенный «b» (нет значений, отличных от «v», «a» и «b»).

Антидот1 может убивать соседние вирусы только в горизонтальном и вертикальном направлениях, но антидот2 может убивать соседние (если есть) вирусы в горизонтальном, вертикальном и диагональном направлениях.

Как только антидоты активируются, сколько вирусов останется в конце?

Примеры:

Входные данные:

vv
vv

Выход: 4

Входные данные:

av
vv

Выход: 1

Входные данные:

vvv
vbv
vvv

Выход: 0

Входные данные:

bvb
bav
vab
vvv
vvb
vvv
vvv
bva
vav

Выход: 3

Киара Дан
источник
2
Я ценю ваши правки @Grimy - тоже хороший вызов :)
pixma140
2
@KevinCruijssen, это не неправильной формы.
Киара Дан
4
Можем ли мы взять 3 (разных) произвольных значения вместо «v», «a» и «b»?
attinat
2
обертываются ли антивирусы (т. е. «a» в нижнем ряду уберет «v» в верхнем ряду)?
Брайан
2
@Kiara Dan Я бы посоветовал не допускать каких-либо 3-х различных значений, так как это удерживает некоторый символ в вызове, а также возможную принудительную хитрость с кодовыми точками букв.
lirtosiast

Ответы:

8

Python 3 , 135 байт

j=''.join
p='j(s)'+4*'.replace("%s%s","%s%s")'%(*'vbbbbvbbavacvaca',)
f=lambda x:j(eval(2*'(eval(p)for s in zip(*'+'x))))')).count('v')

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

-2 байта благодаря Кевину Круйссену

объяснение

Заменяет все «v» на «b», если находится рядом с «b». Далее, заменяет все «v» на «c», если находится рядом с «a». Вторая итерация с транспонированной версией массива очищает все вертикальные и диагональные вирусы. Наконец, он вернет оставшееся количество символов «v».


Как более читаемая рекурсивная функция (155 байт)

Jitse
источник
3
Вы можете удалить пространство после y>1else. Хороший подход, хотя. Сначала я не был уверен, как это соотносится с диагональю b, но, похоже, это работает очень хорошо из-за ваших замен. :) +1 от меня.
Кевин Круйссен
@KevinCruijssen Спасибо! Уход за диагоналями происходит путем замены «v» на «a», если они находятся рядом с «b». Во второй итерации соседние 'v' затем удаляются.
Джитсе
3
Для следующего входа, выход должен быть 3, но его возвращение 4: BVB БАВ VVV ВВБ VAB VVV VVV BVA вав
Киара Dan
1
@KevinCruijssen Нашел еще более короткий, но ваше предложение сохраняет еще один байт!
Джитсе
5

JavaScript (ES7), 108 байт

Вводит в виде матрицы символов.

f=m=>(g=(y,X,V)=>m.map(r=>r.map((v,x)=>V?v>f&V>'a'>(x-X)**2+y*y-2?r[x]=n--:0:v<f?g(-y,x,v):n++)|y++))(n=0)|n

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

Аналогично моему первоначальному ответу, но на V>'a'>(x-X)**2+y*y-2самом деле выполнение на 1 байт короче, чем использование шестнадцатеричного трюка, описанного ниже. ¯ \ _ (ツ) _ / ¯


JavaScript (ES7), 109 байт

Вводит в виде матрицы символов.

f=m=>(g=(y,X,V)=>m.map(r=>r.map((v,x)=>V?v>f&(x-X)**2+y*y<V-8?r[x]=n--:0:v<f?g(-y,x,'0x'+v):n++)|y++))(n=0)|n

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

Как?

A1=(x1,y1)A2=(x2,y2)

Q(A1,A2)=(x2x1)2+(y2y1)2

Учитывая целочисленные координаты, это выглядит следующим образом:

854585212541145212585458

Следовательно:

  • A1A2Q(A1,A2)<2
  • A1A2Q(A1,A2)<3

238

  • A16810=210
  • B16810=310

комментарии

f =                      // named function, because we use it to test if a character
                         // is below or above 'm'
m => (                   // m[] = input matrix
  g = (                  // g is a recursive function taking:
    y,                   //   y = offset between the reference row and the current row
    X,                   //   X = reference column
    V                    //   V = reference value, prefixed with '0x'
  ) =>                   //
    m.map(r =>           // for each row r[] in m[]:
      r.map((v, x) =>    //   for each value v at position x in r[]:
        V ?              //     if V is defined:
          v > f &        //       if v is equal to 'v'
          (x - X) ** 2 + //       and the quadrance between the reference point and
          y * y          //       the current point
          < V - 8 ?      //       is less than the reference value read as hexa minus 8:
            r[x] = n--   //         decrement n and invalidate the current cell
          :              //       else:
            0            //         do nothing
        :                //     else:
          v < f ?        //       if v is either 'a' or 'b':
            g(           //         do a recursive call:
              -y,        //           pass the opposite of y
              x,         //           pass x unchanged
              '0x' + v   //           pass v prefixed with '0x'
            )            //         end of recursive call
          :              //       else:
            n++          //         increment n
      ) | y++            //   end of inner map(); increment y
    )                    // end of outer map()
  )(n = 0)               // initial call to g with y = n = 0
  | n                    // return n
Arnauld
источник
спасибо @Arnauld, ты можешь также объяснить код?
Киара Дан
3
@KiaraDan Готово.
Арно
3

05AB1E , 33 30 29 байт

2F.•s¯}˜?•2ô€Â2ä`.:S¶¡øJ»}'v¢

Попробуйте онлайн или проверьте еще несколько тестов .

Порт @Jitse 's Python 3 ответа , так что обязательно проголосуйте за него!
-1 байт благодаря @Jitse .

Объяснение:

Унаследованная версия обладает тем преимуществом, что может выполнять сжатие / транспонирование списка строк, когда новой версии потребуется явное Sи J, поскольку она работает только со списками символов. Но новая версия по-прежнему на 3 байта короче благодаря использованию €Âв сочетании с более короткой сжатой строкой. В прежней версии будет хранить только последнее значение в стеке внутри карты, но в новой версии он будет хранить все значения в стеке внутри карты.

2F                  # Loop 2 times:
  .•s¯}˜?•          #  Push compressed string "vbvabbca"
   2ô               #  Split it into parts of size 2: ["vb","va","bb","ca"]
     €Â             #  Bifurcate (short for duplicate & reverse copy) each:
                    #   ["vb","bv","va","av","bb","bb","ca","ac"]
       2ä           #  Split it into two parts:
                    #   [["vb","bv","va","av"],["bb","bb","ca","ac"]]
         `          #  Push both those lists separated to the stack
          .:        #  Replace all strings once one by one in the (implicit) input-string
            S       #  Then split the entire modified input to a list of characters
             ¶¡     #  Split that list by newlines into sublists of characters
               ø    #  Zip/transpose; swapping rows/columns
                J   #  Join each inner character-list back together to a string again
                 »  #  And join it back together by newlines
}'v¢               '# After the loop: count how many "v" remain

Посмотрите эту подсказку 05AB1E (раздел Как сжимать строки, не являющуюся частью словаря? ), Чтобы понять, почему .•s¯}˜?•это так "vbvabbca".

Кевин Круйссен
источник
Вам не нужно bc=>, baесли вы подаете заявление bv=> baраньше av=> ac. Таким образом .•6øнãI•(сжатой формы `" bvavbaac ") достаточно, экономя 2 байта.
Grimmy
@Grimy Это приводит к 1 вместо 3 для последнего контрольного примера как бы то ни было.
Кевин Круйссен
1
@KevinCruijssen два шага замены вместо трех, кажется, делают трюк в конце концов
Jitse
@ Джитс Спасибо. Строка сжатия на 2 байта короче, но мне теперь нужно .:(заменить все один раз) вместо :(продолжайте заменять все, пока ее больше не будет). Тем не менее, -1. :) Спасибо, что дали мне знать.
Кевин Круйссен
2

Java 10, 211 209 байт

m->{int i=m.length,j,c=0,f,t,I,J;for(;i-->0;)for(j=m[i].length;j-->0;c+=m[i][j]>98?f/9:0)for(f=t=9;t-->0;)try{f-=m[I=i+t/3-1][J=j+t%3-1]==98||Math.abs(I-i+J-j)==1&m[I][J]<98?1:0;}catch(Exception e){}return c;}

Модификация моего ответа на вызов All the single eights .
-2 байта благодаря @ceilingcat .

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

Объяснение:

m->{                           // Method with char-matrix parameter & int return-type
  int i=m.length,              //  Amount of rows
      j,                       //  Amount of columns
      c=0,                     //  Virus-counter, starting at 0
      f,                       //  Flag-integer
      t,I,J;                   //  Temp-integers
  for(;i-->0;)                 //  Loop over the rows of the matrix
    for(j=m[i].length;j-->0    //   Inner loop over the columns
        ;c+=                   //     After every iteration: increase the counter by:
            m[i][j]>98         //      If the current cell contains a 'v'
             f/9               //      And the flag is 9:
                               //       Increase the counter by 1
            :0)                //      Else: leave the counter unchanged by adding 0
      for(f=t=9;               //    Reset the flag to 9
          t-->0;)              //    Loop `t` in the range (9, 0]:
         try{f-=               //     Decrease the flag by:
           m[I=i+t/3-1]        //      If `t` is 0, 1, or 2: Look at the previous row
                               //      Else-if `t` is 6, 7, or 8: Look at the next row
                               //      Else (`t` is 3, 4, or 5): Look at the current row
            [J=j+t%3-1]        //      If `t` is 0, 3, or 6: Look at the previous column
                               //      Else-if `t` is 2, 5, or 8: Look at the next column
                               //      Else (`t` is 1, 4, or 7): Look at the current column
            ==98               //      And if this cell contains a 'b'
            ||Math.abs(I-i+J-j)==1
                               //      Or if a vertical/horizontal adjacent cell
              &m[I][J]<98?     //      contains an 'a'
               1               //       Decrease the flag by 1
            :0;                //      Else: leave the flag unchanged by decreasing with 0
         }catch(Exception e){} //     Catch and ignore any ArrayIndexOutOfBoundsExceptions,
                               //     which is shorter than manual checks
  return c;}                   //  And finally return the virus-counter as result
Кевин Круйссен
источник
1

Древесный уголь , 39 байт

WS⊞υι≔⪫υ⸿θPθ≔⁰ηFθ«≧⁺›⁼ιv⁺№KMb№KVaηι»⎚Iη

Попробуйте онлайн! Ссылка на подробную версию кода. Объяснение:

WS⊞υι≔⪫υ⸿θPθ

Соедините входные строки с \rсимволами и выведите результат на холст.

≔⁰η

Очистить количество живых вирий.

Fθ«

Цикл по символам на входе.

≧⁺›⁼ιv⁺№KMb№KVaη

Если текущий символ является вирусом и нет соседних bs в каком-либо направлении или as ортогонально, то увеличивайте число живых virii.

ι»

Повторите со следующим персонажем.

⎚Iη

Очистите холст и распечатайте общее количество живых вирий.

Нил
источник
1

Perl ( -00lp), 82 байта

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

/.
/;$,="(|..{@-})";$;="(|.{@-,@+})";$_=s/(a$,|b$;)\Kv|v(?=$,a|$;b)/ /s?redo:y/v//

TIO

Науэль Фуйе
источник