我正在尝试分析2012-2013 NATS调查中的数据from this location。 zip文件夹中有三个文件,标记为2012-2013 NATS format.sas,formats.sas7bcat和nats2012.sas7bdat。第三个文件包含实际数据,但第二个文件包含与数据一起使用的标签;也就是说,例如,如果原始数据文件中的变量“Race”具有类别1,2,3和4,则标签显示这些类别代表“高加索人”,“非裔美国人”,“西班牙裔”和'其他'。 我已经能够使用'sas7bdat'包将sas7bdat文件导入到R中,但是当我尝试进行交叉制表时,我无法看到每个单元格代表哪个类别。例如,如果我尝试这样做:
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
据我所知,将标签包含在数据中的唯一方法是手动输入数据,但是有240个变量,此外,还有format.sas7bcat文件形式的标签。有没有办法将格式文件导入R,以便标签可以附加到变量?这是在SAS中完成的,但我现在没有访问权限。感谢您的帮助。
答案 0 :(得分:6)
这应该是一个单行:
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
使用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
您可以轻松地将其写入函数以便更常用:
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
并适用于整个数据集
sas[] <- lapply(sas, do_fmt)
sas$SMOKSTATUS_R[1:4]
# [1] "Never smoker" "Former smoker" "Former smoker" "Never smoker"
虽然有时会失败如下。这似乎与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
因此,您可以使用一些简单的正则表达式来解析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])))
})
例如,烟雾类型格式(其中一个失败的格式)会被解析为:
sp['A5_']
# $A5_
# 'INAPPLICABLE' 'REFUSED' 'DK'
# "-1" "-7" "-8"
# 'NOT ASCERTAINED' 'PREMADE CIGARETTES' 'ROLL-YOUR-OWN' 'BOTH'
# "-9" "1" "2" "3"
然后您可以再次使用该功能来应用数据
table(do_fmt(sas['SMOKTYPE'], sp['A5_']))
# 'BOTH' 'DK' 'INAPPLICABLE'
# 736 17 51857
# 'PREMADE CIGARETTES' 'REFUSED' 'ROLL-YOUR-OWN'
# 7184 2 396
答案 1 :(得分:1)
formats.sas
文件应该是可读的并且可以解析为列标签向量,然后您可以像应用任何列标签向量一样应用它们。
如果你想要根据你的问题标记分类变量,这可能是你最关心的,这应该是相当简单的。您将看到如下代码:
value RACEF
1 = 'Caucasian'
2 = 'African-American'
3 = 'Hispanic'
4 = 'Other'
;
您只需要将其解析为向量。
如果你很幸运,他们的类别格式名称将与列名相同(可能与我在该示例中的F一样);如果是这样的话,你可能只是想知道如何直接应用它们。
如果不是,那么你必须解析程序的后半部分。它将由以下行组成:
format
race RACEF.
gender SEXF.
income INCRF.
...
;
当然,这显示了列名和格式名之间的关系,因此告诉您应该使用哪个列名矢量来标记哪个列。