bioinfoer

找回密码
立即注册
搜索
热搜: 活动 交友 discuz
发新帖

0

收听

12

听众

390

主题
发表于 2025-4-11 17:22:33 | 查看: 496| 回复: 1

背景

画出文章PMID: 24239208里的nomogram图和对比曲线

画出新型nomogram

应用场景

列线图(nomogram,诺莫图)是在平面直角坐标系中用一簇互不相交的线段表示多个独立变量之间函数关系的图。

将Logistic回归或Cox回归的结果进行可视化呈现,给出其发病风险或比例风险。

输入数据

至少要有临床信息 easy_input.csv,还可以加入lasso部分输出的 lasso_output.txt信息等等。

pbc<-read.table("easy_input.txt")

pbc$catbili <- cut(pbc$bili,breaks=c(-Inf, 2, 4, Inf),
                   labels=c("low","medium","high"))
pbc$died <- pbc$status==2

head(pbc)

pbc前几行如下:

开始画nomogram

这里提供两种风格的nomogram画法,用到不同的包

经典版

#install.packages("rms")
library(rms)

dd<-datadist(pbc)
options(datadist="dd")
options(na.action="na.delete")
summary(pbc$time)
coxpbc<-cph(formula = Surv(time,died) ~  age + catbili + sex + copper + stage + trt ,data=pbc,x=T,y=T,surv = T,na.action=na.delete)  #,time.inc =2920

print(coxpbc)
surv<-Survival(coxpbc) 
surv3<-function(x) surv(1825,x)
surv4<-function(x) surv(2920,x)

x<-nomogram(coxpbc,fun = list(surv3,surv4),lp=T,
            funlabel = c('5-year survival Probability','8-year survival Probability'),
            maxscale = 100,fun.at = c(0.95,0.9,0.8,0.7,0.6,0.5,0.4,0.3,0.2,0.1))

pdf("nomogram_classical.pdf",width = 12,height = 10)
plot(x, lplabel="Linear Predictor",
     xfrac=.35,varname.label=TRUE, varname.label.sep="=", ia.space=.2, 
     tck=NA, tcl=-0.20, lmgp=0.3,
     points.label='Points', total.points.label='Total Points',
     total.sep.page=FALSE, 
     cap.labels=FALSE,cex.var = 1.6,cex.axis = 1.05,lwd=5,
     label.every = 1,col.grid = gray(c(0.8, 0.95)))
dev.off()

输出公式

#print(x)
#install.packages("nomogramEx")
library(nomogramEx)
nomogramEx(nomo=x,np=2,digit = 9)
# $RESULT
# [1] "The equation of each variable as follows:"
# 
# [[2]]
# [1] "points = 1.720070183 * age + -43.001754587"
# 
# [[3]]
# 
# [[4]]
# 
# [[5]]
# [1] "points = 0 * copper ^3 + 0 * copper ^2 + 0.166666667 * copper + 0"
# 
# [[6]]
# 
# [[7]]
# 
# [[8]]
# [1] "5-year survival Probability = 5.9e-08 * points ^3 + -5.1054e-05 * points ^2 + 0.008126192 * points + 0.583874866"
# 
# [[9]]
# [1] "8-year survival Probability = 5.9e-08 * points ^3 + -4.3812e-05 * points ^2 + 0.004242289 * points + 0.835039968"

新型nomogram

regplot可以交互式调整nomogram,代码更简单,输出的图形更漂亮。

regplot那段代码粘贴到Console里,回车,就会在Plots窗口出现图。

鼠标点击栏目上方蓝色按钮,可以选择变量展示的类型:箱图、小提琴图、密度曲线图等;需要什么图,导出PDF进行AI修改即可。

鼠标点击Esc或按键盘上的Esc可以退出交互模式

#install.packages("regplot","vioplot","sm","beanplot")
library(survival)
pbccox <- coxph(formula = Surv(time,died) ~  age + catbili + sex + 
                   copper + stage + trt , data = pbc)

library(regplot)

pdf("nomogram_new.pdf")
regplot(pbccox,
        observation = pbc[2,],  # 展示第2行观测值的计分
        failtime = c(1095, 1825),  # 预测3年(1095天)和5年(1825天)风险
        prfail = TRUE,  # Cox模型需设为TRUE以表示累积风险
        showP = TRUE,   # 显示P值显著性
        droplines = FALSE,  # 关闭观测值计分垂直线
        dencol = "#A6CEE3",  # 连续变量分布填充色
        boxcol = "#1F78B4",  # 分类变量分布填充色
        obscol = "#33adff",      # 观测值标记颜色
        spkcol = "#2166AC",
        rank = "sd",         # 按变量系数标准差排序sd
        interval = "confidence"  # 展示置信区间
)
dev.off()
# clickable=TRUE  #设置允许交互
regplot(pbccox,
        observation = pbc[2,],  # 展示第2行观测值的计分
        failtime = c(1095, 1825),  # 预测3年(1095天)和5年(1825天)风险
        prfail = TRUE,  # Cox模型需设为TRUE以表示累积风险
        showP = TRUE,   # 显示P值显著性
        droplines = FALSE,  # 关闭观测值计分垂直线
        dencol = "#A6CEE3",  # 连续变量分布填充色
        boxcol = "#1F78B4",  # 分类变量分布填充色
        obscol = "#33adff",      # 观测值标记颜色
        spkcol = "#2166AC",
        rank = "sd",         # 按变量系数标准差排序sd
        interval = "confidence",  # 展示置信区间
        clickable=TRUE  #设置允许交互
)

展示逻辑回归,支持"lm", "glm", "coxph", "survreg" "negbin"

## Plot a logistic regression, showing odds scale and confidence interval
pbcglm <- glm(died ~  age + catbili + sex + copper, family = "binomial", data=pbc )

regplot(pbcglm, 
        observation=pbc[1,], 
        odds=TRUE, 
        interval="confidence")

绘制calibration curve进行验证

采用和nomogram一样的变量进行多因素cox回归

5年

f5<-cph(formula = Surv(time,died) ~  age + catbili + sex + copper +stage + trt,data=pbc,x=T,y=T,surv = T,na.action=na.delete,time.inc = 1825) 

#参数m=50表示每组50个样本进行重复计算
cal5<-calibrate(f5, cmethod="KM", method="boot",u=1825,m=50,B=1000) 

pdf("calibration_5y.pdf",width = 8,height = 8)
plot(cal5,
     lwd = 2,#error bar的粗细
     lty = 1,#error bar的类型,可以是0-6
     errbar.col = c("#2166AC"),#error bar的颜色
     xlim = c(0,1),ylim= c(0,1),
     xlab = "Nomogram-prediced OS (%)",ylab = "Observed OS (%)",
     cex.lab=1.2, cex.axis=1, cex.main=1.2, cex.sub=0.6) #字的大小
lines(cal5[,c('mean.predicted',"KM")], 
      type = 'b', #连线的类型,可以是"p","b","o"
      lwd = 2, #连线的粗细
      pch = 16, #点的形状,可以是0-20
      col = c("#2166AC")) #连线的颜色
mtext("")
box(lwd = 1) #边框粗细
abline(0,1,lty = 3, #对角线为虚线
       lwd = 2, #对角线的粗细
       col = c("#224444")#对角线的颜色
       ) 
dev.off()

8年

f8<-cph(formula = Surv(time,died) ~  age + catbili + sex + copper +stage + trt,data=pbc,x=T,y=T,surv = T,na.action=na.delete,time.inc = 2920) 
cal8<-calibrate(f8, cmethod="KM", method="boot",u=2920,m=50,B=1000)

plot(cal8,
     lwd = 2,
     lty = 1,
     errbar.col = c("#B2182B"),
     xlim = c(0,1),ylim= c(0,1),
     xlab = "Nomogram-prediced OS (%)",ylab = "Observed OS (%)",
     col = c("#B2182B"),
     cex.lab=1.2,cex.axis=1, cex.main=1.2, cex.sub=0.6)
lines(cal8[,c('mean.predicted',"KM")],
      type= 'b',
      lwd = 2,
      col = c("#B2182B"),
      pch = 16)
mtext("")
box(lwd = 1)
abline(0,1,lty= 3,
       lwd = 2,
       col =c("#224444"))

收藏回复 显示全部楼层 道具 举报

发表于 2025-4-11 17:22:49

同时展示两条curve

pdf("calibration_compare.pdf",width = 8,height = 8)
plot(cal5,lwd = 2,lty = 0,errbar.col = c("#2166AC"),
     bty = "l", #只画左边和下边框
     xlim = c(0,1),ylim= c(0,1),
     xlab = "Nomogram-prediced OS (%)",ylab = "Observed OS (%)",
     col = c("#2166AC"),
     cex.lab=1.2,cex.axis=1, cex.main=1.2, cex.sub=0.6)
lines(cal5[,c('mean.predicted',"KM")],
      type = 'b', lwd = 1, col = c("#2166AC"), pch = 16)
mtext("")

plot(cal8,lwd = 2,lty = 0,errbar.col = c("#B2182B"),
     xlim = c(0,1),ylim= c(0,1),col = c("#B2182B"),add = T)
lines(cal8[,c('mean.predicted',"KM")],
      type = 'b', lwd = 1, col = c("#B2182B"), pch = 16)

abline(0,1, lwd = 2, lty = 3, col = c("#224444"))

legend("topleft", #图例的位置
       legend = c("5-year","8-year"), #图例文字
       col =c("#2166AC","#B2182B"), #图例线的颜色,与文字对应
       lwd = 2,#图例中线的粗细
       cex = 1.2,#图例字体大小
       bty = "n")#不显示图例边框
dev.off()

回复 显示全部楼层 道具 举报

您需要登录后才可以回帖 登录 | 立即注册

QQ|Archiver|手机版|小黑屋|bioinfoer ( 萌ICP备20244422号 )

GMT+8, 2025-10-22 23:50 , Processed in 0.087935 second(s), 34 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表