Koder kodów kreskowych Code 39

16

Napisz funkcję lub program, który koduje ciąg znaków do kodu kreskowego w formacie Code 39 , w którym każdy znak jest kodowany jako pięć pasków oddzielonych czterema przerwami. Albo dwa słupki i jeden z odstępów są szerokie, a inne wąskie (kody 10 * 4), lub trzy z odstępów są szerokie i żaden z słupków nie jest (4 kody). Daje to 44 różne kody, z których jeden jest kodem zastrzeżonym używanym do oznaczenia początku i końca zakodowanego ciągu.

Wyzwanie

Dane wejściowe to ciąg zawierający tylko znaki z zestawu

1234567890ABCDEFGHIJKLMNOPQRSTUVWXYZ-. +/$%

Dane wyjściowe to ciąg kodowany jako ciąg kodu kreskowego. Wąska przerwa i przerwy między znakami są pojedynczą przestrzenią, a szeroka przerwa to trzy spacje. Wąski słupek jest sekwencją bajtów UTF-8 dla znaku Unicode „Pełny blok”, █, tj. 0xE2 0x96 0x88A szeroki słupek to trzy takie sekwencje bajtów / znaki ( ███). Pełna lista kodów to:

      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

Słupki i spacje są przeplatane, zaczynając od słupka, więc na przykład Q to

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

Po zakodowaniu wszystkich znaków ciąg jest rozdzielany na obu końcach znakiem █ █ ███ ███ █. Przerwa między znakami, pojedyncza spacja, jest wstawiana między każdą literę. Twoje rozwiązanie może generować końcowe spacje i końcowy znak nowej linii (w tej kolejności).

Przykłady

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

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

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

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

Standardowe formaty wejścia / wyjścia są dozwolone, a standardowe luki są niedozwolone. To jest , więc wygrywa najkrótszy kod mierzony w bajtach!

Angs
źródło
1
Czy możemy użyć drukowalnego znaku ASCII (możesz wybrać, który ma być dozwolony) zamiast █?
Erik the Outgolfer
W przypadku języka takiego jak BrainFuck, co się liczy jako „jedno użycie”?
l4m2
1
@Angs Miałem na myśli na wyjściu w zasadzie, nie w kodzie. Nakładanie kar za użycie #postaci jest niesprawiedliwe , ponieważ na przykład "#"nie jest to jedyny powód, dla którego można ją zastosować w języku.
Erik the Outgolfer
@EriktheOutgolfer To, co miałem na myśli, było w literałach łańcuchowych i tym podobnych, ale biorąc pod uwagę punkt l4m2, najlepiej byłoby to zabronić. W końcu czy są jakieś języki, które nie są w stanie wyprowadzić trzech bajtów?
Angs

Odpowiedzi:

7

JavaScript (ES6), 225 212 bajtów

Zaoszczędzono 4 bajty dzięki @ 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]))

Wypróbuj online!

W jaki sposób?

Tabelę można zmienić w taki sposób, aby 9-bitowa maska ​​binarna znaku została szybko wydedukowana z wiersza i kolumny przy użyciu następującego wzoru:

n = m XOR (2 << k)

z:

  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
źródło
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)
14m2
@ l4m2 To nie wydaje się być prawidłowe. Udało mi się jednak znaleźć poprawkę na 221 bajtów .
Erik the Outgolfer
@EriktheOutgolfer Myślę, że to w zasadzie to, co L4m2 przesłał z pełnym blokiem zamiast „#”. A może coś mi brakuje?
Arnauld
@Arnauld Ich oryginał nie ma odpowiedniego miejsca.
Erik the Outgolfer
3

Czerwony , 452 445 bajtów

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

Wypróbuj online!

Spróbuję dalej grać w golfa, ale nie oczekuję wiele od tego naiwnego rozwiązania.

Galen Iwanow
źródło
2

Java 10, 455 bajtów

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

Wypróbuj online.

Wyjaśnienie:

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
Kevin Cruijssen
źródło
2

C (gcc) , 311 , 303 bajtów

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

Wypróbuj online!

-8 dzięki pułapce cat

Wykorzystuje strategię kodowania z odpowiedzi Arnaulda. Łącze TIO zawiera-w przełącznik i płytę kotła do usuwania ostrzeżeń, są one niepotrzebne i dlatego nie są uwzględniane w zapisie.

Oprócz schematu kodowania wyjaśnionego przez Arnaulda, drugą sztuczką jest utrzymanie wzmiennej jako przełączania między 2 a 0 ( w=2*!w). To pozwala mi wybierać pomiędzy pierwszym i drugim ciągiem w b. Pierwsza to spacja, druga to wypełniony prostokąt.

LambdaBeta
źródło
sprytne, powinieneś to opublikować :)
LambdaBeta
2

C (gcc) , 241 239 227 213 207 bajtów

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

Wypróbuj online!

Na podstawie implementacji @ LambdaBeta .

Nieco mniej golfa:

#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);
}
sufitowy
źródło
1

Węgiel drzewny , 90 bajtów

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

Wypróbuj online! Uwaga: końcowe miejsce. Link jest do pełnej wersji kodu. Wyjaśnienie:

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

Zawiń ciąg wejściowy ws, *a następnie zamapuj go dwukrotnie, ostatecznie łącząc wynik ze spacjami. W przypadku drugiej mapy istnieje dodatkowa mapa w niejawnym zakresie 0..4, w której dwa podciągi są łączone, a wyniki te są następnie łączone z uprzednio zdefiniowaną stałą pustego ciągu.

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

W przypadku pierwszej mapy wewnętrznej utwórz ciąg utworzony przez dodanie cyfr przyrostowych, wielkich liter alfabetu i symboli -. *+/$%oraz wyszukaj pozycję zamapowanego znaku wejściowego. Na przykład C++odwzorowałby na [12, 40, 40].

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

Pierwszy podciąg reprezentuje spacje przed słupkami. Przed pierwszym słupkiem nie ma nic, ale pozostałe słupki zależą od położenia odwzorowanego znaku wejściowego: jeśli ma ponad 39, to tylko jedno miejsce ma jedno miejsce, a jeśli jest poniżej 40, to tylko jedno miejsce ma trzy spacje, a pozycja jest również konwertowana na kolumnę dzieląc ją przez 10. Jeśli indeks kolumny i pętli różnią się o 2 (moduł 4), to jest to miejsce nieparzyste.

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

Drugi podciąg reprezentuje słupki. Jeśli pozycja jest większa niż 39, zawsze jest jeden pasek, w przeciwnym razie pozycja jest sprawdzana w szeregu bitów odwzorowanych na znaki. Na przykład, jeśli pozycja wynosi 12, to jest ona indeksowana cyklicznie do znaku ', który jest 100111w postaci binarnej, co wskazuje na szerokie słupki w kolumnach 1 i 2. (Przewodzenie 1jest ignorowane, po prostu zapewnia spójną liczbę bitów).

Neil
źródło
1

Perl 5 , 244 bajtów

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

Wypróbuj online!

Zawiera wiele niedrukowalnych i wysokobajtowych znaków, link TIO stanowi xxdreprezentację. Miałem nadzieję, że skończy się to mniej, i być może nadal będę w stanie spakować dane w bardziej wydajny sposób, więc zobaczę, jak pójdę. To tworzy wszystkie permutacje, " "," ","█","███"a następnie odwzorowuje oznaczenia z listy na odpowiednie znaki.

Dom Hastings
źródło
1

Haskell , 275 270 bajtów

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

Wypróbuj online!

Operator, x!nktóry oblicza cyfry x podstawy bazy n, jest używany dwukrotnie do dekompresji kodów. Kody są najpierw kompresowane jako ciągi binarne o szerokości = 1 i wąskim = 0, bez względu na kolor, np R↔10000110↔262. Liczby te są następnie sortowane i różnicowane, aby uzyskać liczby z zakresu [3,66], które są kompresowane odwrotnością algorytmu cyfr binarnych as 0x117CDBC49F9EEEF11C3A659CACB31236. Zawiera tylko połowę kodów, reszta to ich rewersy.

Nie golfowany:

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
źródło