非常详细的GEO数据下载和清理,你要看吗-GSE33532

来自数据集GSE33532,对于第一步获得表达矩阵,做了还蛮详尽的注解过程,希望能对其他人有帮助。虽然看上去全是黑色代码快,但注解理解示例全在这里面了哦!

rm(list = ls())  ## 魔幻操作,一键清空~
options()$repos 
options()$BioC_mirror
options(BioC_mirror="https://mirrors.ustc.edu.cn/bioc/")
options("repos" = c(CRAN="https://mirrors.tuna.tsinghua.edu.cn/CRAN/"))
options(stringsAsFactors = F)#在调用as.data.frame的时,将stringsAsFactors设置为FALSE可以避免character类型自动转化为factor类型


##################     1. 下载表达矩阵和临床信息         ###################
##################  得到表达矩阵dat和分组信息group_lis   ##################
f='GSE33532_eSet.Rdata'
# https://www.ncbi.nlm.nih.gov/geo/query/acc.cgi?acc=GSE33532
library(GEOquery)
# 这个包需要注意两个配置,一般来说自动化的配置是足够的。
#Setting options('download.file.method.GEOquery'='auto')
#Setting options('GEOquery.inmemory.gpl'=FALSE)
if(!file.exists(f)){
  gset <- getGEO('GSE33532', destdir=".",
                 AnnotGPL = F,     ## 注释文件
                 getGPL = F)       ## 平台文件
  save(gset,file=f)   ## 保存到本地
}
load('GSE33532_eSet.Rdata')  ## 载入数据
class(gset)  #查看数据类型
length(gset)  #
class(gset[[1]])
gset
# 因为这个GEO数据集只有一个GPL平台,所以下载到的是一个含有一个元素的list
a=gset[[1]] #
dat=exprs(a) #a现在是一个对象,取a这个对象通过看说明书知道要用exprs这个函数
dim(dat) 

dat[1:4,1:4] #查看dat这个矩阵的1至4行和1至4列,逗号前为行,逗号后为列
pd=pData(a)



tumor=rownames(pd[grepl('tumor',as.character(pd$title)),])
normal=rownames(pd[grepl('normal',as.character(pd$title)),])

dat=dat[,c(tumor,normal)]
group_list=c(rep('tumor',length(tumor)),
             rep('normal',length(normal)))
table(group_list)
dim(dat)
dat[1:4,1:4] 
dat <- as.data.frame(dat)
save(dat,group_list,file = 'step1.1_dat_group.Rdata')




##################   2.获得注释信息   ##################
##################       三种方法     ##################
#1.用getGEO下载平台GPL的注释信息
#2.在GEO网站上下载好平台注释信息后倒入R studio
#3.用bioconductor上的包


#方法一 用getGEO下载平台GPL的注释信息
if(F){
  library(GEOquery)
  gpl<- getGEO('GPL570', destdir=".")
  dim(gpl)
  colnames(Table(gpl)) #查一下列明
  head(Table(gpl)[,c(1,11)]) 
  ids=Table(gpl)[,c(1,11)]
  ##踩坑放松1:会不会有NA值,检查一下,没问题就不管那些个警告了
  tmpids <- na.omit(ids)
  dim(tmpids)
  head(ids)
  save(ids,file='step1.2_ids.Rdata') #三种方法均获得的ids均设置的是'step1.2_ids.Rdata'
}
rm(tmpids)


#方法二 在GEO网站上下载好平台注释信息后导入Rstudio
if(F){
  gpl=read.table('GPL570-55999.txt', #提前下载好了平台GPL570的注释信息
               sep = '\t',quote = '',
               fill=T,header = T)
  dim(gpl)
  colnames(gpl)
  ids=gpl[,c(1,11)]
  ids=ids[ids$Gene.Symbol != '',] #去除空的
  dim(ids)
  head(ids) 
  save(ids,file='step1.2_ids.Rdata')  #三种方法均获得的ids均设置的是'step1.2_ids.Rdata'
}


#方法三 用bioconductor网站上对应平台的基因注释包
#https://mp.weixin.qq.com/s/5_ceRNaRFP3h08Pjd015Wg
if(T){
  library(hgu133plus2.db)
  ls("package:hgu133plus2.db")
  ids<- toTable(hgu133plus2SYMBOL)
  dim(ids)
  head(ids)
  save(ids,file='step1.2_ids.Rdata')  #三种方法均获得的ids均设置的是'step1.2_ids.Rdata'
}

################三种获得注释信息的方法,所得到的对应的行数如下:
#> dim(gpl)
#[1] 54675    16
#> dim(ids)
#[1] 45782     2
#> dim(ids)
#[1] 41932     2



##################        3. ID转换         ##################
##################   intersect/merge/%in%   ##################
##################         match函数        ##################
##################  (1)用intersect函数    ##################
rm(list = ls())
load('step1.1_dat_group.Rdata')
load('step1.2_ids.Rdata') #此处按照第三种从bioconductor获得的注释包来进行id转换
dim(dat)
dim(ids)
table(rownames(dat) %in% ids$probe_id)
intersect(rownames(dat),ids$probe_id) 
length(intersect(rownames(dat),ids$probe_id))
#xt意味着xt(相同)的
xt <- intersect(rownames(dat),ids$probe_id)
dim(xt) 
##踩坑放松1:为什么用dim是得到NULL?

length(xt)######xt现在是一个character,所以用length来查看向量的长度
head(xt)
ids_tmp <- ids[ids$probe_id==xt,]#注:想将ids按照取交集后的向量(xt)从ids中取交集
##踩坑放松2:报错-长的对象长度不是短的对象长度的整倍数

length(ids$probe_id)
length(xt)

###tips1:上面会提示报错,因为当向量用'=='进行判断的时候,就是将两个向量中的元素进行一对一地判断是否相同,之所以报‘长的对象长度不是短的对象长度的整倍数’的报错,是因为ids$probe_id有ids的行数那么多(41932个),而xt则是取交集后的23738个。
###tips2:接下来用match函数取子集,就是按照xt的顺序从大子集ids中取,那么取出来的探针的顺序肯定是和xt的顺序是一样的,然后再从dat中同样按照xt的顺序来取子集,因为ids和dat都是按照xt中的探针的顺序取子集,那么二者的dat中的探针名肯定和ids中的探针名是一一对应的了。


####################  重磅嘉宾-match函数第一次出场  ####################

ids <- ids[match(xt,ids$probe_id),]##ids同样按照probe_id这一列提取子集,是按照坐标位置提取
#用identical函数和setequal函数验证一下(这两个函数都可以),是不是完全一样,也就是说表达矩阵dat的探针名和交集(xt)是否是一一对应的
identical(ids$probe_id,xt)
setequal(ids$probe_id,xt)
save(ids,file='step1.3_ids.Rdata')
  

####################  (2)用merge函数    ##################
rm(list = ls())
load('step1.1_dat_group.Rdata')
load('step1.2_ids.Rdata')
ids<- toTable(hgu133plus2SYMBOL)
dat_tmp <- dat #重新赋值一个dat_tmp
ids_tmp <- ids #重新赋值一个ids_tmp
dim(dat_tmp)
dim(ids_tmp)
#用新赋值后的做探索
dat_tmp <- dat_tmp[ids_tmp$probe_id,] #注:想从dat_tmp表达矩阵中按照ids_tmp$probe_id取出的探针名提取子集,dat_tmp的rownames就是探针名,表面看是没问题呀

##踩坑放松3: dat_tmp <- dat_tmp[ids_tmp$probe_id,]无报错,表面看似风平浪静,背后其实翻江倒海,用dim查看大小
dim(dat_tmp)  #注:刚才的dat_tmp只有25906个探针,现在竟然和ids_tmp的探针名一样多的个数!表达矩阵竟然变大了!
is.na(dat_tmp) #注:我就想看个行名是不是NA,结果在表达矩阵中返回了TRUE和FALSE,所以又学一了is.na函数
table(is.na(rownames(dat_tmp))) #注:还是不对,改成行名也不对。
dat_tmp[1:4,1:4]
is.na(dat_tmp)#注:观察在dat_tmp中为TRUE也就是为NA值的,行名也都是NA,因此用grepl函数抓取下
table(grepl('NA',rownames(dat_tmp))) #注意区分grep和grepl函数的差别。注意此时的结果
#FALSE  TRUE 
#23738 18194
#注意观察:上面的FALSE正好为23738个,同前面用intersect函数获得的ids是不是行数相同?
dat_tmp <- na.omit(dat_tmp) #注:用na.omit去掉NA值
dim(dat_tmp) 

##为用merge函数作准备-dat新建一列,列名是probe
dat$probe_id <- rownames(dat) #注:dat新建一列,列名是probe,此步是为了后面用merge函数
dim(dat)
####小提问:如何知道dat中第101列的列名
colnames(dat[,101]) #注:dat新建probe这一列后,dat现在有101列,我想看看这第101例的列名




##踩坑放松4:colnames(dat[,101])为什么返回NULL,如何获得第101列的列名?
#分解代码:先看dat[,101]
dat[,101]

##报错原因:把上面的代码分解,dat[,101],根据“行在左,列在右”,此时101在“,”右边,因此其实取出的是第101这列的所有的探针名,是一个向量,一个长度同行数相等的向量。向量取子集怎么能用colnanmes来提取呢?
#正确操纵如下:
colnames(dat)
colnames(dat)[101]#注,从向量中取子集,可以按照逻辑值和坐标位置,现在我们想取坐标位置为第101位的内容


#关于merge函数,依然用前两步得到的dat和ids
ids_dat_merge <- merge(ids,dat,by.x ='probe_id',by.y='probe_id')

##踩坑放松5:
#(1)merge函数:dat新建的一列,如果列名是probe,也就是说列名不一样,可不可以?
#(2)merge函数:by.x=probe,by.y=probe_id,怎么改?出来的矩阵是什么样的?
#(3)merge函数:如果想要merge的那一列的向量顺序不一样,merge后的矩阵是怎样的,有什么关系?


#下面是merge函数小理解,新建模拟ids_tmp和dat_tmp
if(F){
  #第1、2个merge问题
  ids_tmp <- data.frame(probe_id = c('1053_at','117_at','121_at','1255_g_at'),
                        symbols = c('RFC2','HSPA6','PAX8','GUCA1A'))
  dat_tmp <- data.frame(row.names = c('1053_at','117_at','121_at','1316_at','1320_at'),
                        GSM835268=c(1,1,1,1,1),
                        GSM835269=c(2,2,2,2,2),
                        probe = c('1053_at','117_at','121_at','1316_at','1320_at'))
  ids_tmp
  dat_tmp
  merge1 <- merge(ids_tmp,dat_tmp,by.x='probe_id',by.y='probe')
  merge1
  merge2 <- merge(dat_tmp,ids_tmp,by.x='probe',by.y='probe_id')
  merge2
  
  #第3个merge问题,如果要merge的探针名顺序不一样
  ids_tmp <- data.frame(probe_id = c('1053_at','117_at','121_at','1255_g_at'),
                        symbols = c('RFC2','HSPA6','PAX8','GUCA1A'))
  dat_tmp <- data.frame(row.names = c('121_at','117_at','1053_at','1316_at','1320_at'),
                        GSM835268=c(1,1,1,1,1),
                        GSM835269=c(2,2,2,2,2),
                        probe = c('121_at','117_at','1053_at','1316_at','1320_at'))
  ids_tmp
  dat_tmp
  merge3 <- merge(ids_tmp,dat_tmp,by.x='probe_id',by.y='probe')
  merge3
  merge4 <- merge(dat_tmp,ids_tmp,by.x='probe',by.y='probe_id')
  merge4
  #从merge3、merge4相同的那列顺序没有改变,这是最后的矩阵中dat和ids顺序有差别
}



dim(ids_dat_merge) #注:检查数据,dim看大小
ids_dat_merge[1:4,1:4] #注:取子集看数据情况
ids_dat_merge <- na.omit(ids_dat_merge) #注:检查下是否有NA值,注意理解merge函数,通过merge函数获得ids_dat_merge是ids和dat中*都都都*存在的元素。思考,如果想得到merge(x,y)后,在x中有不存在与y中的元素也出现在merge后的矩阵中,用什么参数?
dim(ids_dat_merge)
ids_dat_merge####用$符号看一下,现在的表达矩阵有哪些列(列名)按完$健,按tab健

##接下来进行从ids中按照merge后的ids_dat_merge表达矩阵中按照探针名字,从ids中取子集
ids_tmp <- ids[ids_dat_merge$probe_id,] #注:我已经得到了dat和ids共有列探针(probe)merge后的表达矩阵了,想按照merge后的ids_dat_merge中的probe_id这一列提取探针名字,然后将ids按照这个探针名字提取子集从而想生成一个ids_tmp,从而获得和表达表达矩阵的探针名相同的ids注释!
dim(ids_tmp)
ids_tmp[1:4,1:2]#注:一看数据具体情况,竟然都是NA!

#踩坑放松6:从数据框中提取子集,可以按rownames也就是行名,也可以按照坐标,可是为什么都是NA呢?


dim(ids)
ids[1:4,1:2]
#答:ids是行名为1、2、3、4...,并不是探针名!从ids中取子集时,要么按照探针名(行名),要么按照坐标位置,要么按照逻辑值,就这三种,三种,三种!重要的事情说三遍哦!而ids_dat_merge$probe_id后得到的是探针名probe_id,而ids的行名又不是探针名probe_id,而是1、2、3、4...,所以此处取出的都是NA!

#既然不是,那我把ids的行名变成探针名probe_id不就好了嘛!
rownames(ids_tmp) <- ids_dat_merge$probe_id 
##看下大小
dim(ids_tmp)
ids_tmp[1:4,1:2] #注:虽然重新复制了探针名,但是需要注意此处的ids_tmp由于上一步取子集的错误操作,现在都是NA,所以需要重新获得一个ids_tmp.


####################  重磅嘉宾-match函数第二次出场  ####################

ids_tmp <- ids[match(ids_dat_merge$probe_id,ids$probe_id),] 
#注:上面的match函数,从ids$probe_id的探针名中按照ids_dat_merge$probe_id在ids$probe_id中的排名位置,也就是坐标位置,从ids中按照这个坐标位置取子集,得到ids_tmp
######下面是match函数小理解
if(T){
  use_id <- c('A','B','C','D')
  use_id
  m<-data.frame(u=c('B','C','D','A'))
  match(use_id,m$u)
  a<-m[match(use_id,m$u),]
  a
}

dim(ids_tmp)
ids_tmp[1:4,1:2]
ids_tmp <- na.omit(ids_tmp)#注:被NA弄怕了,再用na.omit检查下,match函数用熟练或用对了,就可省略
ids <- ids_tmp
save(ids,file='step1.3_ids.Rdata')

######################    此处小贴士    ###################
#其实前面的到的ids_dat_merge矩阵中已经含有基因symbol这一列了,是不是可以就把矩阵中的symbol变成行名就可以了呢?不就已经完成id转换了嘛?那试一下

ids_dat_merge[1:4,1:4]
rownames(ids_dat_merge) <- ids_dat_merge$symbol

##踩坑放松8:rownames(ids_dat_merge) <- ids_dat_merge$symbol 报错,不允许有重复的‘row.names’,因此基因名symbol需要去除重复!


####################  (3)用%in%函数    ##################

rm(list = ls())
load('step1.1_dat_group.Rdata')
load('step1.2_ids.Rdata')
dim(dat)
dim(ids)
ids <- ids[ids$probe_id %in% rownames(dat),]#注:想从dat的rownames中提取出ids$symbol后存在于rownames(dat)中的元素,%in%是进行判断,逻辑值为TRUE的,从ids中提取出来。
dim(ids)
ids[1:4,1:2]#注:此时为什么是ids[1:4,1:2],如果用ids[1:4,1:4]会报错undefined columns selected,是因为ids本身就只有两列。
##踩坑放松9:此处是温馨小提示:同前面踩坑放松6是类似的道理,虽然现在ids的行名是1、2、3、4,但是%in%是进行判断,得到的是逻辑值,所以不用管行名是什么,我们就是按照TRUE的提取!三种提取子集的方法:名字,坐标,逻辑值!


#下面是%in%函数小理解,新建模拟ids_tmp和dat_tmp.
if(F){
  ids_tmp <- data.frame(probe_id = c('1053_at','117_at','121_at','1255_g_at'),
                  symbols = c('RFC2','HSPA6','PAX8','GUCA1A'))
  dat_tmp <- data.frame(row.names = c('1053_at','117_at','121_at','1316_at','1320_at'),
                        GSM835268=c(1,1,1,1,1),
                        GSM835269=c(2,2,2,2,2))
  ids_tmp
  dat_tmp
  m <- ids_tmp[ids_tmp$probe_id %in% rownames(dat_tmp),]
  m
  n <- dat_tmp[rownames(dat_tmp) %in% ids_tmp$probe_id,]
  n
  #如果探针名顺序不一样
  ids_tmp <- data.frame(probe_id = c('1053_at','117_at','121_at','1255_g_at'),
                        symbols = c('RFC2','HSPA6','PAX8','GUCA1A'))
  dat_tmp <- data.frame(row.names = c('121_at','1316_at','1053_at','117_at','1320_at'),
                        GSM835268=c(1,1,1,1,1),
                        GSM835269=c(2,2,2,2,2))
  #注意上面dat_tmp中的row.names的顺序。
  ids_tmp
  dat_tmp
  m <- ids_tmp[ids_tmp$probe_id %in% rownames(dat_tmp),]
  m
  n <- dat_tmp[rownames(dat_tmp) %in% ids_tmp$probe_id,]
  n
  #上面探针名顺序不一样,可以帮助理解取子集,从哪个矩阵取,就按哪个矩阵中的逻辑值取
}


identical(ids$probe_id,rownames(dat))
ids[1:4,1:2]
dat[1:4,1:4]

####################  重磅嘉宾-match函数第三次出场  ####################

dat <- dat[match(ids$probe_id,rownames(dat)),]#注:ids的probe_id是全部存在于dat的rownames里的,因此用match函数从dat中按照ids的顺序提取子集,一定是能得到探针完全对应的。此处就是按照坐标在提取子集,这个坐标指的就是a在b中的顺序,注意,如果数据框的行名是1、2、3、4,与这个1、2、3、4无关,仅仅就是a中的元素一个一个试验在b中排第几位,比如a中的元素依次在b中的元素进行一一对应,排第1位记个1,排第5位记个5,排第10位记个10,然后从b中按照这个1、5、10提取b中的元素。所以从b中提取的元素就是和a完全相同的。

save(ids,file = 'step1.3_ids.Rdata')#注,上面的dat取子集是按照坐标位置提取的,此处并没有保存dat,但是要知道此出dat用match函数提取子集的方法。


#######################  4.将探针名转换为基因名  ################
######################    基因名symbol去重复 ######################
rm(list = ls())
load('step1.1_dat_group.Rdata')
load(file='step1.3_ids.Rdata')
dim(dat)
dat[1:4,1:4]
dim(ids)
ids[1:4,1:2]
dat <- dat[ids$probe_id,]#注:从dat表达矩阵(其实是数据框,就是这么叫一下表达矩阵)中按照ids$probe_id也就是探针名字,提取子集。

######下面是数据框提取子集的小理解,按照rownames的名字或者坐标位置均可以提取子集,示例如下:
  if(F){ 
    x <- data.frame(row.names = c('a','b','c'),xm = c('xh','xl','xh'),nl=c(15,18,21))
    x
    x['a',]
    x[1,]
    identical(x['a',],x[1,])
  }

dim(dat)
dim(ids)
ids[1:4,1:2]
#同理,看看dat和ids能不能一一对应
identical(rownames(dat),ids$probe_id)
setequal(rownames(dat),ids$probe_id)
rownames(dat) <- ids$symbol
  #再踩坑一次!这里rownames(dat) <- ids$symbol #这个时候由于rownames的symbol有重复的,会报错,所以先需要进行去重复,还是和上面相同的报错,再一次提醒我们:不允许有重复的'row.names',因此需要基因名去重复


##################基因名symbol去重复如下:
if(T){
  ids$median=apply(dat,1,median) #ids新建median这一列,列名为median,同时对dat这个矩阵按行操作,取每一行的中位数,将结果给到median这一列的每一行
  ids=ids[order(ids$symbol,ids$median,decreasing = T),]#对ids$symbol按照ids$median中位数从大到小排列的顺序排序,将对应的行赋值为一个新的ids
  ids=ids[!duplicated(ids$symbol),]#将symbol这一列取取出重复项,'!'为否,即取出不重复的项,去除重复的gene ,保留每个基因最大表达量结果s
  dim(ids)
  ids[1:3,1:3]
  dat=dat[ids$probe_id,] #新的ids取出probe_id这一列,将dat按照取出的这一列中probe_id所在的每一行组成一个新的dat,这时获得的dat的探针名就是和ids的probe_id这一列是完全对应的
  rownames(dat)=ids$symbol#把ids的symbol这一列中的每一行给dat,作为dat的rowname
  dim(dat)
  dat[1:4,1:4]
  #identical(rownames(dat),ids$symbol) #可验证,也可不验证,只要能确保 dat=dat[ids$probe_id,]这步是正确的
  boxplot(dat)
  dat=log(dat+1)
  save(dat,ids,group_list,file = 'step1.4_dat_ids_group_list.Rdata')

}


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容

  • 这是去年在日本上映的纪录片,《人生果实》,另外一个译名是《积存时间的生活》这部片子主要讲述了 90岁的男主人公...
    婉彤心语阅读 1,377评论 0 4
  • 心大有的时候并非就真的无所谓 而是对一些事物看的淡了些
    _向暖阅读 111评论 0 0
  • 思考:此文记喜还是志忧?怎样成为文言达人 一通读读文题 1.补充题目 张岱三更独往湖心亭看雪 2.赏读雪景 “与”...
    石默默阅读 248评论 0 0
  • 001 002 004 005 006 008 009 010 011 014 015 017 018 019 0...
    水手_6084阅读 195评论 0 0