Code 39 кодировщик штрих-кода

16

Напишите функцию или программу, которая кодирует строку в код 39 штрих-код формата , где каждый символ кодируется в виде пяти столбцов, разделенных четырьмя пробелами. Либо два из столбцов и один из промежутков являются широкими, а другие узкими (коды 10 * 4), или три из промежутков являются широкими, и ни один из столбцов не является (4 кода). Это дает 44 различных кода, из которых один является зарезервированным кодом, который используется для обозначения начала и конца кодированной строки.

Соревнование

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

1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ-. +/$%

Выход - строка, закодированная как строка штрих-кода. Узкий зазор и межсимвольные промежутки представляют собой единое пространство, а широкий промежуток - три пробела. Узкая полоса - это последовательность байтов UTF-8 для символа Unicode «Полный блок», █, т. Е. 0xE2 0x96 0x88Широкая полоса - три таких последовательности байтов / символов ( ███). Полный список кодов:

      Spaces
      0100 0010 0001 1000 1011 1101 1110 0111
Bars
00000                       +    /    $   %
10001   1    A    K    U
01001   2    B    L    V
11000   3    C    M    W
00101   4    D    N    X
10100   5    E    O    Y
01100   6    F    P    Z
00011   7    G    Q    -
10010   8    H    R    .
01010   9    I    S  space          1=wide
00110   0    J    T  start/end      0=narrow

Полосы и пробелы чередуются, начиная с бара, поэтому, например, Q

bar   0 0 0  1     1 
code  █ █ █ ███   ███
space  0 0 0    1

После кодирования всех символов строка разделяется с обоих концов █ █ ███ ███ █. Межсимвольный промежуток, один пробел, вставляется между каждой буквой. Ваше решение может выводить завершающие пробелы и завершающий перевод строки (в этом порядке).

Примеры

""     → "█   █ ███ ███ █ █   █ ███ ███ █"

"A"    → "█   █ ███ ███ █ ███ █ █   █ ███ █   █ ███ ███ █"

"C++"  → "█   █ ███ ███ █ ███ ███ █   █ █ █   █ █   █   █ █   █ █   █   █ █   █ ███ ███ █"

"GOLF" → "█   █ ███ ███ █ █ █ █   ███ ███ ███ █ ███ █   █ █ ███ █ █   ███ █ ███ ███   █ █ █   █ ███ ███ █"

Разрешены стандартные форматы ввода / вывода, а стандартные лазейки запрещены. Это , поэтому выигрывает самый короткий код в байтах!

Angs
источник
1
Можем ли мы использовать печатный символ ASCII (вы можете выбрать, какой из них разрешить) вместо █?
Эрик Outgolfer
Для такого языка, как BrainFuck, что считать «одним использованием»?
l4m2
1
@Angs Я имел в виду на выходе в основном, а не в коде. Несправедливо назначать штрафы за использование #символа, поскольку, например, "#"это не единственная причина, по которой он может использоваться в языке.
Эрик Outgolfer
@EriktheOutgolfer Я имел в виду строковые литералы и тому подобное, но, учитывая точку зрения l4m2, лучше было бы запретить это. В конце концов, есть ли языки, которые не могут выводить три байта?
Анг

Ответы:

7

JavaScript (ES6), 225 212 байт

Сохранено 4 байта благодаря @ l4m2

s=>`#${s}#`.replace(/./g,c=>'0202020202'.replace(/./g,(j,k)=>[C='█',C+C+C,' ','   '][(i="%+/$U1AKV2BLW3CMX4DNY5EOZ6FP-7GQ.8HR 9IS#0JT".indexOf(c),[170,257,260,5,272,17,20,320,65,68,80][i>>2]^2<<i%4*2)>>k&1|j]))

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

Как?

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

n = m XOR (2 << k)

с:

  m  | k: 0   2   4   6
-----+------------------      Examples:
 170 |    %   +   /   $
 257 |    U   1   A   K         'G' = 320 XOR (2 << 4) = 320 XOR 32 = 352
 260 |    V   2   B   L
   5 |    W   3   C   M         '+' = 170 XOR (2 << 2) = 170 XOR 8  = 162
 272 |    X   4   D   N
  17 |    Y   5   E   O
  20 |    Z   6   F   P
 320 |    -   7   G   Q
  65 |    .   8   H   R
  68 |   spc  9   I   S
  80 |    #   0   J   T
Arnauld
источник
s=>`#${s}#`.replace(/./g,c=>'0202020202'.replace(/./g,(j,k)=>[C='#',C+C+C,' ',' '][(i="%+/$U1AKV2BLW3CMX4DNY5EOZ6FP-7GQ.8HR 9IS#0JT".indexOf(c),[,257,260,5,272,17,20,320,65,68,80][i>>2]|(2<<i%4*2^(i<4)*170))>>k&1|j]))(221)
14 м2
@ l4m2 Кажется, это не так. Однако мне удалось найти исправление для 221 байта .
Эрик Outgolfer
@EriktheOutgolfer Я думаю, что это в основном то, что l4m2 представил с полным блоком вместо '#'. Или я что-то упустил?
Арно
@Arnauld Их оригинал не помещается должным образом.
Эрик Outgolfer
3

Красный , 452 445 байт

func[s][q: func[x y z][append/dup x y z]append insert s"*""*"b:
func[n][v: copy[]until[append v n % 2 * 2 + 1
1 > n: n / 2]q v 1 5 - length? v
reverse v]a: q":1234567890:ABCDEFGHIJ:KLMNOPQRST:UVWXYZ-. *"" "44
a/45: #"+"a/56: #"/"a/67: #"$"a/78: #"%"foreach t s[i: index? find a t
k: b pick[0 17 9 24 5 20 12 3 18 10 6]either 0 = j: i % 11[11][j]m:
b pick[8 4 2 16 22 26 28 14]i - 1 / 11 + 1 z: copy""repeat n 5[q z"█"k/(n)
q z" "m/(n)]prin z]]

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

Я попытаюсь играть в гольф дальше, но я не ожидаю многого от этого наивного решения.

Гален Иванов
источник
2

Java 10, 455 байт

s->{String r="",R=r;for(var c:("~"+s+"~").split(""))r+=r.format("%9s",Long.toString(new int[]{148,289,97,352,49,304,112,37,292,100,52,265,73,328,25,280,88,13,268,76,28,259,67,322,19,274,82,7,262,70,22,385,193,448,145,400,208,133,388,196,138,162,168,42}["~1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ-. +/$%".indexOf(c)],2)).replace(' ','0');for(int l=r.length(),i=0,c;i<l;R+=""+(i%9%2<1?c<49?"█":"███":c<49?" ":"   ")+(i%9>7&++i<l?" ":""))c=r.charAt(i);return R;}

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

Объяснение:

s->{                     // Method with String as both parameter and return-type
  String r="",           //  Temp-String, staring empty
         R=r;            //  Result-String, starting empty as well
  for(var c:("~"+s+"~")  //  Prepend and append "~" to the input
            .split(""))  //  And loop over each character
    r+=r.format("%9s",Long.toString( 
                         //   Convert the following integer to a 9-bit binary String:
       new int[]{148,289,97,352,49,304,112,37,292,100,52,
                 265,73,328,25,280,88,13,268,76,28,259,67,322,
                 19,274,82,7,262,70,22,385,193,448,145,400,208,
                 133,388,196,138,162,168,42}
                         //    Array containing all decimal values of the binary Strings
       ["~1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ-. +/$%".indexOf(c)]
                         //    with the index based on the current character
   ,2)).replace(' ','0');
                         //   (also part of the integer to 9-bit binary String conversion)
  for(int l=r.length(),i=0,c;i<l;
                         //  Loop `i` over the temp-String
      R+=                //    After every iteration: append the following to the result:
         ""+(i%9%2<1?    //     If `i` modulo-9 modulo-2 is 0:
              c<49?      //      And `c` is '0':
               "█"       //       Append a single block
              :          //      Else:
               "███"     //       Append three blocks
             :c<49?      //     Else-if `c` is '0':
              " "        //      Append a single space
             :           //     Else:
              "   ")     //      Append three spaces
         +(i%9>7         //     If `i` modulo-9 is 8,
           &++i<l?       //     and this is not the very last character
            " "          //      Append a space delimiter
           :             //     Else:
            ""))         //      Append nothing more
    c=r.charAt(i);       //   Set `c` to the current character in `r`
  return R;}             //  Return the result
Кевин Круйссен
источник
2

C (gcc) , 311 , 303 байта

(*P)()=printf;i;j;w;m[]={170,257,260,5,272,17,20,320,65,68,80};char*b=" \0\xE2\x96\x88",*t="%+/$U1AKV2BLW3CMX4DNY5EOZ6FP-7GQ.8HR 9IS#0JT",*c;p(x){j=256;w=2;while(j){P(b+w);if(x&1){P(b+w);P(b+w);}x/=2;j>>=1;w=2*!w;}P(" ");}f(char*_){p(82);for(c=_;*c;c++){i=strchr(t,*c)-t;p(m[i>>2]^(2<<(i&3)*2));}p(82);}

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

-8 благодаря потолку

Использует стратегию кодирования из ответа Арно. TIO Link включает в себя -wпереключатель и шаблон для удаления предупреждений, они не нужны и, следовательно, не включены в счет.

Помимо схемы кодирования, описанной Арно, еще одна хитрость заключается в том, чтобы поддерживать wпеременную как переключающуюся между 2 и 0 ( w=2*!w). Это позволяет мне выбирать между первой и второй строками b. Первый - это пробел, второй - закрашенный прямоугольник.

LambdaBeta
источник
умный, вы должны опубликовать это :)
LambdaBeta
2

C (gcc) , 241 239 227 213 207 байт

#define P printf(i%2?" ":"█")
i;p(x){for(i=9;i--;x/=2)P,x&1&&P+P;P;}f(char*_){p(82);for(char*t="%+/$U1AKV2BLW3CMX4DNY5EOZ6FP-7GQ.8HR 9IS#0JT";*_;p(L"ªāĄ\5Đ\21\24ŀADP"[i/4]^2<<i%4*2))i=index(t,*_++)-t;p(82);}

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

Основано на реализации @ LambdaBeta .

Немного меньше в гольф:

#define P printf(i%2?" ":"█")
i;
p(x){
 for(i=9;i--;x/=2)
  P,
  x&1&&
   P+
   P;
 P;
}
f(char*_){
 p(82);
 for(char*t="%+/$U1AKV2BLW3CMX4DNY5EOZ6FP-7GQ.8HR 9IS#0JT";*_;p(L"ªāĄ\5Đ\21\24ŀADP"[i/4]^2<<i%4*2))
  i=index(t,*_++)-t;
 p(82);
}
ceilingcat
источник
1

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

⪫EE⪫**S⌕⁺⁺⭆…αχ﹪⊕μχα-. *+/$%ι⪫E⁵⁺⎇μ× ∨⁼›ι³⁹⁼²﹪⁻÷ι∨›ι³⁹χμ⁴¦³ω×█∨›ι³⁹∨§↨℅§.6':+3<-59ι²⊕μ³ω 

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

⪫EE⪫**S...⪫E⁵⁺...ω 

Оберните входную строку в *s, а затем дважды сопоставьте ее, в результате чего результат объединяется с пробелами. Для второй карты существует дополнительная карта в неявном диапазоне0..4 , где две подстроки объединяются, и эти результаты затем объединяются с предопределенной пустой строковой константой.

⌕⁺⁺⭆…αχ﹪⊕μχα-. *+/$%ι

Для первой внутренней карты создайте строку, сформированную путем взятия увеличенных цифр, прописного алфавита и символов -. *+/$%, и найдите положение сопоставленного входного символа. Например, C++будет отображаться в [12, 40, 40].

⎇μ× ∨⁼›ι³⁹⁼²﹪⁻÷ι∨›ι³⁹χμ⁴¦³ω

Первая подстрока представляет пробелы перед столбцами. Перед первым столбцом ничего нет, но остальные столбцы зависят от положения сопоставленного входного символа: если он больше 39, то только одно место имеет один пробел, а если ему меньше 40, то только одно место имеет три пробелы, и позиция также преобразуется в столбец путем деления его на 10. Если столбец и индекс цикла отличаются на 2 (по модулю 4), то это нечетное место.

×█∨›ι³⁹∨§↨℅§.6':+3<-59ι²⊕μ³

Вторая подстрока представляет столбцы. Если позиция больше 39, то всегда есть один столбец, в противном случае позиция ищется в массиве битов, сопоставленных с символами. Например, если позиция равна 12, то она циклически индексируется для символа ', который находится 100111в двоичном формате, указывая широкие полосы в столбцах 1 и 2. (Ведущий 1игнорируется, он просто обеспечивает постоянный счетчик битов.)

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

Perl 5 , 244 байта

$l="{ ,   ,...,.........}"x9;@h{$",qw($ % + - . /),0..9,'',A..Z}=(<"$l">)[q{..=9...Z.>>...J=9..j.%r1...=).[.>=-..Ux?V.^=8.>.6o+ax_.=(..B=,..Yx6..b=(..#r'p.^...G.<=,..Sx5B.\=(.V.It..z:..-z.=<6.}=~s/./ord$&/gre=~/.{6}/g];s/^|./$h{$&} /g;$\=$h{''}

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

Содержит много непечатаемых и старших байтов, ссылка TIO обеспечивает xxdпредставление. Я надеялся, что это закончилось бы меньше, и я мог бы все еще быть в состоянии упаковать данные более эффективным способом, таким образом я посмотрю, как я иду. Это создает все перестановки " "," ","█","███"и затем отображает признаки списка в соответствующие символы.

Дом Гастингс
источник
1

Haskell , 275 270 байт

z=0:z
x!0=[]
x!n=mod n x:x!div n x
e c|Just a<-lookup c.zip"W3YZ56C$EF. 89*HM/ORU1QGNTDJ7%40LSBIP2-+XVKA".((++)<*>(reverse<$>))$take 9.(++z).(2!)<$>scanl(+)7(67!0x117CDBC49F9EEEF11C3A659CACB31236)=zipWith(\l c->c<$[1..2*l+1])(a++[0])(cycle"█ ")>>=id
f s='*':s++"*">>=e

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

Оператор, x!nкоторый вычисляет x-базовые цифры n, используется дважды для распаковки кодов. Коды сначала сжимаются в виде двоичных строк с широким = 1 и узким = 0 без учета цвета, например R↔10000110↔262. Эти числа затем сортируются и дифференцируются, чтобы получить числа в диапазоне [3,66], которые сжимаются с помощью обратного алгоритма двоичных чисел в виде0x117CDBC49F9EEEF11C3A659CACB31236 . Это содержит только половину кодов, остальные обратные.

Ungolfed:

z=0:z                       -- infinite zeroes for padding
x!0=[]                      -- digits of n in base x, recursion base case
x!n=mod n x:x!div n x       -- digits of n in base x
e c | Just a                -- read upwards from (*)
  <-lookup c                -- lookup code, given letter
  . zip"W3YZ56C$EF. 89*HM/ORU1QGNTDJ7%40LSBIP2-+XVKA" 
                            -- combine codes with correct letters
  . ((++)<*>(reverse<$>))   -- concatenate list and list with reverse codes
  $ take 9                  -- take nine
  . (++z)                   -- pad with infinite zeroes on right
  . (2!)                    -- convert to binary digits – [[1,1,1],[1,0,1,1]…]
  <$> scanl (+) 7           -- cumulative sum starting from 7 – [7,13,19,22,25…]
        (67!0x117CDBC49F9EEEF11C3A659CACB31236)       
                            -- (*) digits in base 67 – [6,6,3,3,3,9,5,7…]
  = zipWith
      (\l c->c<$[1..2*l+1]) -- combine width and color, length is 2*l+1
      (a++[0])              -- list of widths as 0/1, 0 added for interchar gap
      (cycle"█ ")           -- infinite list of bar colors, leftovers are unused
  >>=id                     -- concatenate
f s='*':s++"*">>=e          -- surround string with delimiters, map e and concat
Angs
источник