Codificador de código de barras Code 39


16

Escreva uma função ou programa que codifique uma sequência em um código de barras no formato Código 39 , em que cada caractere seja codificado como cinco barras separadas por quatro intervalos. Duas das barras e uma das lacunas são amplas e outras são estreitas (códigos 10 * 4), ou três das lacunas são amplas e nenhuma das barras é (4 códigos). Isso fornece 44 códigos diferentes, dos quais um é um código reservado usado para denotar o início e o fim da sequência codificada.

O desafio

A entrada é uma sequência que contém apenas caracteres do conjunto

1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ-. +/$%

A saída é a sequência codificada como uma sequência de código de barras. O espaço estreito e os espaços entre caracteres são um espaço único e um espaço amplo é de três espaços. A barra estreita é a sequência de bytes UTF-8 para o caractere Unicode "Bloco completo", █, ou seja, 0xE2 0x96 0x88e a barra larga representa três dessas seqüências / caracteres de bytes ( ███). A lista completa de códigos é:

      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

As barras e os espaços são intercalados, começando em uma barra; portanto, por exemplo, Q é

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

Depois de codificar todos os caracteres, a cadeia é delimitada nas duas extremidades com █ █ ███ ███ █. O espaço entre caracteres, um único espaço, é inserido entre cada letra. Sua solução pode gerar espaços à direita e uma nova linha à direita (nessa ordem).

Exemplos

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

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

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

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

Formatos de entrada / saída padrão são permitidos e brechas padrão não são permitidas. Isso é , então o código mais curto medido em bytes vence!


1
Podemos usar um caractere ASCII imprimível (você pode escolher qual permitir) em vez de █?
Erik the Outgolfer

Para uma linguagem como o BrainFuck, o que conta como "um uso"?
L4m2

1
@ Angs eu quis dizer na saída basicamente, não no código. Não é muito justo penalizar os usos do #personagem, pois, por exemplo, "#"não é o único motivo pelo qual ele pode ser usado em um idioma.
Erik the Outgolfer

@EriktheOutgolfer O que eu quis dizer foi em literais de string e afins, mas considerando o argumento de l4m2, talvez seja melhor desabilitá-lo. Afinal, existem idiomas que não podem gerar três bytes?
Angs

Respostas:


7

JavaScript (ES6), 225 212 bytes

Guardado 4 bytes graças a @ 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]))

Experimente online!

Quão?

A tabela pode ser reorganizada de modo que a máscara binária de 9 bits do caractere seja rapidamente deduzida de sua linha e coluna usando a seguinte fórmula:

n = m XOR (2 << k)

com:

  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

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)
l4m2

@ l4m2 Isso não parece ser válido. No entanto, eu consegui encontrar uma correção para 221 bytes .
Erik the Outgolfer

@EriktheOutgolfer Acho que é basicamente o que l4m2 enviou com um bloco inteiro em vez de '#'. Ou eu estou esquecendo de alguma coisa?
Arnauld

@ Arnauld O original deles não se encaixa adequadamente.
Erik the Outgolfer

3

Vermelho , 452 445 bytes

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]]

Experimente online!

Vou tentar jogar mais, mas não espero muito dessa solução ingênua.


2

Java 10, 455 bytes

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;}

Experimente online.

Explicação:

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 bytes

(*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);}

Experimente online!

-8 graças a ceilingcat

Usa estratégia de codificação da resposta de Arnauld. O link TIO inclui-w chave e clichê para remover avisos, que são desnecessários e, portanto, não estão incluídos na pontuação.

Além do esquema de codificação, conforme explicado por Arnauld, o outro truque aqui é manter a wvariável alternando entre 2 e 0 ( w=2*!w). Isso me permite escolher entre a primeira e a segunda string b. O primeiro é um espaço, o segundo é o retângulo preenchido.


inteligente, você deve publicá-la :)
LambdaBeta

2

C (gcc) , 241 239 227 213 207 bytes

#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);}

Experimente online!

Baseado na implementação do @ LambdaBeta .

Um pouco menos golfe:

#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);
}

1

Carvão , 90 bytes

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

Experimente online! Nota: Espaço à direita. Link é a versão detalhada do código. Explicação:

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

Coloque a sequência de entrada em *se mapeie-a duas vezes, finalmente juntando o resultado com espaços. Para o segundo mapa, existe um mapa adicional sobre o intervalo implícito 0..4, onde duas substrings são concatenadas e esses resultados são unidos à constante de cadeia vazia predefinida.

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

Para o primeiro mapa interno, crie uma sequência formada usando os dígitos incrementados, o alfabeto maiúsculo e os símbolos -. *+/$%e procure a posição do caractere de entrada mapeado. Por exemplo, C++mapearia para [12, 40, 40].

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

A primeira substring representa os espaços antes das barras. Não há nada antes da primeira barra, mas as outras barras dependem da posição do caractere de entrada mapeado: se tiver mais de 39 anos, apenas um local terá um único espaço; se estiver abaixo de 40, apenas um local terá três. espaços e a posição também é convertida em uma coluna dividindo-a por 10. Se a coluna e o índice do loop diferirem por 2 (módulo 4), esse é o local ímpar.

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

A segunda substring representa as barras. Se a posição estiver acima de 39, sempre haverá uma barra; caso contrário, a posição será pesquisada em uma matriz de bits mapeados para caracteres. Por exemplo, se a posição for 12, ela será indexada circularmente ao caractere ', que está 100111em binário, indicando barras largas nas colunas 1 e 2. (O líder 1é ignorado, simplesmente garante uma contagem consistente de bits.)


1

Perl 5 , 244 bytes

$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{''}

Experimente online!

Contém muitos caracteres não imprimíveis e de bytes altos, o link TIO fornece uma xxdrepresentação. Eu esperava que isso acabasse menor, e talvez ainda seja possível compactar os dados de uma maneira mais eficiente, para ver como vou. Isso cria todas as permutações de " "," ","█","███"e depois mapeia as indicações da lista para os caracteres correspondentes.


1

Haskell , 275 270 bytes

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

Experimente online!

O operador x!nque calcula os dígitos da base x de n é usado duas vezes para descomprimir os códigos. Os códigos são compactados primeiro como cadeias binárias com largura = 1 e largura = 0, sem levar em conta a cor, por exemplo R↔10000110↔262. Esses números são classificados e diferenciados para obter números no intervalo [3,66], que são compactados com o inverso do algoritmo de dígito binário como 0x117CDBC49F9EEEF11C3A659CACB31236. Ele contém apenas metade dos códigos, o restante são reversos.

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
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.