假设我们有一个数字1.000633,我想计算小数点后的零数,直到分数中的第一个非零数字,答案应为3.对于0.002,答案应为2。
R中没有这样的功能可以提供帮助。我已经在包装DescTools中探索了Ndec函数,但它没有完成这项工作。
答案 0 :(得分:23)
使用regexpr
及其match.length
参数
attr(regexpr("(?<=\\.)0+", x, perl = TRUE), "match.length")
答案 1 :(得分:13)
这是另一种可能性:
zeros_after_period <- function(x) {
if (isTRUE(all.equal(round(x),x))) return (0) # y would be -Inf for integer values
y <- log10(abs(x)-floor(abs(x)))
ifelse(isTRUE(all.equal(round(y),y)), -y-1, -ceiling(y))} # corrects case ending with ..01
示例:
x <- c(1.000633, 0.002, -10.01, 7.00010001, 62.01)
sapply(x,zeros_after_period)
#[1] 3 2 1 3 1
答案 2 :(得分:10)
我们可以使用sub
ifelse(grepl("\\.0", str1),
nchar(sub("[^\\.]+\\.(0+)[^0]+.*", "\\1", str1)), NA)
#[1] 3 2 3 3 2
或使用stringi
library(stringi)
r1 <- stri_extract(str1, regex="(?<=\\.)0+")
ifelse(is.na(r1), NA, nchar(r1))
#[1] 3 2 3 3 2
只是为了检查它是否适用于任何奇怪的情况
str2 <- "0.00A-Z"
nchar(sub("[^\\.]+\\.(0+)[^0]+.*", "\\1", str2))
#[1] 2
str1 <- as.character(c(1.000633, 0.002, 0.000633,
10.000633, 3.0069006))
答案 3 :(得分:7)
使用rle
功能:
#test values
x <- c(0.000633,0.003,0.1,0.001,0.00633044,10.25,111.00012,-0.02)
#result
sapply(x, function(i){
myNum <- unlist(strsplit(as.character(i), ".", fixed = TRUE))[2]
myNumRle <- rle(unlist(strsplit(myNum, "")))
if(myNumRle$values[1] == 0) myNumRle$lengths[1] else 0
})
#output
# [1] 3 2 0 2 2 0 3 1
答案 4 :(得分:6)
使用str_count
包中的stringr
的另一种方法,
x <- as.character(1.000633)
str_count(gsub(".*[.]","",x), "0")
#[1] 3
编辑:这会计算小数后的所有零,直到第一个非零值。
y <- c(1.00215, 1.010001, 50.000809058, 0.1)
str_count(gsub(".*[.]","",gsub("(?:(0+))[1-9].*","\\1",as.character(y))),"0")
#[1] 2 1 3 0
答案 5 :(得分:4)
floor( -log10( eps + abs(x) - floor( abs( x ) ) ) )
答案 6 :(得分:0)
您可以使用sub
,因为我们不需要跳转。因此不需要gsub
nchar(sub(".*\\.(0*).*","\\1",str1))
[1] 3 2 3 3 2
,其中
str1 <- as.character(c(1.000633, 0.002, 0.000633,
10.000633, 3.0069006))