
R語言:函數使用技巧(循環、if族/for、switch、repeat、ifelse、stopifnot)
1、循環
[plain] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
##循環for
iris
allzl=unique(iris$setosa)
for (i in 1:2){
pp=iris[iris$setosa==allzl[i],]
plot(pp$Sepal.Length~pp$Sepal.Width)
}
for循環中,需要將數值組合起來,如果數據整齊可以用matrix;如果不整齊,用list,不等長合并的時候,rbind.fill函數可以很好將數據進行合并,并且補齊沒有匹配到的缺失值為NA。
案 例
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
temp<-matrix(data = NA,181,31)
for (i in 1:31){
temp[,i]<-filter(data[i]/7, rep(1, 7))
}
yatmdata<-data.frame(temp)
代碼利用matrix先定義一個181*31的空值矩陣,然后往里面灌數字。
2、switch分支語句
[plain] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
##switch分支語句
switch(1,mean(1:10),rnorm(4)) #執行mean(1:10)
switch(2,mean(1:10),rnorm(4)) #執行rnorm(4)
#由switch(x)來選擇執行那個函數
3、while循環語句
注意執行順序,先執行f[i]+f[i+1]<1000,然后往下走,與下面repeat有區別
[plain] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
##while循環語句
#計算斐波那契數列
f=1
f[2]=1
i=1
while(f[i]+f[i+1]<1000){
f[i+2]=f[i]+f[i+1]
i=i+1
}
f
#注意執行順序,先執行f[i]+f[i+1]<1000,然后往下走,與下面repeat有區別
4、repeat循環
常常與if聯用。
[plain] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
##repeat語句
#計算斐波那契數列
f=1
f[2]=1
i=1
repeat{
f[i+2]=f[i]+f[i+1]
i=i+1
if (f[i]+f[i+1]>1000) break
};f
#與if常常聯用,注意執行順序,f[i]+f[i+1]>1000,與while<1000不同
與if常常聯用,注意執行順序,f[i]+f[i+1]>1000,與while<1000不同。
5、if函數+function
if和while都是需要數據TRUE/FALSE這樣的邏輯類型變量,這就意味著,if內部,往往是對條件的判別,例如 is.na, is.matrix, is.numeric等等,或者對大小的比較,如,if(x > 0), if(x == 1), if(length(x)== 3)等等。
if后面,如果是1行,則花括號可以省略,否則就必須要將所有的語句都放在花括號中。這和循環是一致的
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
fun.test <- function(a, b, method = "add"){
if(method == "add") { ## 如果if或者for/while;
res <- a + b ## 等后面的語句只有一行,則無需使用花括號。
}
if(method == "subtract"){
res <- a - b
}
return(res) ## 返回值
}
### 檢驗結果
fun.test(a = 10, b = 8, method = "add")
fun.test(a = 10, b = 8, method = "substract")
同時if還有類似與excel的用法——ifelse
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
ifelse(Age > 30, "Old", "Young")
Age變量>30,則輸出old;<30,輸出Young
————————————————————————————————————————————————————————————
Function與循環函數結合的實踐案例
1、函數如何輸出?——print、return&list
如果是單個輸出,直接用1.3方法即可
如果有很多輸出項目,那么需要return(終止運算,并輸出return中的項目)最終輸出的項目
R中默認的情況是將最后一句作為返回值。
1.1 return&list組合
return和list的組合輸出結果比較合理。(來自R語言︱噪聲數據處理、數據分組——分箱法(離散化、等級化))
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
sbdeep=function(data,parts,xiaoz){
parts<-parts #分幾個箱
xiaoz<-xiaoz #極小值
value<-quantile(data,probs = seq(0,1,1/parts)) #這里以data等比分為4段,步長為1/4
number<-mapply(function(x){
for (i in 1:(parts-1))
{
if(x>=(value[i]-xiaoz)&x<value[i+1])
{
return(i)
}
}
if(x+xiaoz>value[parts])
{
return(parts)
}
return(-1)
},data)
#打標簽L1L2L3L4
return(list(degree=paste("L",number,sep=""),degreevalue=number,value=table(value),number=table(number))) #將連續變量轉化成定序變量,此時為L1,L2,L3,L4...根據parts
}
該函數是對單個序列數據進行等深分箱,可以返回四類:
一個基于L1L2L3....的每個指標標簽序列degree;
標簽序列值degreevalue,
每個百分位數對應的變量值value,
不同百分點的數量number。
1.2 print直接輸出
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
function(){
print(plot(cv.out))
}
print可以直接輸出.
1.3 直接輸出——一一般都是直接輸出
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
function(){
a=c(1:50)
a
}
其中a就是直接寫在末尾,當做輸出項。
2、function中應用if switch函數
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
test=function(mode=c("all", "out", "in")){
mode <- switch(mode, out = 1, `in` = 2, all = 3)
if (as.numeric(mode)==1) {
t=1
}
if (as.numeric(mode)==2) {
t=2
}
if (as.numeric(mode)==3) {
t=3
}
t=t+1
return(t+4)
}
a=test(mode="out")
test(mode="in")
test(mode="all")
解決場景:編寫函數時候,可能嵌套很多模型的時候,就需要用這個流程。
switch函數,輸入mode,執行相應的內容,此時是mode選擇“all”,則執行返回1,;mode選擇"out"則返回2;
然后用if去進行每個數字背后的建模,注意“==”
"in"注意要引號,因為會跟內嵌函數重疊
3 異常值處理——如何報錯
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
# 異常處理,當僅輸入一個數據的時候,告知不能計算標準差
if(length(x) == 1){
stop("can not compute sd for one number,
a numeric vector required.\n")
}
————————————————————————————————————
應用一:if族有哪些成員呢?——if/ifelse/stopifnot
在函數中,if的應用場景非常多,用來識別某類情況前提下,再執行下一個。
其中筆者就見過這樣三類if:if-else ifelse stopifnot
1、if-else
這個很常見,就是需要注意一下,if-else的寫法,來看經管之家論壇一位壇友的提醒與使用心得:
if(){}else{} 表示先執行if括號后面的條件語句,如果正確就執行第一個大括號里的程序,如果錯誤就執行else后面大括號里的語句。
有一種情況,r會報錯:
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
if(){}
else{}
就是這種情況,即else語句換了一行執行時,這是r會認為if語句已經執行完畢,但執行else發現前面無法執行,因此報錯,在這里要提醒使用r的同志們,else必須緊挨著if語句后的大括號,這時才不會出錯。
2、ifelse
跟If-else其實是一模一樣的,但是效率提高很多,是提高代碼運算效率很高的函數。ifelse()的句法格式類似于if()函數,但其運算速度卻有了巨大的提升。即使是在沒有預設數據結構且沒有簡化條件語句的情況下,其運算效率仍高于上述的兩種方法。
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
ifelse(test, yes, no)
ifelse返回的是結果,有一點麻煩的是,不像if-else一樣,可以寫一些分布計算的東西,譬如現在有以下一種情況:
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
a<-c+d
sum(a>2) #在c大于2的情況下,要計算a大于2的個數
這個分步情況在if-else里面很好解決,但是在ifelse里面可不容易,只能接受一步,所以盡量把運算鏈合并在一起。
3、stopifnot
這個函數跟Ifelse有點像,但是很奇特。stopifnot(c>2),如果正確執行,那么就會啥都沒發生,如果錯誤了,就會跳入Debug模式,報錯,讓函數立刻停下來。
這個stopifnot跟trycatch函數聯合使用,威力無比。
用tryCatch跳過:
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
result = tryCatch(
{expr},
warning = function(w) {warning-handler-code},
error = function(e) { error-handler-code},
finally = {cleanup-code}
)
出現warning、error時候怎么處理,就可以跳過了。例子:
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
result = tryCatch(
{segmentCN(txt)},
warning = function(w) {"出警告啦"},
error = function(e) { "出錯啦"},
)
分詞時候,容易因為Lapply中斷之后,就不會運行了,這樣功虧一簣所以可以用這個辦法跳過。
————————————————————————————————————
應用二:如何在循環中,實時輸出時間消耗?
想知道循環中進行到哪里?這樣可以合理安排函數進程。那么怎么辦呢?
第一辦法:使用Rstudio 1.0版本,里面有一個Profiling with profvis,可以很好的對你函數每一步的耗時進行參看。
當然,這個不能實時輸出內容。數據分析師培訓
第二辦法:利用difftime函數
[html] view plain copy
print?在CODE上查看代碼片派生到我的代碼片
t1 = Sys.time()
for (i in 1:5){
a=a+1
b=a*a
print(difftime(Sys.time(), t1, units = 'sec'))
}
先預設當前時間,然后用difftime+print方式,循環輸出。
數據分析咨詢請掃描二維碼
若不方便掃碼,搜微信號:CDAshujufenxi
CDA數據分析師證書考試體系(更新于2025年05月22日)
2025-05-26解碼數據基因:從數字敏感度到邏輯思維 每當看到超市貨架上商品的排列變化,你是否會聯想到背后的銷售數據波動?三年前在零售行 ...
2025-05-23在本文中,我們將探討 AI 為何能夠加速數據分析、如何在每個步驟中實現數據分析自動化以及使用哪些工具。 數據分析中的AI是什么 ...
2025-05-20當數據遇見人生:我的第一個分析項目 記得三年前接手第一個數據分析項目時,我面對Excel里密密麻麻的銷售數據手足無措。那些跳動 ...
2025-05-20在數字化運營的時代,企業每天都在產生海量數據:用戶點擊行為、商品銷售記錄、廣告投放反饋…… 這些數據就像散落的拼圖,而相 ...
2025-05-19在當今數字化營銷時代,小紅書作為國內領先的社交電商平臺,其銷售數據蘊含著巨大的商業價值。通過對小紅書銷售數據的深入分析, ...
2025-05-16Excel作為最常用的數據分析工具,有沒有什么工具可以幫助我們快速地使用excel表格,只要輕松幾步甚至輸入幾項指令就能搞定呢? ...
2025-05-15數據,如同無形的燃料,驅動著現代社會的運轉。從全球互聯網用戶每天產生的2.5億TB數據,到制造業的傳感器、金融交易 ...
2025-05-15大數據是什么_數據分析師培訓 其實,現在的大數據指的并不僅僅是海量數據,更準確而言是對大數據分析的方法。傳統的數 ...
2025-05-14CDA持證人簡介: 萬木,CDA L1持證人,某電商中廠BI工程師 ,5年數據經驗1年BI內訓師,高級數據分析師,擁有豐富的行業經驗。 ...
2025-05-13CDA持證人簡介: 王明月 ,CDA 數據分析師二級持證人,2年數據產品工作經驗,管理學博士在讀。 學習入口:https://edu.cda.cn/g ...
2025-05-12CDA持證人簡介: 楊貞璽 ,CDA一級持證人,鄭州大學情報學碩士研究生,某上市公司數據分析師。 學習入口:https://edu.cda.cn/g ...
2025-05-09CDA持證人簡介 程靖 CDA會員大咖,暢銷書《小白學產品》作者,13年頂級互聯網公司產品經理相關經驗,曾在百度、美團、阿里等 ...
2025-05-07相信很多做數據分析的小伙伴,都接到過一些高階的數據分析需求,實現的過程需要用到一些數據獲取,數據清洗轉換,建模方法等,這 ...
2025-05-06以下的文章內容來源于劉靜老師的專欄,如果您想閱讀專欄《10大業務分析模型突破業務瓶頸》,點擊下方鏈接 https://edu.cda.cn/g ...
2025-04-30CDA持證人簡介: 邱立峰 CDA 數據分析師二級持證人,數字化轉型專家,數據治理專家,高級數據分析師,擁有豐富的行業經驗。 ...
2025-04-29CDA持證人簡介: 程靖 CDA會員大咖,暢銷書《小白學產品》作者,13年頂級互聯網公司產品經理相關經驗,曾在百度,美團,阿里等 ...
2025-04-28CDA持證人簡介: 居瑜 ,CDA一級持證人國企財務經理,13年財務管理運營經驗,在數據分析就業和實踐經驗方面有著豐富的積累和經 ...
2025-04-27數據分析在當今信息時代發揮著重要作用。單因素方差分析(One-Way ANOVA)是一種關鍵的統計方法,用于比較三個或更多獨立樣本組 ...
2025-04-25CDA持證人簡介: 居瑜 ,CDA一級持證人國企財務經理,13年財務管理運營經驗,在數據分析就業和實踐經驗方面有著豐富的積累和經 ...
2025-04-25