Produzir uma tabela XOR


19

Introdução

O XOR é um portão lógico digital que implementa um ou exclusivo. Na maioria das vezes, isso é mostrado como ^. Os quatro resultados possíveis em binário:

0 ^ 0 = 0
0 ^ 1 = 1
1 ^ 0 = 1
1 ^ 1 = 0

Isso também pode ser visto como módulo 2 de adição no binário. Em decimal, precisamos converter o decimal em binário 35 = 100011e 25 = 11001. Para calcular o valor XOR, os colocamos um sobre o outro:

100011
 11001 ^
--------
111010  = 58 in decimal

A tarefa : quando receber um valor inteiro N maior que 1, produza uma tabela XOR com o tamanho N + 1. Por exemplo, N = 5:

 0 1 2 3 4 5
 1 0 3 2 5 4
 2 3 0 1 6 7
 3 2 1 0 7 6
 4 5 6 7 0 1
 5 4 7 6 1 0

Você pode ver que existe um espaço na frente de cada número, porque o valor mais alto da tabela possui o comprimento 1. No entanto, se usarmos N = 9, obteremos a seguinte grade:

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

O valor mais alto tem o comprimento 2, portanto, o valor está alinhado à direita com o comprimento 3 (comprimento mais alto + 1).

Regras:

  • O espaço em branco à esquerda não é obrigatório, apenas se usado (ou não) de forma consistente
  • Você deve gerar uma tabela no formato mostrado acima.
  • O preenchimento entre as colunas deve ser o menor possível
  • Isso é , então a submissão com a menor quantidade de bytes ganha!

Quanto preenchimento é permitido entre colunas? Apenas a quantidade mínima possível?
Martin Ender

@ MartinBüttner Sim, a quantidade mínima possível. Vou adicionar isso à descrição.
Adnan


Olhando para os exemplosoutput an XOR table with the size N+1
edc65

Qual é o valor máximo para N? ... (para N == 1000000 o tamanho da tabela seria perto de 10 Terabyte)
edc65

Respostas:


12

MATL , 10 bytes

0i2$:tXgZ~

O compilador (e em particular este programa) agora parece funcionar no Octave, embora ainda precise de algum refinamento. Você pode usar provisoriamente esse commit do GitHub .

Editar (30/03/16) : Experimente online!

Exemplo

>> matl 0i2$:tXgZ~
> 9
0  1  2  3  4  5  6  7  8  9
1  0  3  2  5  4  7  6  9  8
2  3  0  1  6  7  4  5 10 11
3  2  1  0  7  6  5  4 11 10
4  5  6  7  0  1  2  3 12 13
5  4  7  6  1  0  3  2 13 12
6  7  4  5  2  3  0  1 14 15
7  6  5  4  3  2  1  0 15 14
8  9 10 11 12 13 14 15  0  1
9  8 11 10 13 12 15 14  1  0

Explicação

0i2$:       % vector 0, 1, 2, ... up to input number
t           % duplicate
Xg          % nd-grid
Z~          % bitxor

Compiler trabalho em Octave em breve ...
Luis Mendo

... bem, espero. Alguma idéia sobre isso , alguém?
Luis Mendo

5
Definitivamente, a ferramenta certa para o trabalho. +1
a spaghetto

11
Após alguns ajustes, o compilador parece funcionar no Octave (4.0.0). Preciso fazer mais testes, mas todos os programas que experimentei, incluindo este, funcionam. Enquanto eu publico uma nova versão, aqui está um link para o commit atual do GitHub para executar este código no Octave. A linguagem não mudou (ainda é liberar 5.0.0), por isso é mais cedo do que este desafio
Luis Mendo

5

Utilitários Bash + BSD, 45

eval echo \$[{0..$1}^{0..$1}]|rs -jg1 $[$1+1]

Estou esperando há muito tempo para encontrar um uso rs. Parece ser bom. rspode precisar ser instalado nos sistemas Linux. Mas ele sai da caixa no OS X.

  • $1expande para N e, portanto, echo \$[{0..$1}^{0..$1}]expande paraecho $[{0..N}^{0..N}]
  • Isso é evaled:
  • A expansão da cinta se expande para $[0^0] $[0^1] $[0^2] ... $[0^N] ... $[N^N]
  • Esta é uma série de expansões aritméticas xor que se expandem para uma linha de termos
  • rs(remodelar) reformula esta linha para N + 1 linhas. -jjustifica à direita e -g1fornece uma largura de calha igual a 1. Isso garante que a tabela de saída final tenha uma largura mínima entre as colunas.

Testei até N = 1000, o que levou 3,8 segundos. Teoricamente, N grande é possível, embora o bash fique sem memória em algum momento com o uso de memória (N + 1) ² da expansão do suporte.


3

JavaScript (ES6) 120 122

Edite 2 bytes salvos thx ETHproductions

Uma função anônima. Nota: o número na tabela é limitado a 7 dígitos, mais do que razoável, dado o tamanho geral de uma tabela, permitindo números maiores

Agora, devo encontrar uma maneira mais curta de obter o tamanho máximo das colunas, evitando logaritmos

n=>(a=Array(n+1).fill(-~Math.log10(2<<Math.log2(n)))).map((m,i)=>a.map((z,j)=>`       ${i^j}`.slice(~m)).join``).join`
`

Teste

f=n=>(a=Array(n+1).fill(-~Math.log10(2<<Math.log2(n)))).map((m,i)=>a.map((z,j)=>`       ${i^j}`.slice(~m)).join``).join`\n`

function update() {
  var v=+I.value
  O.textContent = f(v)
}  

update()
N: <input id=I type=number value=9 oninput='update()'>
<pre id=O></pre>


Impressionante, adoro o uso de ~mpara capturar um espaço extra. O uso de uma sequência de modelos pode salvar dois bytes:(z,j)=>`(7 spaces)${i^j}`.slice(~m)
ETHproductions

@ETHproductions 1) boa sugestão, obrigado 2) como você conseguiu colocar um ... (não posso nem perguntar) ... um char (96) dentro de 2 char (96)?
Edc65

Ah, você só usa vários backticks, assim: (ignore this padding) ``abc`def`` (ignore this too)abc`def
Aparece

3

C, 114 128 152

Editar contagem de espaço simplificada, inspirada no trabalho de Khaled A Khunaifer

Função CA que segue as especificações.

T(n){int i=0,j,x=1,d=0;while(x<=n)x+=x,++d;for(;i<=n;i++)for(j=0;j<=n;j++)printf("%*d%c",d*3/10+1,i^j,j<n?32:10);}

Experimente inserir n como entrada, padrão 9

Menos golfe

T(n)
{
   int i=0, j, x=1,d=0;
   while(x<=n) x+=x,++d; // count the digits
   // each binary digit is approximately 0.3 decimal digit
   // this approximation is accurate enough for the task
   for(;i<=n;i++)
     for(j=0;j<=n;j++)
       printf("%*d%c",d*3/10+1,
              i^j,
              j < n ? 32:10); // space between columns, newline at end
}

3

C, 103 bytes

Y(n){int i,j=n++;char*f="%2u";while(j/=8)++f[1];for(;j<n;++j,puts(""))for(i=0;i<n;++i)printf(f,i^j);}

11
Bem-vindo à programação de quebra-cabeças e código de golfe! Pelo que posso ver, sua resposta, como muitas outras aqui, infelizmente, está errada no ponto de alinhamento da coluna.
Edc65

Tente com a entrada 5 (muito espaço entre as colunas) ou 65 (muito pouco)
edc65

isso seria bom, finalmente, até 512, finalmente aqui ...
Ros

3

Gelatina, não concorrente

7 bytes Esta resposta não é concorrente, pois usa recursos que pós-datam o desafio.

0r©^'®G

Experimente online!

Como funciona

0r©^'®G  Main link. Input: n (integer)

0r       Range; yield [0, ..., n].
  ©      Save the range in the register.

     ®   Yield the range from the register.
   ^'    XOR each integer in the left argument with each integer in the right one.
      G  Grid; separate rows by newlines, columns by spaces, with a fixed width for
         all columns. Since the entries are numeric, align columns to the right.

2
Parabéns pela sua resposta :) 1000
Adnan

3

R, 38 bytes

Normalmente, o R requer muitos bytes apenas para formatar a saída. Nesse caso, é exatamente o oposto. outerque geralmente se refere ao produto externo de duas matrizes, pode, quando fornecida, uma função executar isso nas margens dos vetores. Nesse caso, aplicamos a função XOR bit a bit bitwXor.

names(x)=x=1:scan();outer(x,x,bitwXor)

2

CJam, 29 27 bytes

ri:X),_ff{^s2X2b,#s,)Se[}N*

Teste aqui.

Explicação

ri      e# Read input and convert to integer.
:X      e# Store in X.
),      e# Get range [0 1 ... X].
_ff{    e# Nested map over all repeated pairs from that range...
  ^     e#   XOR.
  s     e#   Convert to string.
  2     e#   Push 2.
  X2b,  e#   Get the length of the base-2 representation of X. This is the same as getting
        e#   getting the base-2 integer logarithm and incrementing it.
  #     e#   Raise 2 to that power. This rounds X up to the next power of 2.
  s,    e#   Convert to string and get length to determine column width.
  )     e#   Increment for additional padding.
  Se[   e#   Pad string of current cell from the left with spaces.
}
N*      e# Join with linefeeds.

Bem, isso foi rápido, haha ​​#
315 Adnan

2

MathCAD, 187 bytes

insira a descrição da imagem aqui

O MathCAD lida com tabelas construídas facilmente - mas não possui absolutamente Xor bit a bit, nem conversores decimais para binários ou binários para decimais. As funções for iteram pelos valores possíveis. Os lugares i, a2, Xa e Xb são mantidos. O loop while se converte ativamente em binário e durante a conversão em binário também executa a função xor (a pequena cruz com o círculo ao redor). Ele armazena o número binário em um número de base 10 composto por 0 e 1. Este é então convertido antes de ser armazenado na matriz M através da função de soma.

Isso pode ser resolvido facilmente (mesmo trocando os espaços reservados por outros mais curtos), mas achei que iria postá-lo e ver se alguém pode jogar o conversor binário em decimal mais do que qualquer outra coisa.


2

k4, 50 bytes

{-1@" "/:'(-|//#:''x)$x:$2/:''~x=/:\:x:0b\:'!1+x;}

Por exemplo:

  {-1@" "/:'(-|//#:''x)$x:$2/:''~x=/:\:x:0b\:'!1+x;}9
 0  1  2  3  4  5  6  7  8  9
 1  0  3  2  5  4  7  6  9  8
 2  3  0  1  6  7  4  5 10 11
 3  2  1  0  7  6  5  4 11 10
 4  5  6  7  0  1  2  3 12 13
 5  4  7  6  1  0  3  2 13 12
 6  7  4  5  2  3  0  1 14 15
 7  6  5  4  3  2  1  0 15 14
 8  9 10 11 12 13 14 15  0  1
 9  8 11 10 13 12 15 14  1  0

2

C, 149 bytes

int i=0,j,n,k=2;char b[256];scanf("%d",&n);while((k*=2)<=n);k^=k-1;while(i++<=n&&putchar(10))for(j=0;j<=n;j++)printf(" %*d",sprintf(b,"%d",k),i-1^j);
---------

Detalhado

#include <stdio.h>

int main()
{
    int i=0, j, n, k=2;
    char b[256] = { 0 };

    scanf("%d", &n);

    // find max xor value in the table
    while((k*=2)<=n); k^=k-1;

    printf("> %d ~ %d", k, sprintf(b,"%d",k));

    while(i++ <= n && putchar(10))
    {
        for(j = 0; j <= n;j++)
        {
            printf(" %*d", sprintf(b,"%d",k), (i-1)^j);
        }
    }

    return 0;
}

11
Assim como seu post em java, não há nenhuma tentativa de alinhar corretamente as colunas. Essa é a única parte difícil deste desafio. Este código fornece uma saída incorreta quando entrada <8 ou> 64.
edc65

@ edc65 Descobri como alinhar, o valor máximo xor é binário 11..1ao significativo no valor de entrada n, pode ser feito encontrando primeiro a potência mais próxima de 2 e, em seguida, xor com o número anterior,0001 xor 1110 = 1111
Khaled.K

Bom, você está no caminho certo agora. Ainda assim, você deve testar mais: o loop com k fornece resultado errado para n = 8 (e sua resposta curta não inicializa a variável local i para 0)
edc65

Este circuito simples deve fazer: for(k=1;k<=n;)k*=2;k--;. Agora vejo que é muito menor do que a minha tentativa C para o mesmo (o meu é melhor para o desempenho, mas o desempenho não importa neste desafio)
edc65

@ edc65 que você precisa fazer a 2^k xor 2^k -1favor max{2^k<=n}ou a 2^k -1favor min{2^k>=n}. para obter tudo 11..1lá dentro
Khaled.K

2

Python 3, 133 131 bytes

import math
n=int(input())
r,p=range(n+1),print
for y in r:[p(end='%%%dd '%len(str(2**int(math.log2(n)+1)-1))%(x^y))for x in r];p()

1

Mathematica, 108 bytes

StringRiffle[Thread[Map[ToString,a=Array[BitXor,{#,#}+1,0],{2}]~StringPadLeft~IntegerLength@Max@a],"
"," "]&

Desconsidere o erro, é apenas Thread não sei o que está fazendo.


1

Emacs Lisp, 193 bytes

(defun x(n)(set'a 0)(set'n(1+ n))(while(< a n)(set'b 0)(while(< b n)(princ(format(format"%%%ss "(ceiling(log(expt 2(ceiling(log n 2)))10)))(logxor a b)))(set'b(1+ b)))(set'a(1+ a))(message"")))

Ungolfed:

(defun x(n)
  (set'a 0)
  (set'n(1+ n))
  (while(< a n)
    (set'b 0)
    (while(< b n)
      (princ
        (format
          ;; some format string magic to get the length of the longest
          ;; possible string as a format string
          (format "%%%ss " (ceiling(log(expt 2(ceiling(log n 2)))10)))
          (logxor a b)))
      (set'b(1+ b)))
    (set'a(1+ a))
    ;; new line
    (message"")))

A saída é enviada para o *Message*buffer, que seria stdoutse xfosse usada dentro de um script.

(x 9)
 0  1  2  3  4  5  6  7  8  9 
 1  0  3  2  5  4  7  6  9  8 
 2  3  0  1  6  7  4  5 10 11 
 3  2  1  0  7  6  5  4 11 10 
 4  5  6  7  0  1  2  3 12 13 
 5  4  7  6  1  0  3  2 13 12 
 6  7  4  5  2  3  0  1 14 15 
 7  6  5  4  3  2  1  0 15 14 
 8  9 10 11 12 13 14 15  0  1 
 9  8 11 10 13 12 15 14  1  0

1

Python 2, 114 bytes

Demorou alguns anos para encontrar uma maneira de fazer preenchimento de largura variável .format()(alguns, não muito) e ajustá-lo da maneira certa, mas acho que tenho tudo para especificar agora. Poderia usar mais golfe nesse cálculo de largura.

N=input()+1
for i in range(N):print''.join(' {n:>{w}}'.format(w=len(`2**(len(bin(N))-2)`),n=i^r)for r in range(N))

1

Objeto Caché , 127 bytes

x(n)s w=$L(2**$bitfind($factor(n),1,100,-1))+1 f i=0:1:n { w $j(i,w) } f i=1:1:n { w !,$j(i,w) f j=1:1:n { w $j($zb(i,j,6),w) } }

Detalhado:

x(n)
 set w=$Length(2**$bitfind($factor(n),1,100,-1))+1
 for i=0:1:n {
     write $justify(i,w)
 }
 for i=1:1:n {
     write !,$justify(i,w)
     for j=1:1:n {
         write $justify($zboolean(i,j,6),w)
     }

 }

1

Pyke, 8 bytes (não concorrente)

hD]UA.&P

Explicação:

         - auto-add eval_or_not_input() to the stack
h        - increment the input
 D]      - Create a list containing [inp+1, inp+1]
   U     - Create a 2d range 
    A.^  - Deeply apply the XOR function to the range
       P - print it out prettily

Experimente aqui


0

Python 2, 77 bytes

n=input()+1
t=range(n)
for i in t: print "%3d"*n % tuple([x^i for x in t])

3
Bem-vindo à programação de quebra-cabeças e código de golfe! Pelo que posso ver, sua resposta, como muitas outras aqui, infelizmente, perde o ponto de alinhamento da coluna.
cat


0

Excel VBA, 95 bytes

Função de janela imediata VBE anônima que leva as entradas do intervalo [A1]e as saídas para o console.

For y=0To[A1]:For x=0To[A1]:z=x Xor y:?Spc([Len(2^-Int(-Log(A1,2)))]-Len(z))Str(z);:Next:?:Next

0

Pequeno Básico , 499 bytes

Um script que recebe a entrada do TextWindowobjeto e gera a mesma

a=TextWindow.Read()
For y=0To a
For x=0To a
n=x
b()
z1=z
n=y
b()
l=Math.Max(Array.GetItemCount(z),Array.GetItemCount(z1))
o=0
For i=1To l
If z1[i]<>z[i]And(z[i]=1Or z1[i]=1)Then
z2=1
Else 
z2=0
EndIf
o=o+z2*Math.Power(2,i-1)
EndFor
TextWindow.Write(Text.GetSubText("      ",1,Text.GetLength(Math.Power(2,Math.Ceiling(Math.Log(a)/Math.Log(2))))-Text.GetLength(o))+o+" ")
EndFor
TextWindow.WriteLine("")
EndFor
Sub b
z=0
c=0  
While n>0
c=c+1
z[c]=Math.Remainder(n,2)
n=Math.Floor(n/2)
EndWhile
EndSub

Experimente em SmallBasic.com Usa o Silverlight e, portanto, deve ser executado no IE ou Edge

Selecione o console preto, digite input número inteiro e pressione Enter.

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.