31
votes

I found out that there is function called .hex.to.dec in the fBasics package.

When I do .hex.to.dec(a), it works.

I have a data frame with a column samp_column consisting of such values:

a373, 115c6, a373, 115c6, 176b3

When I do .hex.to.dec(samp_column), I get this error:

"Error in nchar(b) : 'nchar()' requires a character vector"

When I do .hex.to.dec(as.character(samp_column)), I get this error:

"Error in rep(base.out, 1 + ceiling(log(max(number), base = base.out))) : invalid 'times' argument"

What would be the best way of doing this?

5
strtoi will do it. Or the long way: Rutils::as.character.binmode and then convert from character to number with as.numeric :-) - Carl Witthoft
Please read about how to format your question and then about how to make a reproducible example that includes your data. - Thomas
Thank you Carl! For some reason, this function did not show up in my Google searches, but it seems to work really well. :) Cheers! - albay_aureliano

5 Answers

42
votes

Use base::strtoi to convert hexadecimal character vectors to integer:

strtoi(c("0xff", "077", "123"))
#[1] 255  63 123
14
votes

There is a simple and generic way to convert hex <-> other formats using "C/C++ way":

V <- c(0xa373, 0x115c6, 0xa373, 0x115c6, 0x176b3)

sprintf("%d", V)
#[1] "41843" "71110" "41843" "71110" "95923"

sprintf("%.2f", V)
#[1] "41843.00" "71110.00" "41843.00" "71110.00" "95923.00"

sprintf("%x", V)
#[1] "a373"  "115c6" "a373"  "115c6" "176b3"
8
votes

As mentioned in @user4221472's answer, strtoi() overflows with integers larger than 2^31.

The simplest way around that is to use as.numeric().

V <- c(0xa373, 0x115c6, 0x176b3, 0x25cf40000)

as.numeric(V)
#[1]       41843       71110       95923 10149429248
7
votes

strtoi() has a limitation of 31 bits. Hex numbers with the high order bit set return NA:

> strtoi('0x7f8cff8b')
[1] 2139946891
> strtoi('0x8f8cff8b')
[1] NA
0
votes

To get a signed value with 16 bits:

  temp <- strtoi(value, base=16L)
  if (temp>32767){ temp <- -(65535 - temp) }

In a general form:

  max_unsigned <- 65535 #0xFFFF
  max_signed <- 32767 #0x7FFF
  temp <- strtoi(value, base=16L)
  if (temp>max_signed){ temp <- -(max_unsigned- temp) }