我试图从这个位置分析2012-2013 NATS调查的数据.zip文件夹中有三个文件,标记为2012-2013 NATS format.sas,formats.sas7bcat和nats2012.sas7bdat.第三个文件包含实际数据,但第二个文件包含与数据一起使用的标签; 也就是说,例如,如果原始数据文件中的变量"Race"具有类别1,2,3和4,则标签显示这些类别代表"高加索人","非裔美国人","西班牙裔"和'其他'.我已经能够使用'sas7bdat'包将sas7bdat文件导入到R中,但是当我尝试进行交叉制表时,我无法看到每个单元格代表哪个类别.例如,如果我尝试这样做:
table(SMOKSTATUS_R, RACEETHNIC)
Run Code Online (Sandbox Code Playgroud)
我得到的是:
RACEETHNIC
SMOKSTATUS_R 1 2 3 4 5 6 7 8 9
1 4045 455 55 7 63 0 675 393 373
2 1183 222 38 2 26 0 217 255 154
3 14480 957 238 14 95 3 1112 950 369
4 23923 2532 1157 23 147 1 1755 3223 909
5 81 18 4 0 1 0 11 17 9
Run Code Online (Sandbox Code Playgroud)
据我所知,将标签包含在数据中的唯一方法是手动输入数据,但有240个变量,此外,还有format.sas7bcat文件形式的标签.有没有办法将格式文件导入R,以便标签可以附加到变量?这是在SAS中完成的,但我现在没有访问权限.谢谢你的帮助.
这应该是一个单行:
library('haven')
sas <- read_sas('nats2012.sas7bdat', 'formats.sas7bcat')
with(sas, table(SMOKSTATUS_R, RACEETHNIC))
# RACEETHNIC
# SMOKSTATUS_R 1 2 3 4 5 6 7 8 9
# 1 4045 455 55 7 63 0 675 393 373
# 2 1183 222 38 2 26 0 217 255 154
# 3 14480 957 238 14 95 3 1112 950 369
# 4 23923 2532 1157 23 147 1 1755 3223 909
# 5 81 18 4 0 1 0 11 17 9
table(names(attr(sas[, 'SMOKSTATUS_R'], 'labels')[sas[, 'SMOKSTATUS_R']]),
names(attr(sas[, 'RACEETHNIC'], 'labels')[sas[, 'RACEETHNIC']]))
# Amer. Indian, AK Nat. Only, Non-Hispanic
# Current everyday smoker 63
# Current some days smoker 26
# Former smoker 95
# Never smoker 147
# Unknown 1
Run Code Online (Sandbox Code Playgroud)
使用haven在数据读取,但也给你一些有用的attributes,即变量标签:
attributes(sas$SMOKSTATUS_R)
# $label
# [1] "SMOKER STATUS (4-level)"
#
# $class
# [1] "labelled"
#
# $labels
# Current everyday smoker Current some days smoker Former smoker
# 1 2 3
# Never smoker Unknown
# 4 5
#
# $is_na
# [1] FALSE FALSE FALSE FALSE FALSE
Run Code Online (Sandbox Code Playgroud)
您可以轻松地将其写入函数以便更普遍地使用:
do_fmt <- function(x, fmt) {
lbl <- if (!missing(fmt))
unlist(unname(fmt)) else attr(x, 'labels')
if (!is.null(lbl))
tryCatch(names(lbl[match(unlist(x), lbl)]),
error = function(e) {
message(sprintf('formatting failed for %s', attr(x, 'label')),
domain = NA)
x
}) else x
}
table(do_fmt(sas[, 'SMOKSTATUS_R']),
do_fmt(sas[, 'RACEETHNIC']))
# Amer. Indian, AK Nat. Only, Non-Hispanic
# Current everyday smoker 63
# Current some days smoker 26
# Former smoker 95
# Never smoker 147
# Unknown 1
Run Code Online (Sandbox Code Playgroud)
并适用于整个数据集
sas[] <- lapply(sas, do_fmt)
sas$SMOKSTATUS_R[1:4]
# [1] "Never smoker" "Former smoker" "Former smoker" "Never smoker"
Run Code Online (Sandbox Code Playgroud)
虽然有时会失败,如下所示.这看起来像haven包有问题
attr(sas$SMOKTYPE, 'labels')
# INAPPLICABLE REFUSED DK NOT ASCERTAINED
# -4.00000 -0.62500 -0.50000 -0.46875
# PREMADE CIGARETTES ROLL-YOUR-OWN BOTH
# 1.00000 2.00000 3.00000
Run Code Online (Sandbox Code Playgroud)
因此,您可以使用一些简单的正则表达式来解析format.sas文件
locf <- function(x) {
x <- data.frame(x, stringsAsFactors = FALSE)
x[x == ''] <- NA
indx <- !is.na(x)
x[] <- lapply(seq_along(x), function(ii) {
idx <- cumsum(indx[, ii])
idx[idx == 0] <- NA
x[, ii][indx[, ii]][idx]
})
x[, 1]
}
fmt <- readLines('~/desktop/2012-2013-NATS-Format/2012-2013-NATS-Format.sas')
## not sure if comments are allowed in the value definitions, but
## this will check for those in case
fmt <- gsub('\\*.*;|\\/\\*.*\\*\\/', '', fmt)
vars <- gsub('(?i)value\\W+(\\w*)|.', '\\1', fmt, perl = TRUE)
vars <- locf(vars)
regex <- '[\'\"].*[\'\"]|[\\w\\d-]+'
vals <- gsub(sprintf('(?i)\\s*(%s)\\s*(=)\\s*(%s)|.', regex, regex),
'\\1\\2\\3', fmt, perl = TRUE)
View(dd <- na.omit(data.frame(values = vars, formats = vals,
stringsAsFactors = FALSE)))
sp <- split(dd$formats, dd$values)
sp <- lapply(sp, function(x) {
x <- Filter(nzchar, x)
x <- strsplit(x, '=')
tw <- function(x) gsub('^\\s+|\\s+$', '', x)
sapply(x, function(y)
setNames(tw(y[1]), tw(y[2])))
})
Run Code Online (Sandbox Code Playgroud)
因此,例如,烟雾类型格式(其中一个失败的格式)将被解析为:
sp['A5_']
# $A5_
# 'INAPPLICABLE' 'REFUSED' 'DK'
# "-1" "-7" "-8"
# 'NOT ASCERTAINED' 'PREMADE CIGARETTES' 'ROLL-YOUR-OWN' 'BOTH'
# "-9" "1" "2" "3"
Run Code Online (Sandbox Code Playgroud)
然后,您可以再次使用该功能应用于数据
table(do_fmt(sas['SMOKTYPE'], sp['A5_']))
# 'BOTH' 'DK' 'INAPPLICABLE'
# 736 17 51857
# 'PREMADE CIGARETTES' 'REFUSED' 'ROLL-YOUR-OWN'
# 7184 2 396
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
4270 次 |
| 最近记录: |