En matemáticas e informática , el sistema de numeración hexadecimal (también base 16 o simplemente hex ) es un sistema de numeración posicional que representa números utilizando una base de dieciséis. A diferencia del sistema decimal que representa números utilizando diez símbolos, el hexadecimal utiliza dieciséis símbolos distintos, más frecuentemente los símbolos "0"–"9" para representar valores del 0 al 9 y "A"–"F" (o "a"–"f") para representar valores del diez al quince.
Los desarrolladores de software y los diseñadores de sistemas utilizan ampliamente los números hexadecimales porque proporcionan una representación conveniente de los valores codificados en binario . Cada dígito hexadecimal representa cuatro bits (dígitos binarios), también conocidos como nibble (o nybble). [1] Por ejemplo, un byte de 6 bits puede tener valores que van desde 000000 hasta 111111 (de 0 a 63 decimal) en formato binario, que se pueden escribir como 00 a 3F en formato hexadecimal.
En matemáticas, normalmente se utiliza un subíndice para especificar la base. Por ejemplo, el valor decimal711 se expresaría en hexadecimal como 2C7 16 . En programación, varias notaciones denotan números hexadecimales, generalmente con un prefijo. El prefijo 0x
se usa en C , que denotaría este valor como 0x2C7
.
En la codificación de transferencia se utiliza el sistema hexadecimal Base 16 , en el que cada byte del texto sin formato se divide en dos valores de 4 bits y se representa mediante dos dígitos hexadecimales.
En la mayoría de los casos de uso actuales, las letras A–F o a–f representan los valores 10–15, mientras que los números 0–9 se utilizan para representar sus valores decimales.
No existe una convención universal para utilizar mayúsculas o minúsculas, por lo que cada una de ellas es predominante o preferida en entornos particulares según los estándares o convenciones de la comunidad; incluso se utiliza una combinación de mayúsculas y minúsculas. Algunas pantallas de siete segmentos utilizan la combinación de mayúsculas y minúsculas "A b C d E F" para distinguir los dígitos A–F entre sí y del 0 al 9.
Existe cierta estandarización en el uso de espacios (en lugar de comas u otros signos de puntuación) para separar valores hexadecimales en una lista larga. Por ejemplo, en el siguiente volcado hexadecimal , cada byte de 8 bits es un número hexadecimal de 2 dígitos, con espacios entre ellos, mientras que el desplazamiento de 32 bits al comienzo es un número hexadecimal de 8 dígitos.
00000000 57 69 6b 69 70 65 64 69 61 2c 20 74 68 65 20 66 00000010 72 65 65 20 65 6e 63 79 63 6c 6f 70 65 64 69 61 20 74 68 61 74 20 61 6e 79 6f 6e 65 20 63 61 6e 00000030 20 65 64 69 74 0a
En contextos en los que la base no está clara, los números hexadecimales pueden ser ambiguos y confundirse con números expresados en otras bases. Existen varias convenciones para expresar valores de forma inequívoca. Un subíndice numérico (escrito en decimal) puede dar la base explícitamente: 159 10 es decimal 159; 159 16 es hexadecimal 159, que equivale a 345 10 . Algunos autores prefieren un subíndice de texto, como 159 decimal y 159 hex , o 159 d y 159 h .
Donald Knuth introdujo el uso de un tipo de letra particular para representar un radio particular en su libro The TeXbook . [2] Las representaciones hexadecimales se escriben allí en un tipo de letra de máquina de escribir : 5A3 , C1F27ED
En los sistemas de texto lineal, como los utilizados en la mayoría de los entornos de programación informática, han surgido diversos métodos:
0x
para indicar una constante hexadecimal puede haber tenido su origen en los sistemas IBM Stretch . Se deriva del 0
prefijo que ya se utilizaba para las constantes octales . Los valores de bytes se pueden expresar en hexadecimal con el prefijo \x
seguido de dos dígitos hexadecimales: '\x1B'
representa el carácter de control Esc ; "\x1B[0m\x1B[25;1H"
es una cadena que contiene 11 caracteres con dos caracteres Esc incrustados. [3] Para generar un entero como hexadecimal con la familia de funciones printf , se utiliza el código de conversión de formato %X
o .%x
ode;
T
x
T
FFh
05A3H
0FFh
FFh
0x42
$
como prefijo: $5A3
, $C1F27ED
.H'ABCD'
(para ABCD 16 ). De manera similar, Fortran 95 utiliza Z'ABCD'.16#5A3#
, 16#C1F27ED#
. Para las constantes de vector de bits, VHDL utiliza la notación x"5A3"
, x"C1F27ED"
. [6]8'hFF
, donde 8 es el número de bits en el valor y FF es la constante hexadecimal.16r
:16r5A3
16#
: 16#5A3
, 16#C1F27ED
.#x
y #16r
. También se puede configurar las variables *read-base* [7] y *print-base* [8] a 16 para cambiar el lector y la impresora de un sistema Common Lisp a la representación de números hexadecimales para leer e imprimir números. Por lo tanto, los números hexadecimales se pueden representar sin el código de prefijo #x o #16r, cuando la base de entrada o salida se ha cambiado a 16.&H
:&H5A3
&
para hexadecimal. [10]0h
prefijo: 0h5A3
,0hC1F27ED
16r
para indicar números hexadecimales: 16r5a3
, 16rC1F27ED
. Los números binarios, cuaternarios (base 4) y octales se pueden especificar de manera similar.X'5A3'
o X'C1F27ED'
, y se usa en ensamblador, PL/I , COBOL , JCL , scripts, comandos y otros lugares. Este formato también era común en otros sistemas IBM (y ahora obsoletos). Ocasionalmente se usaban comillas en lugar de apóstrofos.A veces se sabe que los números son hexadecimales.
%
: http://www.example.com/name%20with%20spaces
donde %20
es el código para el carácter de espacio (en blanco) , punto de código ASCII 20 en hexadecimal, 32 en decimal.U+
seguido del valor hexadecimal, por ejemplo, U+00A1
el signo de exclamación invertido (¡).#
: magenta , por ejemplo, se representa como #FF00FF
. [11] CSS también permite abreviaturas de 3 dígitos hexadecimales con un dígito hexadecimal por componente: #FA3
abrevia #FFAA33
(un naranja dorado: ).=
Espa=F1a
AA213FD51B3801043FBC
...:
). Esta, por ejemplo, es una dirección IPv6 válida: 2001:0db8:85a3:0000:0000:8a2e:0370:7334
o abreviada eliminando los ceros iniciales como 2001:db8:85a3::8a2e:370:7334
( Las direcciones IPv4 generalmente se escriben en decimal).3F2504E0-4F89-41D3-9A0C-0305E82C3301
.El uso de las letras A a la F para representar los dígitos superiores al 9 no era universal en la historia temprana de las computadoras.
Como no había números tradicionales para representar las cantidades del diez al quince, se volvieron a emplear letras alfabéticas como sustituto. La mayoría de los idiomas europeos carecen de palabras no decimales para algunos de los números del once al quince. Algunas personas leen números hexadecimales dígito por dígito, como un número de teléfono, o utilizando el alfabeto fonético de la OTAN , el Alfabeto Fonético Conjunto del Ejército y la Marina o un sistema ad hoc similar . A raíz de la adopción del hexadecimal entre los programadores del IBM System/360 , Magnuson (1968) [23] sugirió una guía de pronunciación que daba nombres cortos a las letras del hexadecimal; por ejemplo, "A" se pronunciaba "ann", B "bet", C "chris", etc. [23] Rogers (2007) [24] publicó en línea otro sistema de nombres que intenta hacer que la representación verbal sea distinguible en cualquier caso, incluso cuando el número real no contiene los números A–F. En las tablas siguientes se enumeran ejemplos. Babb (2015) elaboró otro sistema de nombres, basándose en un chiste de Silicon Valley . [25]
Otros han propuesto utilizar las convenciones verbales del Código Morse para expresar dígitos hexadecimales de cuatro bits, donde "dit" y "dah" representan cero y uno, respectivamente, de modo que "0000" se expresa como "dit-dit-dit-dit" (....), dah-dit-dit-dah (-..-) expresa el dígito con un valor de nueve, y "dah-dah-dah-dah" (----) expresa el dígito hexadecimal para el decimal 15.
Se han ideado sistemas de conteo de dígitos tanto para sistemas binarios como hexadecimales. Arthur C. Clarke sugirió utilizar cada dedo como un bit de encendido/apagado, lo que permite contar con los dedos desde cero hasta 1023 10 con diez dedos. [26] Otro sistema para contar hasta FF 16 (255 10 ) se ilustra a la derecha.
El sistema hexadecimal puede expresar números negativos de la misma manera que el decimal: −2A para representar −42 10 , −B01D9 para representar −721369 10 y así sucesivamente.
El hexadecimal también se puede utilizar para expresar los patrones de bits exactos utilizados en el procesador , por lo que una secuencia de dígitos hexadecimales puede representar un valor con signo o incluso un valor de punto flotante . De esta manera, el número negativo −42 10 se puede escribir como FFFF FFD6 en un registro de CPU de 32 bits (en complemento a dos ), como C228 0000 en un registro FPU de 32 bits o C045 0000 0000 0000 en un registro FPU de 64 bits (en el estándar IEEE de punto flotante ).
Así como los números decimales se pueden representar en notación exponencial , también lo pueden hacer los números hexadecimales. La notación P utiliza la letra P (o p , por "potencia"), mientras que E (o e ) cumple una función similar en la notación decimal E. El número después de la P es decimal y representa el exponente binario . Aumentar el exponente en 1 multiplica por 2, no por 16: 20p0 = 10p1 = 8p2 = 4p3 = 2p4 = 1p5 . Por lo general, el número se normaliza de modo que los dígitos hexadecimales comiencen con 1. (cero suele ser 0 sin P ).
Ejemplo: 1.3DEp42 representa 1.3DE 16 × 2 42 10 .
La notación P es requerida por el estándar de punto flotante binario IEEE 754-2008 y puede usarse para literales de punto flotante en la edición C99 del lenguaje de programación C. [ 27] Usando los especificadores de conversión %a o %A , esta notación puede ser producida por implementaciones de la familia de funciones printf siguiendo la especificación C99 [28] y el estándar POSIX Single Unix Specification (IEEE Std 1003.1) . [29]
La mayoría de las computadoras manipulan datos binarios, pero es difícil para los humanos trabajar con una gran cantidad de dígitos incluso para un número binario relativamente pequeño. Aunque la mayoría de los humanos están familiarizados con el sistema de base 10, es mucho más fácil convertir el binario en hexadecimal que en decimal porque cada dígito hexadecimal se corresponde con un número entero de bits (4 10 ). Este ejemplo convierte 1111 2 en base diez. Dado que cada posición en un numeral binario puede contener un 1 o un 0, su valor se puede determinar fácilmente por su posición desde la derecha:
Por lo tanto:
Con un poco de práctica, convertir 1111 2 en F 16 en un solo paso se vuelve fácil (ver la tabla en la representación escrita). La ventaja de usar hexadecimal en lugar de decimal aumenta rápidamente con el tamaño del número. Cuando el número se vuelve grande, la conversión a decimal es muy tediosa. Sin embargo, cuando se convierte en hexadecimal, es trivial considerar la cadena binaria como grupos de 4 dígitos y convertir cada uno en un solo dígito hexadecimal. [30]
Este ejemplo muestra la conversión de un número binario a decimal, asignando cada dígito al valor decimal y sumando los resultados.
Compare esto con la conversión a hexadecimal, donde cada grupo de cuatro dígitos puede considerarse independientemente y convertirse directamente:
La conversión de hexadecimal a binario es igualmente directa. [30]
Aunque el sistema cuaternario (base 4) se utiliza poco, se puede convertir fácilmente de y hacia hexadecimal o binario. Cada dígito hexadecimal corresponde a un par de dígitos cuaternarios, y cada dígito cuaternario corresponde a un par de dígitos binarios. En el ejemplo anterior 2 5 C 16 = 02 11 30 4 .
El sistema octal (base 8) también se puede convertir con relativa facilidad, aunque no de forma tan trivial como con las bases 2 y 4. Cada dígito octal corresponde a tres dígitos binarios, en lugar de cuatro. Por lo tanto, podemos convertir entre octal y hexadecimal mediante una conversión intermedia a binario seguida de la reagrupación de los dígitos binarios en grupos de tres o cuatro.
Como ocurre con todas las bases, existe un algoritmo sencillo para convertir la representación de un número en hexadecimal realizando operaciones de división de números enteros y resto en la base de origen. En teoría, esto es posible desde cualquier base, pero para la mayoría de los humanos, solo el decimal y para la mayoría de las computadoras, solo el binario (que se puede convertir con métodos mucho más eficientes) se pueden manejar fácilmente con este método.
Sea d el número a representar en hexadecimal, y la serie h i h i−1 ...h 2 h 1 los dígitos hexadecimales que representan el número.
"16" puede ser reemplazado por cualquier otra base que se desee.
A continuación se muestra una implementación en JavaScript del algoritmo anterior para convertir cualquier número en hexadecimal en una representación de cadena. Su propósito es ilustrar el algoritmo anterior. Sin embargo, para trabajar con datos en serio, es mucho más recomendable trabajar con operadores bit a bit .
función toHex ( d ) { var r = d % 16 ; if ( d - r == 0 ) { volver a Char ( r ); } volver a Hex (( d - r ) / 16 ) + toChar ( r ); } función toChar ( n ) { const alpha = "0123456789ABCDEF" ; devolver alfa . charAt ( n ); }
También es posible realizar la conversión asignando a cada lugar de la base de origen la representación hexadecimal de su valor posicional, antes de realizar la multiplicación y la suma para obtener la representación final. Por ejemplo, para convertir el número B3AD a decimal, se puede dividir el número hexadecimal en sus dígitos: B (11 10 ), 3 (3 10 ), A (10 10 ) y D (13 10 ), y luego obtener el resultado final multiplicando cada representación decimal por 16 p ( siendo p la posición del dígito hexadecimal correspondiente, contando de derecha a izquierda, comenzando con 0). En este caso, tenemos que:
B3AD = (11 × 16 3 ) + (3 × 16 2 ) + (10 × 16 1 ) + (13 × 16 0 )
que es 45997 en base 10.
Muchos sistemas informáticos proporcionan una utilidad de calculadora capaz de realizar conversiones entre varios sistemas numéricos, frecuentemente incluidos los hexadecimales.
En Microsoft Windows , la utilidad Calculadora se puede configurar en modo Programador, que permite realizar conversiones entre bases de datos de 16 (hexadecimal), 10 (decimal), 8 ( octal ) y 2 ( binario ), las bases más utilizadas por los programadores. En el modo Programador, el teclado numérico en pantalla incluye los dígitos hexadecimales de la A a la F, que están activos cuando se selecciona "Hex". Sin embargo, en el modo hexadecimal, la Calculadora de Windows solo admite números enteros.
Las operaciones elementales como la división se pueden realizar indirectamente a través de la conversión a un sistema numérico alternativo , como el sistema decimal comúnmente utilizado o el sistema binario donde cada dígito hexadecimal corresponde a cuatro dígitos binarios.
Alternativamente, también se pueden realizar operaciones elementales directamente dentro del propio sistema hexadecimal, basándose en sus tablas de suma y multiplicación y sus algoritmos estándar correspondientes, como la división larga y el algoritmo de resta tradicional.
Al igual que con otros sistemas numéricos, el sistema hexadecimal se puede utilizar para representar números racionales , aunque las expansiones repetidas son comunes ya que dieciséis (10 16 ) tiene un solo factor primo: dos.
Para cualquier base, 0,1 (o "1/10") siempre es equivalente a uno dividido por la representación de ese valor base en su propio sistema numérico. Por lo tanto, ya sea dividiendo uno por dos para binario o dividiendo uno por dieciséis para hexadecimal, ambas fracciones se escriben como 0.1
. Debido a que la base 16 es un cuadrado perfecto (4 2 ), las fracciones expresadas en hexadecimal tienen un período impar con mucha más frecuencia que las decimales, y no hay números cíclicos (excepto dígitos simples triviales). Los dígitos recurrentes se exhiben cuando el denominador en términos más bajos tiene un factor primo que no se encuentra en la base; por lo tanto, cuando se usa la notación hexadecimal, todas las fracciones con denominadores que no son una potencia de dos dan como resultado una cadena infinita de dígitos recurrentes (como tercios y quintos). Esto hace que el hexadecimal (y el binario) sean menos convenientes que el decimal para representar números racionales, ya que una proporción mayor se encuentra fuera de su rango de representación finita.
Todos los números racionales finitamente representables en hexadecimal son también finitamente representables en decimal, duodecimal y sexagesimal : es decir, cualquier número hexadecimal con un número finito de dígitos también tiene un número finito de dígitos cuando se expresa en esas otras bases. Por el contrario, sólo una fracción de aquellos finitamente representables en las últimas bases son finitamente representables en hexadecimal. Por ejemplo, el decimal 0,1 corresponde a la representación recurrente infinita 0,1 9 en hexadecimal. Sin embargo, el hexadecimal es más eficiente que el duodecimal y el sexagesimal para representar fracciones con potencias de dos en el denominador. Por ejemplo, 0,0625 10 (un dieciseisavo) es equivalente a 0,1 16 , 0,09 12 y 0;3,45 60 .
La siguiente tabla muestra las expansiones de algunos números irracionales comunes en decimal y hexadecimal.
Las potencias de dos tienen expansiones muy simples en hexadecimal. Las primeras dieciséis potencias de dos se muestran a continuación.
Las unidades de medida tradicionales chinas eran de base 16. Por ejemplo, un jīn (斤) en el antiguo sistema equivale a dieciséis taeles . El suanpan ( ábaco chino ) se puede utilizar para realizar cálculos hexadecimales como sumas y restas. [31]
Al igual que con el sistema duodecimal , ha habido intentos ocasionales de promover el hexadecimal como el sistema numérico preferido. Estos intentos a menudo proponen una pronunciación y símbolos específicos para los numerales individuales. [32] Algunas propuestas unifican las medidas estándar para que sean múltiplos de 16. [33] [34] Una de esas primeras propuestas fue presentada por John W. Nystrom en Project of a New System of Arithmetic, Weight, Measure and Coins: Proposed to be called the Tonal System, with Sixteen to the Base , publicado en 1862. [35] Nystrom, entre otras cosas, sugirió el tiempo hexadecimal , que subdivide un día por 16, de modo que hay 16 "horas" (o "10 tims ", pronunciado tontim ) en un día. [36]
La palabra hexadecimal se registró por primera vez en 1952. [37] Es macarrónica en el sentido de que combina el griego ἕξ (hex) "seis" con el latín -decimal . La alternativa totalmente latina sexadecimal (compárese la palabra sexagesimal para la base 60) es más antigua y se usa al menos ocasionalmente desde fines del siglo XIX. [38] Todavía se usa en la década de 1950 en la documentación de Bendix . Schwartzman (1994) sostiene que el uso de sexadecimal puede haberse evitado debido a su sugerente abreviatura de sexo . [39] Muchos idiomas occidentales desde la década de 1960 han adoptado términos equivalentes en formación al hexadecimal (por ejemplo, el francés hexadécimal , el italiano esadecimale , el rumano hexazecimal , el serbio хексадецимални , etc.) pero otros han introducido términos que sustituyen palabras nativas para "dieciséis" (por ejemplo, el griego δεκαεξαδικός, el islandés sextándakerfi , el ruso шестнадцатеричной , etc.)
La terminología y la notación no se establecieron hasta finales de la década de 1960. En 1969, Donald Knuth argumentó que el término etimológicamente correcto sería senidenario , o posiblemente sedenario , un término latino que pretendía transmitir "agrupado por 16" modelado sobre binario , ternario , cuaternario , etc. Según el argumento de Knuth, los términos correctos para la aritmética decimal y octal serían denario y octonario , respectivamente. [40] Alfred B. Taylor utilizó senidenario en su trabajo de mediados del siglo XIX sobre bases numéricas alternativas, aunque rechazó la base 16 debido a su "número incómoda de dígitos". [41] [42]
La notación actual que utiliza las letras A a F se establece como el estándar de facto a partir de 1966, a raíz de la publicación del manual Fortran IV para IBM System/360 , que (a diferencia de las variantes anteriores de Fortran) reconoce un estándar para ingresar constantes hexadecimales. [43] Como se señaló anteriormente, NEC (1960) y The Pacific Data Systems 1020 (1964) utilizaron notaciones alternativas . El estándar adoptado por IBM parece haber sido ampliamente adoptado en 1968, cuando Bruce Alan Martin en su carta al editor del CACM se queja de que
Con la ridícula elección de las letras A, B, C, D, E, F como símbolos numéricos hexadecimales, que se suma a los ya problemáticos problemas de distinguir los números octales (o hexadecimales) de los números decimales (o nombres de variables), ha llegado el momento de reconsiderar nuestros símbolos numéricos. ¡Esto debería haberse hecho antes de que las malas decisiones se convirtieran en un estándar de facto!
El argumento de Martin era que el uso de los numerales del 0 al 9 en números no decimales "nos implica un esquema de valor posicional de base diez": "¿Por qué no utilizar símbolos (y nombres) completamente nuevos para los siete o quince dígitos distintos de cero necesarios en octal o hexadecimal? Incluso el uso de las letras A a P sería una mejora, pero símbolos completamente nuevos podrían reflejar la naturaleza binaria del sistema". [19] También argumentó que "reutilizar letras alfabéticas para dígitos numéricos representa un gigantesco paso atrás respecto de la invención de glifos distintos y no alfabéticos para numerales hace dieciséis siglos" (como los numerales Brahmi , y más tarde en un sistema de numeración hindú-arábigo ), y que los estándares ASCII recientes (ASA X3.4-1963 y USAS X3.4-1968) "deberían haber preservado seis posiciones de tabla de códigos después de los diez dígitos decimales, en lugar de llenarlas innecesariamente con caracteres de puntuación" (":;<=>?") que podrían haberse colocado en otro lugar entre las 128 posiciones disponibles.
Base16 (como nombre propio sin espacio) también puede referirse a una codificación de binario a texto que pertenece a la misma familia que Base32 , Base58 y Base64 .
En este caso, los datos se dividen en secuencias de 4 bits y cada valor (entre 0 y 15 inclusive) se codifica utilizando uno de los 16 símbolos del conjunto de caracteres ASCII . Aunque se pueden utilizar 16 símbolos cualesquiera del conjunto de caracteres ASCII, en la práctica, siempre se eligen los dígitos ASCII "0" a "9" y las letras "A" a "F" (o las minúsculas "a" a "f") para alinearlos con la notación escrita estándar para números hexadecimales.
La codificación Base16 tiene varias ventajas:
Las principales desventajas de la codificación Base16 son:
La compatibilidad con la codificación Base16 es omnipresente en la informática moderna. Es la base del estándar W3C para la codificación de porcentajes de URL , donde un carácter se reemplaza por un signo de porcentaje "%" y su forma codificada en Base16. La mayoría de los lenguajes de programación modernos incluyen directamente compatibilidad con el formato y el análisis de números codificados en Base16.
{{cite book}}
: CS1 maint: location missing publisher (link)"\x1B[0m\x1B[25;1H"
especifica la secuencia de caracteres Esc [ 0 m Esc [ 2 5; 1 H . Estas son las secuencias de escape utilizadas en un terminal ANSI que restablecen el conjunto de caracteres y el color, y luego mueven el cursor a la línea 25.&
como prefijo valores octales&O
. (Microsoft BASIC utiliza principalmente como prefijo octal y &H
como prefijo hexadecimal, pero el ampersand por sí solo produce una interpretación predeterminada como prefijo octal.Esta base se utiliza porque un grupo de cuatro bits puede representar cualquiera de los dieciséis números diferentes (de cero a quince). Al asignar un símbolo a cada una de estas combinaciones, llegamos a una notación llamada sexadecimal (generalmente "hex" en la conversación porque nadie quiere abreviar "sexo"). Los símbolos en el lenguaje sexadecimal son los diez dígitos decimales y en la máquina de escribir G-15, las letras "u", "v", "w", "x", "y" y "z". Estas son marcas arbitrarias; otras computadoras pueden usar caracteres alfabéticos diferentes para estos últimos seis dígitos.