From bebfd5348e891cfe2cfef88d586fffc95c079d34 Mon Sep 17 00:00:00 2001 From: kwells4 Date: Thu, 7 Dec 2023 16:53:06 +0000 Subject: [PATCH] =?UTF-8?q?Deploying=20to=20gh-pages=20from=20@=20rnabioco?= =?UTF-8?q?/bmsc-7810-pbda@9170468c1f939131ab1bd3423f1e070835ad674c=20?= =?UTF-8?q?=F0=9F=9A=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- classes.html | 19 +- .../figure-html5/unnamed-chunk-63-1.png | Bin 0 -> 168597 bytes .../header-attrs-2.14/header-attrs.js | 12 + .../img/matrix_image.jpg | Bin 0 -> 12146 bytes posts/2023-12-08-class-8-matricies/index.html | 3058 +++++++++++++++++ posts/posts.json | 45 +- search.json | 6 +- 7 files changed, 3122 insertions(+), 18 deletions(-) create mode 100644 posts/2023-12-08-class-8-matricies/class8_matricies_files/figure-html5/unnamed-chunk-63-1.png create mode 100644 posts/2023-12-08-class-8-matricies/class8_matricies_files/header-attrs-2.14/header-attrs.js create mode 100644 posts/2023-12-08-class-8-matricies/img/matrix_image.jpg create mode 100644 posts/2023-12-08-class-8-matricies/index.html diff --git a/classes.html b/classes.html index 3b2166f..7125163 100644 --- a/classes.html +++ b/classes.html @@ -107,7 +107,7 @@ @@ -2416,6 +2416,23 @@

Class 7: Integrated Data Analysis

+ + + +
+ +
+
+

Class 8: Introduction to matricies

+
+

+
+
More articles » diff --git a/posts/2023-12-08-class-8-matricies/class8_matricies_files/figure-html5/unnamed-chunk-63-1.png b/posts/2023-12-08-class-8-matricies/class8_matricies_files/figure-html5/unnamed-chunk-63-1.png new file mode 100644 index 0000000000000000000000000000000000000000..8f085f6e691708a7bbeab9bd0b6d1cd7dd36cddb GIT binary patch literal 168597 zcmdSBbzD^47d|=+At8!{C`t${qYauA4UaIlV%)p;Fi z>A3PD?hEL^?)3EV6Z=Au#deyV4+m%V&5Q00Uy86RmJMr~A3<5O3w+y6wdRzjmZhlA z9FG`3FsSEXw#iut;z=&nNG1rpxY@C-nAAl}GFL88R2flbTir6eHHBRAogQ^r93=}Y zyJYZslnr``E{|st^R=j9>f=XoM7>W3xlacVcgQQQSL}G_zP+yBYy(foko(rJDVgQCG3z)UnxJ7Ky=TmKOOY;hb1CR!nu^j^8`^5!;VLf$E_} zmwl+m@XOYtxSU^qc|wt(Gck;Kn$K%p^oioYyxf}P2P?#=vg(mD=@03TcXhm~&d(sS zvPC-UxHGzAEGfnU@4MDZHh*5cwp((=1WD@ZtBE{5b&>%x{*{ZTz;lC@p#W~p^|U2e z*Gjly`RQia)M*Q-&mxbdwRm5LSenIi*-MK--aj8M zIHK{g>+agNn=UGb_-nWrEg9VlHoluBKja$ybW1o=DNNoW{6fDDH^PS^n(-296X+&2 zA{ZkU45dHPLK}T{(_A*Cx)#QC(%Rd=G$n(SZpJEUCe12ovuAvM>^w-mGu%xr$K{O6_qnz6h-kVJK6_`wm=pWxd=8Nw)H*~X9Xil*8 zCaF(_gTY6|6)80{=`{1Z@}dQ{IwJSx4O%PLR{@=aIZ<_DJge zhxcZx>N7NtAClbW^K=u*p$euosw8SCa4^g9$@Qnra+$Z!E$aOuqpk3oh{HSm+mZmj zn*FzjtjaQHSsvZ1^nAFmmhR@ODm9oN@Y&^uoh0EMZxe4pX=<7Gj)7;#f@*He|LAtc z7Pg%D`obDjToOC`{6o{kw>zZy{9LDxPu3+~oU53rJl~mqq9n>{5>nvBFZ$j=d4G>~ z$Ro|%C4oqEmqTJh|DD*$@m&0l9jg{$f6H&xOORaY^!w|Y_c=b^8H0-U1b(R3Ri|su zVVqktxNpxk3hDQ*Ha^(d^YmH4?;eAcB_7}(u-zy7bhoz!1PD>{Tgn!2ID`xQObo$$ zVhtexKjDEtbl?vJ0!@S-eL^svh=24m{?C8B;9=$gfxsZj*JQQt8(ztospF zp!%9g%u)lU`~aD#NI^k;oZt#dPhI}OMXt{giJA9F@4zOguLRtYQ)VHCvTzkXPbQ|k z=8U$M8C#xv>*i+RZs9K1zFHe^856%aJDY71HB)o5F=F46k`hYH6GRfRW@hdNEz0#>E4I$@Ra%z zex+RWkZ*uG#)A z=@*vFf86_M6&Dg<3xg%{eo6s^%S6&`dIS_O2;Tqk*VBg(yfUSWnnb_We)N6Z^T-kq z*AWJM5&WP3_>zJNLQ5jw{J$;>V|og%%XAs3@IUSeW1@u4KcM?RzDjwENlI!&cSrA> zyzk(`&TMudO3_eSv@?cW4zCr;Q7Cvs_{~R zS=`h8Fbc;USOA}llWk8XEmW2;`T5V&kLsqtn8`BFBAZ|HaIekvsi$&j^%)X-?<|xSx-X3yMpEn;%=P4OOg6s? z+6(kZ<_341ZSIpx`^)M9{Un$43a_!N$Dgk}rZBGj#-NlPLW|dMjZFC#jX3U))_egl{wfgN=-{+ z)8=-c?|-SWmR8tFV@h5&vprRY*bADhT0slQtW3AJJBbZH{d13r%{c1l90y38=z$#@ zQ4rQ~=V{uK%2J0hUQC9%$fSwaLdgxU6NIdPS!BIyyi{m_*HUMANhMOKQfqRQaMDOm zgo{QZU}SIX^(yqy-Hp1-@!FRGNZ4 zD6cgd6(`H;2L;aJDo`xpb0yFEF_UnPCt# zO@4sNpC|F?NiXC8e>C$cDkl5OTmJLiIvOa!xZw}FOoIR6=&(Xk3V+rAKiw3zZ+7p@ zg4|zT0~|c8uEKTB3^-^EFAb;Wwb{OWzBH9r%?O^;9KO3h6bA}!2MXHuUb1)Xa05X} zdkT%>{uD$Tb(%;2V=`8f^;&>lkwpx0I^N6vsr!Uv9ELGY_#PKWg10l8?5i3p3=$IZ zmxmBDjdKQMX=SJ!?C%A>EBD@7=64*2m9L;B0;isRev}dKsq#7a9EX441~A(LMQ3ZP!2WZ^^*UL2t$Y(IRF(RVXx*tO575S$^8$`{poA=1rYBUJqWBWDvoz(KUvWo$pe!TDBiJP*uG z)uy|vV%-*pITL)=V|KUa@@rhygVpx$HiQgh_%wJPeEk&Ewry8?uwVQF;o%B=*K<+FCueWyzO(3$Z-sqR4UVU;xv&2Rx%b+xmBxE4Q zmA*WjO|_Y?c9#Vu;Vw)zP-@$!X498f5u=3RspA*1CjR*(C3 zjNyKrq1RIN2EQR$iGyVsr)(_10QK zVFtVUDu}P<*79^aAFh@l>$V-23p4>)+SkM44)-|kgJ_e-MsQIBc*vqCnJB=51rl5c|OB{4hc@Hlpn5k$o5Y)^}% zdf+AfT5#s0VdKg2L9Iko>N0!9BrLNRFWv*^L*yPwn)mWUzC=vs&iaCFy2{*V5=NiZ zmy)X;SC<RD5{twq-hsRzhvS?eDvfNl4 zNlx)WP-`Se`%JvjHU43Q!ACt1F>9nK+L@S6@UwrQ>uHQ{SP^qIf*sy^Q5dzJb?&UF zatp>f`w=M(e@0&Qx_;4TAZF$eJ%7o&<$lcJA`W!*6Y5LrIJJ?R$$Y zRjnXZ3*8pK_oc388FqQYH{w3#_4%chKf* ziuQxyI^~8fF?Abbetaiob_)hBUscqeZB&5MUF`AAsy84fB80j|wWf(PyHZ;SsliIdh48D7m(5|3uTlrHsr>6zQ>69H{phO&)nJ`5|g5%F0eP zvFaS(leNefQtL(cNQxb6Q0-!9xB-%X^98aFRNPjiw!OJ}b#g)l>6q89sj2IhMlB27 zTIxc0sEwui#{l2SP@ewujD&_WVS0U&( z4v#(msntJf!xTfXu@2&U+JTzlo>pH!KC6KYgG_zh2QfQSXx;+th0_-GopN>KGOjAy z>p3qGaKiFQ_w}>;2M>vM5K^U05ga=Y%;bYkzXF-M@vJ41xgtK!&-sDZ{`6XfWUDw1%NANzy ze1ji3kBiocgPD=a(yFbgI13+MpS+Mf%=B#RJ#kM?jV>4qrxPJCDXY5VkB1%&Zrn)U zOs!~?du|H7kQkn@L@k5Ea9Jm^FwlxkB{J+_f1`G4uyo9qJ5+i*lcZt~q_gC`QI8Nm zy(C}vbf=*$5r`)A;aw65PRt$_UDEr?UUZGBJNF~r79c%46*DG|g@6@$` zu7@O4*W6E&yGoV7%=@PiKB7hkZ?4Di^Lwq^l`l7Qr@1W*)>uD4uYSN~*it9tICOAh zv1&z4z_r`{`QpPIWyQB*O);d~lTjw(8T{V7##Q~^eQ;4h?1^#^1!~WxWu;tpU3gy~ zc&r8Fw$vLb3KohZuoSMgwt?LwRso@{rRJPzBzDL%w<*Xc4l-=A6fg5fav~D!YF#g#`;6RC zdRk~bhqm(97hEk=HPh8hOnZm(1$M1<$P5%%76#156_7uyYJBC@j&V9W=-o4m+YcfhcgK>X-$YEeHH5N1N zV&YAJ&^9QwAGzi(+T%W4?d3i;#q@}b_BAd?>%NT@vTgR+7_n3+!G2~Oepo zo$0hF9&U{h@TvShTU(3ZTIqj#*L-cZ$Koy#s%y7Xkuj!CYPDP^Tl>c=Msz)J0p63+ zn~B$B1)B(73>9176(;LM>6F-S%M>9eKGnb>|A#s1M$^961>nm%JJI10P{LvEr+XMV zu;?T@Guxi*SBB0_Y{`|W@rW*EwE0RF-f?ornWz1?KRn*b!%mU&Rj-C4UC?4RF6B}k zoDV=`&ogvs6XFLY{7Q;Bs2oS^P9z0Q+A>|ups}HN94QGMjN0E@@{6<1tRPBE-(H)G zdub2qSBuAwXDj;>N9f&$nXf)Rex(l~ZdK{F&}t=uvY5Rv(z*IAROU4vK2daH9MPyC zwzxL{|JZZ;&taHUiHTWa33`%jiTv62CXNKV2ff?vJ!w(Kv)qG0J`(K-AC6;upy$gJ zUcmhI2-JY zHGTE#xMJHr-Xh;vqEfk-!Q43dna9V;&k0kH*J&OcJTaREwvUjyb| zxH}c+aJ|I=lru&4uaBLz;)g2pz#zO?CVatHf1$;v5B=#8MWL)Wgw1iOxFRn#Z^^Bh zGec<_jlz?K1YDqIf=7C_)uYFVA17u-){c4oda>dY-yFpmb#Tgm8<@3_;m zdB?{3%zvi4L1jwYo!O~*`L`ttiThJV=f9a*GcLez6xEZ0&CvIS*0mCTI>SzlsI}!U zP83uD(4bMurYOn?%SSt5yR|%I8>g4OuKCwiOD-o{v;V#rns7ex93WX18&S`g3&uBC zl4ULs40CDBtmuyD3(Y=)Y2c+N5~Y0sasF2+5&DYMq0954`956;)oe00FBV*oNue8I%m%rpeDTZ0pv3WTS?MW zfWrXVK3R4MfNtgcqPqUD39OD`?KPiC%>}Dda2QhxF5Q5@F5Ps$YjRA#*owI=PQ=}V zC(AgM02Cb;Ky~y?+vy5thTJ1kO~5HcO9nkDGx3KYcg9W$UHk1>MgH{ucL@xUCPbm3 z&;5yn0mX3cRbh&b)k~IKEm|G}k=K<7tfsQ6s>|J;2fbOUQM|JGPW4{UT)gROceSA3 z>w8aK5NPjqnYZ;$V~D>k;k@+Xpo-=du<=X*glF)g>i9ylf74YT73$*MR3N;Z?PncVp_b&Oc~0+Qu5*&!oHB~r$B(?ME&<$dhUn;j!41q$Hda)MNalq8+?BNshlTi4#TzgOQ>#X7t@RRoAmA%m2e0)UN znell6^*Hw12J!RUp2GN(;M!}d4&Qn5m2PuZMm$%Pq4P640IHzsq{a`4c8N}$zj#i>{&@T|?s%$b zao23beE$uFPzL(GcT0mME#2SV4cQ0TN`-&<^1@&BajDX|U)hu(nX8VtEOS{?6zX)U zzt*poG7UgwakvjVJrTS&s4dvI53+S5;DavWiU#k4y^ZCq>4Z4R&P=O(<7zWN ztg&y>1}}3`1q>5Uv_!SPc#rW@P$~x}AWA}J@i(LY-QQnowoQ5+hOuUWDsy+_o5*yI z8-N66K2T_`S!kwgVgSxFONv5pI2msl%%U}#Id-E>WNg(abfP}#>Jt{d^XAMHHv&M& zc&7G*ZPM7EGqYSImEf0C1YDr~>C53FaZ}7;#ZZ~u?}0BEWsDJs_PPApc;?M1L2)*= zPZOQ(oLcDw$mgDbCDbdid8H>~zqLH+JYI)4?KWl!*uNP;X(LND4B&1iqD*{WQgNt7 zuK*@YV0vSI;BHMEUoWF)Z616B0J&zcT?-kBNycKZm*P$?xQ@Hvm$ekgaoW zkWom?hdXCimh7t>5y&@ZMK>QD`74t@%mQ~9)D_#15dy?aVZ=~CL-6-h&1UCJWm2w8Q&W_#b*D)eEu2+2-IJD(G_i{_-rob!LhXJQOsC>towFQ=@njsntl zXEym*eS8weKzZ~Oaa!$6x9vh7*p~=a#SpXiTgxqcaG9$4!j{&OdqeMHmcVLSBBQ9j zfHO(f9yEM()*(;^YLGt}?4;oeOjrVBqAn=9i|!2pj`JS$wA<%+DX-?nm#k^K2KmDpnl*70GL)|2c%7`v4T13-eQ(dmCdSDGSAtK6#@%UxWB!O_3J&=ifH8x=$fsVHfU6KQXn-C6Zh$hrB(yH8ZfK0srqMVSL-CQLi< z_U;qCxegO^fRv}Yh58J?`8``Irv|MKL#VUF?>JCOK>XQb&IdC%LG9>vz6YiD0ggMs z075Ed>*m`_lbya~^}NG$Kx`Vx>a)Fiiu~aSpz6*(?{zJX2OKj19S)CI#0l2{uOz&o zlco8|aHzsHX8@(6I6O;4Ar4j3%1}eKT2%siD%urWv%hUC5Zr6HyAo>J?wGd{gM@xr zrzOv`PFyJIGZv^k5a+7kF35Au3@|e~2&2}+Q>c^gjp+J)Zo@VF6AdV9v|}f%@hqjd zjZCG7Q!yL3?21W;Q*mrgMRu0ESh|#XZP-soE@>$iVlS3!bp2#_u;PHP@~#(|mZq0F z;&?=-*94H3{Toimez1@zzx|a2%@hT4D%iW7?8W^`pB=YOY(`dA*0bQR?*|2-=tq2y zxPBLOWvX>2>INDXv)M{O#iv1t3mj@=VP$*r2U2c; zqHGB~DQ%Umj%zWcC*{{{XK$HpTO&p#@XBV&G`SA<@%M4J0(D;)G&5UrHF3$9p*F0pApX@e4b1nUN-+K>*$LF@fJD_4%zJu+) z+KHqm4Ez`kAxMTq!fdw<{q3{OgHK7wPiHh!`3F%+hRPonIqqaH=!8^Oqj=$f_Ihbx zU;b%?Cf)4{2%WS7HVhA8-2{Rf=@JUOdW9E;EQcLGkXx7yapzaDRM=*RI9INAskT`O z)5~_hyJsLk3U`O~0BQy5_+@wG#@jpcT72{`5lw7SW(JBJS4G<$7rzMnAaw3BK@^(- zUe+9q6kFa|s*U69si@>j8B6YCTvo`pQu!6GDRp57tli4TPBI`)Y^`QOYP&sGKVjxm zY0AF8CzE#%L`r$RliV0~ksT^`{35pn4rVHdN;RKm$L@b47jv7B#sU8qOkKgWfQeJi zNV4ggD_6!5DW!8o_-Q0m?9n5-B6j^0GyKz%>|vPww{SvQ|biO z11-I-%8|7IZehWbfLJne+@P{1g~n5gZcRkAZR-xs`0i~t0fH2nwFinop_KM6490xC zEJTKSbpN!5&)7?Xt`&zQt`qGNj* zP3Qy}6CC~Qu|8ngX8%g>Rhy6l@kDX$47z$Szaf;7uVVEx$baey&+kOtCbgk<;sjv) z=fMoW>hG`WT&1WFs%_$aDBXQb8OkPH)ReI`+$HP%HiqF-vyV{L7n%=#2S@_SG^3Ag#nae}zC>TXhiZ@HZqBH2|^ zS}Z-;5)}mq6;)84w>>_o-$z7F7u|V(`#g^6XelPMEwFTyb|M6w<4Zxbd?yC|v6zhM zvf3Hm8W6i)f*3U#*}jTNY&}&DiYAs9)JIvVi5=F#GcankICyvC*p8@snzIO$_@fpq z@Gf7~_Xow+n9tXzyH7b(pyMJCT7V^5c+Q!VjFwvEwcv2u- zO~mp(2xqOUy=o|Nu}_41RR{Y!aQ{i*cC>rt-$Ep=GcAJ3MBtpnu{x9ruEWJHnqljhmYN#9 z3;d3H&q)*9x>ZShy7Ishbcse&LsJp0m&0XPqnb zdYlhCL8$;!l$uo+rPYvOgou4 z0M-rbB8$9uLvEF$TRT&O*}A{rcIH4)Ta-?ndZ4o<(n;Z+c^DoZO=4|X5n};Kpu`wi z$suwJ5G4cU=#D75=j)H(*_Y%Z*h)MTI-SEK954fZ>nAu)p9xy|RV6Wb5N_r-P56`+ zA3tEf^SB@Wk zLrE0Riva0^bRBweKsiJ|A;obg4WrD}kq$q{ z39z?ZNuEVpEQyW@SDZvZPJWI4JNuE?O1V4U?x53oclR2@v;`kFNEV-SpVV ze5a#R4#i2Fk&`bLfc(D&VqVJT!$w-8OpT;QT9s#cG`A|<-T)L*U$RD~{T4LmkBPU5 zg#z`%%1lrO&cSh&Zdl2Gd#E?|Ne3*I=?dY^+*`~5NZ8Bi?0^6A!5Nzt#jcIYvO^O} z@5~DTFyG>V=Tx@RD$pJYN~Moqnk2h|aftqNsT}IC{9#&zx^=Wkhj@L;#NC2|5O2p% z{94le=Q!>_lCO?fwA~#Ka`ek#U;*D~F)%sgRY-stWr4`;ncn*uNc}Hi+;OGW>d$wV zg1Yi@&%Jt%s#^>)e!Uv`{;ikk8Fj9*uST4kdqcN}WpCyagNJI4OGh0(6vgZR9_luY zoCWToWC`YLSDH^tu$MG5xS6-DOsl$^S7&P55r({yfp?Kn2Q6H^>OJ^lEl93aSva5M zGGULc19JV4>4qu&e3}zj3;y2~6_Y8bu8S^<{$fMU|8!i5zO7%Mfct8Lol8=Ad4yCY z*-X_N3PYtH!S4R@WdatKeAb^Fa9cVKusqu2;nLBm2V{d=M=&-M@tXkvU=wfx!3-NmX6!oT z_=?HXZy$XR0=+@i$XF82`HMPHGR1pfPqMq5DUS1;^g7UKF$bNcLeDjuflBv$Konn% z6>;DNlC5^_u9vKeW}{UexH7^T)UWAtosvmwh6Sbp0N1O4>hN@%;teTIjX0c$^$_aLT1qp0)|2 zF2e(;xOJnUa+eGonnz2nT-PhMdb#9%>BRo#x05ULpw-j_I?QxIipwuVNv@;Bb+$K_ z_)3SJgRlvDI(I?gnr?ywRB|Cu$K)0|^Mrs1{Pk&j%Asr2RYTW_MR)qEh0-m4n(GVk zJdUX5In8T$5vYD5Bo@oAg5uq#7qSPqyonDdWg^|jd~|DoR%B~;r5zsc*v!EPeM{L0 zrtAL( z^G`$YL_m>lhALEXE(MO~Udw2M%k4Oyb)x)<^O4yf{&RGei{HWW8h>_qPvAOJXCFAA z?`mplUS}{mY2O^lSqQoy$F**tym2T#W#R;*^d)$#;7ib)pVmxo|JD%N0(P@CG~{Ns zQRO>oUPIBVHLI%#0s42a0H`KN&U$a}JhC#H?aqo|Hc8|I<#Pn6zv6ZQ;V-yxtT|&E z#8jK!Gnj8$>QX(q+!+n2AE_L-ukY_}rCD_*$IyjE_-1TanO2zAKS=lMKj108+i+qF zG@IHoD?#Vh!jpjmN>)H-V@iuYbY}ks!uf-a;!r8H+zKw-r&2@%nc_{J{FQxVPhRDu6&T%7>>x2BuA4oNcIy935KQHHllpD zv6{KJu9UC!8cc#RxW-GcBO~R9-~rG@aFWvrQjk02v>j0m2QYk=YOjsg5o+R1#flIA!)$BT%ot5uECOcP|brP$8U%dXIHC zcxQLT{5YMU68rUN)9n)bYsmq_%abi}C6mPoZvP2uc)$h2JlAHG0WQ9VUz<9QqhhO1 zi5&)$x6wa-|U9GL0NvRHHFomw{o6;1p}bZ)@)9Gt6+~J zqfa-%oM6FJDqtF7g(e3}9r#8nTm|}E`*e)%+RyjDO%A95kiDsz?$`DfO2@97zEKQiV4qnZw9Rh>xW%1aa|hy? zSk8BJ#TcO_W8W?FPRl)@XR#naEfJldyvf9|XFphM&2K*}TE0D-)23E8sOU=fZartV zCnt_+0p&hwy*)3`F3CDr%HTDB#be^7B4sbx#{d-_}gnY&&n=V^%sLhYBpDa?fZl*xn$alYZ1|ZiZ=ohl)7*CwJTvz<# z^Ern|Ej6)E1fvMgl6BRLleRLQH=HY2`S8l-D+#nTibPnB{daf)E$eTJY>Tuh1(w%A&89s0t*W%TL!->Kv0Ps$e(&oSAJR_Wn1 zj{-J49Pk{VYfR6IrCaTQ(;(oy9N6T|iketnn}ZSev1j&>q?Cx&0kn2M5MYT9_#^=5 zYSFG)`g*`P4`?$q;b)fl_^-+Z_KF~a#94(o4^ae_V_sJjr* z5}O|W%15&&QU=)Q+tTZglV9)WM0UCW@+V~G>w1M+9Q)qJbE5Bvk4PdklnuomYt)7H zj}195cBj{sEYd{eZU>wWnVK?N%%d*7mD&nmak2e&io|;x*y1ye`0sGIE1(pv07_vA z@!^pvXAZq@;K+xqTA*a!E+tg_$BqmXj_J>T@)*6NfX4tFF3sBvCkwBLSF(!bMxNL- zi;p8_O3v{J$>R3+Ws{84ZNyrY@K|!%|GNK#jMIB)b_s8*8ggWRP)oSLY z^K17#sp>E#BgJ`cUAA%~*V%3-wI4Bc(9T77lIv8Z z*+oQ#!!sU74b2PBfySG15`IG+SL!10$g8jx#1%oAYRWxAed2EO7b|2xfY5FZCK%Wg zdvbL7w%YynsXXg_#VSE=>$SuP^$E`OR=C=N+CBnkhg5$O8(wrwm`C%j^zR%ofbiuL z57t&J!9Ep1EoR5@0pwI^QDKWy6;NyO=!)odBo34DRhiB@fzn7B5FBk56#9?VuJ&Lv zFKiHBkC)Ut&1ZbRSPn?iPTuDk+~Xa;Di)Ilpy?%#vLo&K?{+w>jvGoh-v!eKdgRvP z>-0aSuD5C9k6e&dY%+5CL}P(@h^$y$92p`QL{ygoiY4w2;>r$A?fBBYLhOTWezisV z2e;X7&iRHJM6W~b-c6$8XH20!74LB8DO&m)=+883OwDwr$S-wfm4a>te;>lk5d61Q zyBi=e^Tzh5P0~YZqpLH5lan)sSFv);+>?o;5R~CjJg3h7;|Ae6|Ir!iy@1XDgk0oc ziLGjZk(%p6N!yj(mRbc#HgrBXmW_1@Eg`34&z{nl#bL!#%mCkGxfKkk<#Jp!_dfY<(7#cgxc zvLr4u0-fQZF^aX{nuFgBMI|dxbCT?BXG$I>mtT+u#2^15(5WupBflZ@A4vYotN{-} zFEpwnKeYJ|-0>5#;{N*ocGGLB{ze)pY^vo)LC=2!H@vWf!rb{^cRm9!auBm{=@57d1D}wG^=W!!VRjLikirfL#O%AT4lyC#X5Q!-{G~jV(!LF zDFfE6P<5oJEF*ls582Ig0TR`nANBd~n+}k9@2&Om`|fVe3shWG#F6v>lDOXp&QuX3 z9Q9g~0hcQ?)aKe< zvA44y{@*58oUyE3iG%e_V&A+xKfL>>pTS$B-?Xz@rGtJgN%*|HFaSH`=Zqr}c5$;v z2ktQPzr6Qwm%?xPLkuZv4P%dh`;i0Uo`k{=oQ)rT$GZR8)5FU!Jp*Th?MF+R^`Bok z^8X(I$J74*>81g@Kl0@kZ5+Rk%j(toR@)7|gEvR<@d1?dGBPqU@sT~K;Cl0nxPjUv zoKDC_sm6Qj!Bh`fCpuyBL-|pTIJ5+XQvm2uD(ma(r#r8?H-Xwi88>|Zl6XvB!`F|U zDT?Nc!{tY$KnTMYNMJ3X)y0e>ZHEV6raX6vP&Wn8qtJO$h3Z?TNo~!NXyxswiQ0ob2W(ebkV~`otjQfvl1B{`S?-`|xwMt+D;I>Zy>H@{ zkS-JpyJ(lq-(<|si1GAI9a79?0yJn(Zg+sTXbX+5A@jzlS3VGKa#+<^8TOahMgjJq zlc4?eCB5(z#TVzrG}+aV7Zd#U%K@3su4wM4@s0EEWT#La)&A|#Mmh_l>m0T}L-Yll2 zn}U+r&VR-6XX*eeG~?*Vyum8q|3M}J{gjU1?CIHpq*?)A5O8yxxfi|G=U)b&6N$wD5#8hSGQNhqW#dqj?{GdV zh&sL-PLG%}jpR>8jd`3%_~#rhQgzgy6%W>m;@7V9xzB8?d4wO2WbnFJ?SzAce;#zY69($Wxa6P1or|B5~ z0YnY0$6-c%GLfl!KjIu}P8rRrMy$UD*y>o75S@?ocwqQnZ)oZ{9^BAXP2CgeYa(a5 z*@M6Zs7RO!?Jc8OQWsT+2PH|1H*YeV!9HOg58yiJlEOv$+82A^?o9-rj9c0YRyi z=!9Fa&{GjW8wgyU=u+hqs-44jDOd286R98q>rAadVaETmp*hi;-AEHhc8he7X)92UNZxrk2*_?3xiuTDWR>z z%zFq6VDo?fY8}a?Qc+I5iEJIyW)(2i)_&)&qdDpR2&aqk;q+2cD)P~ z-2xpG**SUq=ZM6k5I3Vo4?u>G>2#%wX7x~q-6a)=>x(RQ4EUs}MhO!eRk{h;ew(wa ztMS=U0q~}k-+0Vt19jFz4>!ft^t7lZiD(Rm7X`j`#OP%beoATQ_-1gukux*E_+j3b zuvG_>Ksa1PyW>DB2@9x}DS?g|k?L`z`?6R9+QX%2?TO21)3mqv^qbtN4zZ<)mD;>9 zp{`uAdxJ7`-Y!!n$10w<{Bx2!X&n`ySZz(Xcm>>D0VRAdmTz;aI~gF|smmSPdKDhy zyDm;Bq`_tIE>~JJH-i!M1+&tlA_hU2Bs17X+Ojp+g;Ce+2#BIml0R9`tO_O!aEC59tG~|KHez^0# z!{bq^gN$v5WR`)M(ZjOR0lYDRh#>P%x-%_Mf*-4 zhh7bb9mF08@GL6L_i7=)N>oPP{I_iS3zEtwYB8Ddh{wbd9;--~^KW*ykN=ib<5IZ4 zzZaokrW+6h#*v;lFWRAdW8|`I-(0T|hij?w$o=|(;r1BPGfit%uKr43LRa4X&Pa_< zWj+HsAS4V!xarDD6fQytNcFpP`@kKD2yRu~e=C+RYX}D^Q#hgGG}5mNrM0mz6!is> zyW{=%2}ftt!{c?>^X?;r3VNZsp>+x}^`_iCpct_#G4_ts1wzs!Ob< z+Y1W~|E-wf{PT9w@q&OL;hB9%V#*l8U5Nu5YZtp4NL^90iGNbkG;d(eGKQ*r_Z!d% zh{(zq`8N4gWE*2(i2{|}QK4Vg38>4!q>p@taww4Q>?-EB#RxRmozYCceioBa;vq76 zKV@bW1CHkB+`BV^dOL0)a-;HV+V8J=jD3ihVPfv{$9OD>7i*S&LV@(UMY5_!bduo` zP{siLL3Z?lnA#gi)r)5 zK%Z!)!a)u4284P3yPsbiSEUEnY|H($4r-)W^AMb{q(U-$-?tpJx~Td)Or!0)!hmG^ zsi2zTR~(6E>@i6L(!4m~&12~^OjILv`5=}$1O0}=gdX)*-T*K-7U%C+ppO-_HuC#k zv$H6`DhwzTo7dA_F56FUU*~RJ#=1Cx8geUa4t}T?T zee0YKED?Fxen9W4l6Vum-r!^RO~s-;0N-S7eL?v?RszOKqPNgME+^zCHcJ@1IUACN zzI@$eN(K~?W4@;w{t2&dX**s@p+my95hK>c`Xx3v-CE}j`HJqBA(slZz>EPr@!m{F zl7+=6I_!jhDpRh5fy+*L7A$q&5s&l058W+`!lt$R%w}b<{5aF>m{bQU zO?kfuY#g_8@kCM2vr{ez#V0Ad^Yx79uIpG1W@({8)qT77f!FSj2Dm>T8=E|?^FVhi z5A;b$8pS-o1EVZCfJewY$tN1KXN%F_KDgIezFDCayq`b}y6F;nH#rBl0;LQN_#O-J ztb-23Y1C=eBdte2$qYX&+TKO{>ptVbrs6yUI>L}6v9{_uQna&M)m^mJo`kwj$e)xs zEZ9Hfapz0^SYC%po!IWtgf{vg&~wE3y{(g+b{@(0+~oO$2U z7ur&;W+?m3#P+@$YU0jKWXkx#;$0eA%{{7Y>H&B1?ACweH#Gl9Q;(|%2GF6xeTiYk zbN2Chaa)bP(o)MEF`CXqqk{cxtsJ5896Sl~?LkgspV7RlEQ}lz#AU<&w$nMli5lgv zN%%F=Ftg%f#?QdN&TW!2of6 zv|edP(+;TVVy>cMXA{2mOz+LT%p8-V^;hktm9PHY={~AM0)|Pg9O0HbQlNm*OjSB1 z22^To+nW`)1JRHU@^jF8pTmZxJ@b!P#f*Ov9zQQ3$(ndheOCNY#kpB9fX-iR z5H;IXa{R1$OaL2;2^f8N_)_lZR=>X+@R6cyCOPPu2uUYOy)RequJ9W>!2RD+bz@Ud z9nn$+oly_kZi3%mIQr@7GuW5+kJ7-b*%gn|Cr>qz{W-|-hrO<;1E{}h$9W`)|9c4v zb)Col{^XFqc$yB_+vF_OBR$3+_J=^@0nDG?;)VR)lixqV1PS{Ig<>lrgMWVpe!Acf z1Hdl-wSik&|4w3b|;5(OTR1~h^sJ=ws`IR5t#3+@_yWYm1dh5+wQZF>&VT+&K^vG?lR1=@=y zHB}RBaknBipHn+ci@Pe1>KG15LLKvbgtU*E-Ty7&X>c8Yqs$=wK5P5HvKk3q-lGB# zDXTW!r!pf`AVCJxj8p427MUg><>IDA&!T^Ps)RnHkzQZ)N}l{Io+D$uLDVOt&Rat_e~eG^(}Y5Fys%Zy@p{!~k;c*33;G9CKfowbfyTl|0f>kTmWMJsRk{7|KJR^vY`(titu zZ9=y`)IO3pI-3C#v|wCp!mpv20*|1I8~17+oaBihEIgd$({q{#kiV3lQL^zHg|0sX zZ}L#aG5$EE0zh(>J7s{nmj_Bo5iuZH#_5^|rZ}#CCC)V{RlAvI$cZDna1$tMpR%kq z09Q-+_fQzH6O5k}Bxps@w28)M04;x4m4^#}-7j#1pvic7x6%P=5+}|B)W)d|A?HNU z6D9?wvPxS`k)lvsI$pZDL%=CU0NV8DmtNjHGzN^C@C^CUPJg|a$Qs03^<7R2f3FfJ z?<%}EBpf5-SF^k1cMzc*!QKRVx~ZUndL_#oRD^9$Gae9e&NhX@a=GQFc{Lb zp=N#R(7mt0{BJm%f~69J&1I4$MDUaWVUG$2R>NCeY2o zNhHoLW7k0TY63`-A6Z&EvnJ)W9<2rXFM?a4^b!`9Kj>{o0UZ`2Src*ozVZ0SXH@Cx zKtdyv_-H*2C@6@k(mBAO`K2Puc41qzwO+lZ+ulfx!(8v1WUZ7nOQ1wqXKQDm@?V+H z$@p^U#1z?(`ZA5QMx4;)Lex(RCGaPOa{D)h0$yShqL-s8BG>VgLb++eq=wG|YR7Qg zy9L05@kb_^)a)*+c3Njd8M(!j;~?#d&s1y7+2z^Cr>2oEA$oC(Kzf{k^j>aq{y;AB zG^CrAIq{s6l}wlaTfeTv0%HjvzuV*cNmK%9NEon3OZjxXt+1EXaM{}!w|kZ+N^0k% z0J;uu;j1;|(#~W8mG~8zK@%YR;H+qR8bzh0QN6oveKcM{0A^87|MxyEe@gXKB5@6>?`5MetWJMOFqEUt+e~E?|~M* z-fBVy5R4ch9%j)};feCqxDLKJJpH#xC|-~S1-0VNRu?fYqrEr*3T;%bakW0~O*H9P z>%?TBabY+8Vi)ZQUegnI!v-h72DAPqf{+a5KwCO1Os9eyL`^Zj8oc>HvjSn=36n2K zWu09eXqP#t8b)B?>_b3$$p?^k1Q_ois0>~qCZOrSCgDUnEb`BzVR0e;nAAOuK5=ir ztHGJJKy#kBveuSDU$GVrdZWwJ2iz0gK#&bGPS7Res#}OBV?}6jZmW^-Ehu=&m2YM7 zk@C~|5>Ns+Ab!5^>%uG0t0TPbCFq|B%3e*}+fNW$V6H^43`blzf_JN+IXgL1glxJm z*NaP;0EwInT#@kJSdEc=?Vg)JGul0H?EhlxEugC0qOM^$90?U9RHTuR5D=t66a$bh zrKGz%rLPi_1|=a#N{Dm{9t1=w1*GfH9SRCa{p-Pa-|rvq7!1d_UOeZCz1LoAt~uvw zKw9+KFZ4mLK~*T=Fg6l#2N-2K!5co;8N~wuuL&y@_H5&>Ou#8k0c!Xf zWW({D_!;6%;*SF4v`65d=k{A*Gb0C0NTeAg=(SuvC@`O-}lE| zm*n0}bv;@G|4r?{>=i+)?j}uo0>X!!MYe;NpWVf?JCMb*M4S@lbsQ7ludD&?bXeUr z`oPUKC(4iJn%~>`SK@Ej{?DwEF%1KDJ?*nZV#k$aP@=MUR$|mjP;p}Szzl|I+-D;a zkd@v~PbW4E4u8c{?Pq{?WT7xdEb{J}2a1l$e*5r}*wfq3@Gk>TV^QV?qi!G|_=6*A z6}E$q?TK^V!6$U2XN3$8~xbxb{*xafQ&t3nrnXE~2Wz)Y%X4 zI*eT@CEN#?l)I+XR3l@&jFE098_;#d*`C8wDevWOV935au>iZHTz}_2pA)W%8$&bG zneUdUgT&nKLo#&Pto3x)LQm(AGh&S7?{;M)1t-};Uy%3@Ni@(AO;{6@A*GLm>m9Vf ztL!BsD{2wPv^RkWj84_eZ`?%aHUBcC{lkeG`s8eAy>$8DQ_o~H2oc*NJjk*8OWdWa z!_cxY|4y)*0W3qcd<|~(!l(=1mBBG->sqtNspInks)~WXK7xDHQ`t;_17>y84l-!` zEQ=I`cfOof^6!@{w#TjmF)G1ZK8D@M9k-9;RZVSP_H%xVMCD5d*Ks<-9a4n;sL6%=$1I&PUbbuGe{CVP%m4V`$$YBbKN*8;6+^4`l`v?|Q z4xMJg`(LZSK`(=Hv|KHxaxu|yF2&=;&esiqzm1)|2{>i1u_?c>MZIrG%<0K&i{Su@ zoNLVU!ke#_$QNsZsMC6J5)XA8YP#MR88uG#-E$D!_s0a_W}ozpJK$FZNFRctyblM3b`TlLh-XceC&#@0DAd3GD)Saq~hbA`Y{Sa0FkMHs~9y z)D|-HlyebxTOF@_w^S#_x(j#z3_(`Zt9NWJ!|!k)%gXDyZn$wGTF4q`lIGu*Gp!jTiR2~yyIzPmrB}=3 zEF;arg56IMK&b$pkBQ$x-+g`0)w573yaRbF)Cp9FKMiW1EZ%0Aqulen^Akj>-w{N# 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

QdSA}xGn(;V;^ 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 DpM8mf literal 0 HcmV?d00001 diff --git a/posts/2023-12-08-class-8-matricies/class8_matricies_files/header-attrs-2.14/header-attrs.js b/posts/2023-12-08-class-8-matricies/class8_matricies_files/header-attrs-2.14/header-attrs.js new file mode 100644 index 0000000..dd57d92 --- /dev/null +++ b/posts/2023-12-08-class-8-matricies/class8_matricies_files/header-attrs-2.14/header-attrs.js @@ -0,0 +1,12 @@ +// Pandoc 2.9 adds attributes on both header and div. We remove the former (to +// be compatible with the behavior of Pandoc < 2.8). +document.addEventListener('DOMContentLoaded', function(e) { + var hs = document.querySelectorAll("div.section[class*='level'] > :first-child"); + var i, h, a; + for (i = 0; i < hs.length; i++) { + h = hs[i]; + if (!/^h[1-6]$/i.test(h.tagName)) continue; // it should be a header h1-h6 + a = h.attributes; + while (a.length > 0) h.removeAttribute(a[0].name); + } +}); diff --git a/posts/2023-12-08-class-8-matricies/img/matrix_image.jpg b/posts/2023-12-08-class-8-matricies/img/matrix_image.jpg new file mode 100644 index 0000000000000000000000000000000000000000..3d540bb6040b15cd23e2ba0f6e06709cfcb4f276 GIT binary patch literal 12146 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 + + + + + + + + + + + + + + + + + + + +Practical Biological Data Analysis: Class 8: Introduction to matricies + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+

Class 8: Introduction to matricies

+ + + +
+ + + +
+ +

The Rmarkdown for this class is on github

+

Goals for this class

+
    +
  • Learn what is a matrix
  • +
  • Describe difference between matrix and data frame
  • +
  • Perform mathematical functions
  • +
  • Convert between matrix and data frames
  • +
+

Load packages

+ +

Download files

+

Before we get started, let’s download all of the files you will need for the next three classes.

+
+
+
# conditionally download all of the files used in rmarkdown from github 
+source("https://raw.githubusercontent.com/rnabioco/bmsc-7810-pbda/main/_posts/2023-12-08-class-8-matricies/download_files.R")
+
+
+

What is a matrix?

+

A Matrix is an 2 dimensional object in R. We create a matrix using the matrix function

+
+
+
M <- matrix(c(10:21), nrow = 4, byrow = TRUE)
+M
+
+
     [,1] [,2] [,3]
+[1,]   10   11   12
+[2,]   13   14   15
+[3,]   16   17   18
+[4,]   19   20   21
+
+

We can also use as.matrix on an existing dataframe

+
+
+
df <- data.frame("A" = c(10:13), "B" = c(14:17), "C" = (18:21))
+df
+
+
   A  B  C
+1 10 14 18
+2 11 15 19
+3 12 16 20
+4 13 17 21
+
+
+
+
new_mat <- as.matrix(df)
+new_mat
+
+
      A  B  C
+[1,] 10 14 18
+[2,] 11 15 19
+[3,] 12 16 20
+[4,] 13 17 21
+
+

Just like data frames, we can name the rows and columns of the Matrix

+
+
+
rownames(new_mat) <- c("first", "second", "third", "forth")
+colnames(new_mat) <- c("D", "E", "F")
+
+new_mat
+
+
        D  E  F
+first  10 14 18
+second 11 15 19
+third  12 16 20
+forth  13 17 21
+
+

We can look at the structure of the matrix using str

+
+
+
str(new_mat)
+
+
 int [1:4, 1:3] 10 11 12 13 14 15 16 17 18 19 ...
+ - attr(*, "dimnames")=List of 2
+  ..$ : chr [1:4] "first" "second" "third" "forth"
+  ..$ : chr [1:3] "D" "E" "F"
+
+

Here you can see that the type of this structure is int because it is a matrix consisting of integers. We can also see the row names and column names.

+

As with data frames, we can check the size of the matrix using nrow, ncol and dim

+
+
+
nrow(new_mat)
+
+
[1] 4
+
+
ncol(new_mat)
+
+
[1] 3
+
+
dim(new_mat)
+
+
[1] 4 3
+
+

We can also access data using brackets[

+

Selecting a single value:

+
+
+
new_mat[1,2]
+
+
[1] 14
+
+

Selecting a section of the matrix:

+
+
+
new_mat[1:3,2]
+
+
 first second  third 
+    14     15     16 
+
+

If we don’t provide an index for the row, R will return all rows:

+
+
+
new_mat[, 3]
+
+
 first second  third  forth 
+    18     19     20     21 
+
+

The same is true for the columns

+
+
+
new_mat[3,]
+
+
 D  E  F 
+12 16 20 
+
+

Because this matrix has row and column names, we can also pull out data based on those

+
+
+
new_mat["second", "D"]
+
+
[1] 11
+
+

Exercise

+

What value is in row 2 and column 3 of new_mat?

+
+
+
# TODO find the value in the matrix at row 2 and column 3
+
+
+

If we can make a matrix from a data frame, what’s the difference? +Matrices can only have values of one type –> integer, boolean, character, while a dataframe can be a mix of types:

+
+
+
df <- data.frame("A" = c(10:12),
+                 "B" = c("cat", "dog", "fish"),
+                 "C" = c(TRUE, TRUE, FALSE))
+
+df
+
+
   A    B     C
+1 10  cat  TRUE
+2 11  dog  TRUE
+3 12 fish FALSE
+
+
+
+
M <- as.matrix(df)
+
+M
+
+
     A    B      C      
+[1,] "10" "cat"  "TRUE" 
+[2,] "11" "dog"  "TRUE" 
+[3,] "12" "fish" "FALSE"
+
+
+
+
typeof(df[,1])
+
+
[1] "integer"
+
+
typeof(M[,1])
+
+
[1] "character"
+
+

But Matrices can take any type of input

+
+
+
M <- matrix(rep(c(TRUE, FALSE), 4), nrow = 4, byrow = TRUE)
+M
+
+
     [,1]  [,2]
+[1,] TRUE FALSE
+[2,] TRUE FALSE
+[3,] TRUE FALSE
+[4,] TRUE FALSE
+
+
+
+
typeof(M[,1])
+
+
[1] "logical"
+
+

Matrix opearions

+

If you’ve taken linear algebra, you’ve probably worked with matrices before. These same matrix operations can be done in R

+

Basic operations

+

We can do any of the mathematical operations for a matrix and one value. For example, we can add 5 to all values in a matrix, or subtract 2, or divide by 10

+
+
+
M <- matrix(c(10:21), nrow = 4, byrow = TRUE)
+M
+
+
     [,1] [,2] [,3]
+[1,]   10   11   12
+[2,]   13   14   15
+[3,]   16   17   18
+[4,]   19   20   21
+
+
M + 1
+
+
     [,1] [,2] [,3]
+[1,]   11   12   13
+[2,]   14   15   16
+[3,]   17   18   19
+[4,]   20   21   22
+
+
M + 2
+
+
     [,1] [,2] [,3]
+[1,]   12   13   14
+[2,]   15   16   17
+[3,]   18   19   20
+[4,]   21   22   23
+
+
M - 5
+
+
     [,1] [,2] [,3]
+[1,]    5    6    7
+[2,]    8    9   10
+[3,]   11   12   13
+[4,]   14   15   16
+
+
M / 3
+
+
         [,1]     [,2] [,3]
+[1,] 3.333333 3.666667    4
+[2,] 4.333333 4.666667    5
+[3,] 5.333333 5.666667    6
+[4,] 6.333333 6.666667    7
+
+
M * 10
+
+
     [,1] [,2] [,3]
+[1,]  100  110  120
+[2,]  130  140  150
+[3,]  160  170  180
+[4,]  190  200  210
+
+

We can also provide a vector or another matrix to perform element-wise functions with.

+
+
+
vector <- c(2, 3, 4, 5)
+
+M + vector
+
+
     [,1] [,2] [,3]
+[1,]   12   13   14
+[2,]   16   17   18
+[3,]   20   21   22
+[4,]   24   25   26
+
+

Here you can see that each element of the vector is added to a row ie element 1 is added to row 1, element 2 is added to row 2, etc.

+

The same is true for subtraction

+
+
+
M - vector
+
+
     [,1] [,2] [,3]
+[1,]    8    9   10
+[2,]   10   11   12
+[3,]   12   13   14
+[4,]   14   15   16
+
+

And multiplication and division

+
+
+
M / vector
+
+
         [,1]     [,2] [,3]
+[1,] 5.000000 5.500000  6.0
+[2,] 4.333333 4.666667  5.0
+[3,] 4.000000 4.250000  4.5
+[4,] 3.800000 4.000000  4.2
+
+
M * vector
+
+
     [,1] [,2] [,3]
+[1,]   20   22   24
+[2,]   39   42   45
+[3,]   64   68   72
+[4,]   95  100  105
+
+

What happens if there are a different number of rows as elements in the vector?

+
+
+
vector <- c(2, 3, 4)
+
+M
+
+
     [,1] [,2] [,3]
+[1,]   10   11   12
+[2,]   13   14   15
+[3,]   16   17   18
+[4,]   19   20   21
+
+
M + vector
+
+
     [,1] [,2] [,3]
+[1,]   12   14   16
+[2,]   16   18   17
+[3,]   20   19   21
+[4,]   21   23   25
+
+

Note how the vector just gets reused, no error is thrown.

+

We can also perform these operations on two matrices

+
+
+
M1 <- matrix(c(10:21), nrow = 4, byrow = TRUE)
+M2 <- matrix(c(110:121), nrow =4, byrow = TRUE)
+
+M1
+
+
     [,1] [,2] [,3]
+[1,]   10   11   12
+[2,]   13   14   15
+[3,]   16   17   18
+[4,]   19   20   21
+
+
M2
+
+
     [,1] [,2] [,3]
+[1,]  110  111  112
+[2,]  113  114  115
+[3,]  116  117  118
+[4,]  119  120  121
+
+
M1 + M2
+
+
     [,1] [,2] [,3]
+[1,]  120  122  124
+[2,]  126  128  130
+[3,]  132  134  136
+[4,]  138  140  142
+
+

Note how elements in the same position of each matrix are added together

+

Note this also is true of vectors

+
+
+
v1 <- c(1,2,3)
+v2 <- c(4)
+
+v1 + v2
+
+
[1] 5 6 7
+
+
+
+
v3 <- c(5, 6)
+v1 + v3
+
+
[1] 6 8 8
+
+
+
+
v4 <- c(10, 11, 12, 13, 14, 15)
+
+v1 + v4
+
+
[1] 11 13 15 14 16 18
+
+

Exercise +Multiply, subtract, and divide the two matrices M1 and M2

+
+
+
# TODO multiply, subtract and divide M1 and M2
+
+
+

Matrix multiplication

+

We will only briefly touch on matrix multiplication, but one reason matrices are very important in R is that you can perform multiplication with them. Exactly how this is done is explained nicely in a math is fun tutorial.

+

Let’s try to multiply two matrices together. Remember our first matrix has 4 rows and 3 columns:

+
+
+
dim(M)
+
+
[1] 4 3
+
+

So our new matrix must have 3 rows

+
+
+
M2 <- matrix(c(5:19), nrow = 3, byrow = TRUE)
+M2
+
+
     [,1] [,2] [,3] [,4] [,5]
+[1,]    5    6    7    8    9
+[2,]   10   11   12   13   14
+[3,]   15   16   17   18   19
+
+

Let’s perform matrix multiplication with these

+
+
+
M %*% M2
+
+
     [,1] [,2] [,3] [,4] [,5]
+[1,]  340  373  406  439  472
+[2,]  430  472  514  556  598
+[3,]  520  571  622  673  724
+[4,]  610  670  730  790  850
+
+

Performing functions on a matrix

+

So far, a matrix has looked a lot like a dataframe with some limitations. One of the places where matrices become the most useful is performing statistical functions because all items in a matrix are of the same type.

+

For this next section, let’s use some data that I downloaded from the social security office. This has the top 100 boy and girl names by state for 2020.

+

We can now read in the data and convert it to a matrix

+
+
+
names_mat <- read.csv(here("class_8-10_data", "boy_name_counts.csv"),
+                           row.names = 1) %>%
+  as.matrix()
+
+names_mat[1:5, 1:5]
+
+
        Alabama Alaska Arizona Arkansas California
+William     366     36     174      152       1021
+James       304     34     215      105       1148
+John        267     20      97       94        623
+Elijah      254     42     284      143       1586
+Noah        243     35     397      138       2625
+
+

Above you can see that we have the number of males with each name in each state. Looking at the structure, we can see that it is an integer matrix.

+
+
+
str(names_mat)
+
+
 int [1:40, 1:20] 366 304 267 254 243 207 187 183 173 166 ...
+ - attr(*, "dimnames")=List of 2
+  ..$ : chr [1:40] "William" "James" "John" "Elijah" ...
+  ..$ : chr [1:20] "Alabama" "Alaska" "Arizona" "Arkansas" ...
+
+

We can now explore this data set using many of the functions you have already learned such as rowSums and colSums

+

Basic functions

+

First, lets find the sum for all of the rows - how many total babies were named each name?

+
+
+
rowSums(names_mat)
+
+
  William     James      John    Elijah      Noah      Liam     Mason 
+     5067      5005      3295      5921      8118      8512      4040 
+   Oliver     Henry   Jackson    Samuel     Jaxon     Asher   Grayson 
+     5961      4121      3599      3549      2571      3212      2938 
+     Levi   Michael    Carter  Benjamin   Charles     Wyatt    Thomas 
+     3717      4061      3017      5265      2427      3255      2475 
+    Aiden      Luke     David      Owen    Daniel     Logan    Joseph 
+     3872      3230      3554      3369      4223      3933      3356 
+    Lucas    Joshua      Jack Alexander  Maverick   Gabriel     Ethan 
+     4794      2645      3399      4530      2588      3091      4322 
+      Eli     Isaac    Hunter      Ezra  Theodore 
+     2202      3029      1940      2999      3409 
+
+

And then the columns - how many babies were included from each state?

+
+
+
colSums(names_mat)
+
+
    Alabama      Alaska     Arizona    Arkansas  California 
+       5687         986        7371        3428       41256 
+   Colorado Connecticut    Delaware     Florida     Georgia 
+       6396        4211        1117       21661       11791 
+     Hawaii       Idaho    Illinois     Indiana        Iowa 
+       1111        2306       13407        8283        3508 
+     Kansas    Kentucky   Louisiana       Maine    Maryland 
+       3630        5518        4955        1372        6617 
+
+

What if we want to find the percent of children with a given name across all states (divide the value by the row sum * 100) - what percent of total babies for each name came from each state:

+
+
+
percent_mat <- names_mat / rowSums(names_mat) * 100
+percent_mat[1:5, 1:5]
+
+
         Alabama    Alaska  Arizona Arkansas California
+William 7.223209 0.7104796 3.433985 2.999803   20.14999
+James   6.073926 0.6793207 4.295704 2.097902   22.93706
+John    8.103187 0.6069803 2.943854 2.852807   18.90744
+Elijah  4.289816 0.7093396 4.796487 2.415133   26.78602
+Noah    2.993348 0.4311407 4.890367 1.699926   32.33555
+
+

Remember from above that division using a vector will divide every element of a row by one value, so we can only do this using rowSums. In a few minutes we will discuss how do do this on the columns.

+

Summary functions

+

We can also find the minimum, maximum, mean, and median values of the whole matrix and any column. First, lets get summary data for the whole matrix using summary

+
+
+
summary(names_mat)[ , 1:3]
+
+
    Alabama           Alaska         Arizona     
+ Min.   : 61.00   Min.   :14.00   Min.   : 84.0  
+ 1st Qu.: 94.75   1st Qu.:18.00   1st Qu.:143.8  
+ Median :125.00   Median :22.50   Median :174.5  
+ Mean   :142.18   Mean   :24.65   Mean   :184.3  
+ 3rd Qu.:163.00   3rd Qu.:28.00   3rd Qu.:191.2  
+ Max.   :366.00   Max.   :44.00   Max.   :451.0  
+
+

You can see that this calculates the min, max, mean, median, and quartiles for the columns.

+

What if we just want the minimum value for the “Alabama” names? We can run min while subsetting to just the column of interest

+
+
+
min(names_mat[, "Alabama"])
+
+
[1] 61
+
+

We can do the same for the rows Lets try this for “William”

+
+
+
min(names_mat["William",])
+
+
[1] 29
+
+

What if we wanted to find the smallest value in the whole matrix?

+
+
+
min(names_mat)
+
+
[1] 13
+
+

max works the same as min

+

Exercise +Find the maximum value in the for “Noah”

+
+
+
# TODO Find the maximum value for Noah and the whole matrix
+
+
+

We can also find the mean, median, and standard deviation of any part of the matrix

+

By row:

+
+
+
mean(names_mat["William", ])
+
+
[1] 253.35
+
+
+
+
median(names_mat["William", ])
+
+
[1] 178
+
+
+
+
sd(names_mat["William", ])
+
+
[1] 239.8189
+
+

By column:

+
+
+
mean(names_mat[ , "Alabama"])
+
+
[1] 142.175
+
+
+
+
median(names_mat[ , "Alabama"])
+
+
[1] 125
+
+
+
+
sd(names_mat[ , "Alabama"])
+
+
[1] 67.66012
+
+

Transposition

+

One important quality of a matrix is being able to transpose it to interchange the rows and columns - here the rows become columns and columns become rows. We transpose using t() to the matrix. Let’s first look at this using the matrix we started with

+
+
+
M <- matrix(c(10:21), nrow = 4, byrow = TRUE)
+M
+
+
     [,1] [,2] [,3]
+[1,]   10   11   12
+[2,]   13   14   15
+[3,]   16   17   18
+[4,]   19   20   21
+
+
+
+
t(M)
+
+
     [,1] [,2] [,3] [,4]
+[1,]   10   13   16   19
+[2,]   11   14   17   20
+[3,]   12   15   18   21
+
+

Note that the output of transposing either a matrix or a data frame will be a matrix (because the type within a column of a data frame must be the same).

+
+
+
df
+
+
   A    B     C
+1 10  cat  TRUE
+2 11  dog  TRUE
+3 12 fish FALSE
+
+
t(df)
+
+
  [,1]   [,2]   [,3]   
+A "10"   "11"   "12"   
+B "cat"  "dog"  "fish" 
+C "TRUE" "TRUE" "FALSE"
+
+
str(df)
+
+
'data.frame':   3 obs. of  3 variables:
+ $ A: int  10 11 12
+ $ B: chr  "cat" "dog" "fish"
+ $ C: logi  TRUE TRUE FALSE
+
+
str(t(df))
+
+
 chr [1:3, 1:3] "10" "cat" "TRUE" "11" "dog" "TRUE" "12" "fish" ...
+ - attr(*, "dimnames")=List of 2
+  ..$ : chr [1:3] "A" "B" "C"
+  ..$ : NULL
+
+

Note how after the transposition, all items in the original df are now characters and we no longer have a dataframe.

+

Now let’s try this transposition on the names matrix we’ve been working with

+
+
+
transposed_mat <- t(names_mat)
+
+transposed_mat[1:3,1:3]
+
+
        William James John
+Alabama     366   304  267
+Alaska       36    34   20
+Arizona     174   215   97
+
+

Note how the columns are now names and the rows are now states.

+

Remember the note above where we could only divide by the rowSums? Now we can use this transposition to figure out the percent of children in each state with a given name (divide the value by the column sum * 100)

+
+
+
state_percents <- transposed_mat / rowSums(transposed_mat) * 100
+
+state_percents <- t(state_percents)
+
+state_percents[1:3, 1:3]
+
+
         Alabama   Alaska  Arizona
+William 6.435731 3.651116 2.360602
+James   5.345525 3.448276 2.916836
+John    4.694918 2.028398 1.315968
+
+

Above we did this in several steps, but we can also do in in one step:

+
+
+
state_percents_2 <- t(t(names_mat) / colSums(names_mat)) * 100
+
+identical(state_percents, state_percents_2)
+
+
[1] TRUE
+
+

Statistical tests

+

We can also use matrices to perform statistical tests, like t-tests. For instance, are the names Oliver and Noah, or Oliver and Thomas used different amounts?

+

First, let’s normalize the data to account for the fact that each state reported different numbers of births. To do this normalization, let’s first divide each value by the total number of children reported for that state. Remember, we need to first transpose the matrix to be able to divide by the colSums

+
+
+
normalized_mat <- t(t(names_mat) / colSums(names_mat))
+
+
+

Now that we have normalized values, we can do a t-test.

+
+
+
normalized_mat["Oliver", 1:3]
+
+
   Alabama     Alaska    Arizona 
+0.03217865 0.04361055 0.04124271 
+
+
normalized_mat["Noah", 1:3]
+
+
   Alabama     Alaska    Arizona 
+0.04272903 0.03549696 0.05385972 
+
+
normalized_mat["Thomas", 1:3]
+
+
   Alabama     Alaska    Arizona 
+0.02092492 0.02028398 0.01329535 
+
+
+
+
t.test(normalized_mat["Oliver",], normalized_mat["Noah",])
+
+

+    Welch Two Sample t-test
+
+data:  normalized_mat["Oliver", ] and normalized_mat["Noah", ]
+t = -1.1861, df = 36.481, p-value = 0.2433
+alternative hypothesis: true difference in means is not equal to 0
+95 percent confidence interval:
+ -0.010275411  0.002689617
+sample estimates:
+ mean of x  mean of y 
+0.04133411 0.04512700 
+
+

Between Oliver and Noah, there does not seem to be a difference with the data we have. What about Oliver and Thomas?

+
+
+
t.test(normalized_mat["Oliver",], normalized_mat["Thomas",])
+
+

+    Welch Two Sample t-test
+
+data:  normalized_mat["Oliver", ] and normalized_mat["Thomas", ]
+t = 9.3268, df = 21.544, p-value = 5.121e-09
+alternative hypothesis: true difference in means is not equal to 0
+95 percent confidence interval:
+ 0.01858464 0.02922945
+sample estimates:
+ mean of x  mean of y 
+0.04133411 0.01742706 
+
+

Here we can see that there is a difference between the mean values for Oliver and Thomas using a t.test

+

Using dataframes and matricies

+

For many of the tidyverse functions you’ve learned so far, a data frame is required. Fortunately, it is very easy to change between a data frame and a matrix.

+
+
+
normalized_dat <- data.frame(normalized_mat)
+
+str(normalized_dat)
+
+
'data.frame':   40 obs. of  20 variables:
+ $ Alabama    : num  0.0644 0.0535 0.0469 0.0447 0.0427 ...
+ $ Alaska     : num  0.0365 0.0345 0.0203 0.0426 0.0355 ...
+ $ Arizona    : num  0.0236 0.0292 0.0132 0.0385 0.0539 ...
+ $ Arkansas   : num  0.0443 0.0306 0.0274 0.0417 0.0403 ...
+ $ California : num  0.0247 0.0278 0.0151 0.0384 0.0636 ...
+ $ Colorado   : num  0.0369 0.0364 0.0211 0.0317 0.0408 ...
+ $ Connecticut: num  0.0356 0.0359 0.0302 0.024 0.0503 ...
+ $ Delaware   : num  0.0269 0.0322 0.0251 0.0403 0.0466 ...
+ $ Florida    : num  0.0244 0.0278 0.018 0.0436 0.0608 ...
+ $ Georgia    : num  0.0469 0.0369 0.0317 0.0446 0.0494 ...
+ $ Hawaii     : num  0.0261 0.0297 0.0252 0.0342 0.0513 ...
+ $ Idaho      : num  0.0412 0.0399 0.0173 0.0351 0.0308 ...
+ $ Illinois   : num  0.0345 0.0334 0.023 0.0309 0.053 ...
+ $ Indiana    : num  0.0333 0.0321 0.0164 0.0396 0.0391 ...
+ $ Iowa       : num  0.0425 0.0296 0.0154 0.0299 0.0371 ...
+ $ Kansas     : num  0.0342 0.0336 0.0226 0.0358 0.0372 ...
+ $ Kentucky   : num  0.0448 0.0399 0.0268 0.0419 0.0379 ...
+ $ Louisiana  : num  0.0367 0.0349 0.0367 0.0478 0.044 ...
+ $ Maine      : num  0.0277 0.0248 0.016 0.0255 0.035 ...
+ $ Maryland   : num  0.0328 0.0378 0.0213 0.0293 0.0533 ...
+
+

Once we can move between matrices and data frames, we can start to tidy our data for plotting purposes. Let’s plot the distribution of name usage as a violin plot. Here we want the counts to be the y axis and the names to be the y axis.

+

The first thing we need to do is make our matrix into a data frame

+
+
+
names_dat <- data.frame(names_mat)
+
+
+

Next, we will want the names to be a column rather than the row names. We can do this using $ or tibble::rownames_to_column

+
+
+
names_dat <- rownames_to_column(names_dat, "name")
+
+names_dat[1:3,1:3]
+
+
     name Alabama Alaska
+1 William     366     36
+2   James     304     34
+3    John     267     20
+
+
# To set using $
+# names_dat$name <- rownames(names_dat)
+
+
+

Next, we need to pivot_longer from tidyr. We want to take everything but the names column

+
+
+
pivot_columns <- colnames(names_dat)[colnames(names_dat) != "name"]
+
+names_dat <- pivot_longer(names_dat, cols = all_of(pivot_columns),
+                          names_to = "state", values_to = "count")
+
+
+

Note, we can use the pipe %>% from dplyr to put all of this into one statement.

+
+
+
# Here we will specify the columns to keep first
+pivot_columns <- colnames(names_mat)
+
+names_dat <- names_mat %>% 
+  data.frame %>% 
+  rownames_to_column("name") %>% 
+  pivot_longer(cols = all_of(pivot_columns), 
+               names_to = "state", values_to = "count")
+
+
+

With this new data frame, we can now plot the distribution of names

+
+
+
# I first set the theme
+theme_set(theme_classic(base_size = 10))
+
+ggplot(names_dat, aes(x = name, y = count,
+                              fill = name)) + 
+  geom_violin() +
+  theme(axis.text.x = element_text(angle = 90,
+                                   vjust = 0.5, hjust=1)) # rotate x axis
+
+

+
+

There are a few outliers here, almost certainly California. As we discussed above, normalizing the data helps put everything onto the same scale.

+

Exercise +Can you make the same plot as above but use our normalized values?

+
+
+
# TODO make plot above with normalized values
+# Hint start with normalized_dat
+
+
+

I’m a biologist, why should I care?

+
    +
  • Many of your datasets can be analyzed using matrices +
      +
    • If you analyze RNA-seq data, all of the processing and differential testing will be done in a matrix through DESeq2
    • +
    • If you analyze protein data, that is best done in a matrix
    • +
    • Single cell RNA-seq data is analyzed using matrices, especially sparse matrices
    • +
    • Even just many measurements can be analyzed using a matrix
    • +
  • +
  • Matrices are efficient object types for most types of analysis you would want to do
  • +
+

Acknowldgements and additional references

+

The content of this class borrows heavily from previous tutorials:

+ +
+ + +
+ +
+
+ + + + + +
+ + + + + + + diff --git a/posts/posts.json b/posts/posts.json index a6093cd..2d22c45 100644 --- a/posts/posts.json +++ b/posts/posts.json @@ -13,7 +13,24 @@ "categories": [], "contents": "\nThe Rmarkdown for this document is: https://github.com/rnabioco/bmsc-7810-pbda/blob/main/_posts/2023-12-07-class-7-integrated-data-analysis/class-7-integrated-data-analysis.Rmd\nGoals for today\nLearn new ways to get data\nUsing a webscraping library\nUsing the NCBI API\nA look at interactive plots\nImporting Data from Varying Sources\nCSV Files: Use read.csv() from base R or read_csv() from readr package.\nExcel Files: read_excel() from the readxl package.\nDatabases: DBI and RSQLite or RMySQL for SQL databases.\nSPSS, Stata, SAS Files: Use haven package to read these file formats.\nGoogle Sheets: The googlesheets4 package for Google Sheets integration.\nWeb Scraping: rvest for scraping HTML/XML data.\nWeb APIs: Use httr for generic API requests and jsonlite for parsing JSON. Also NCBI specific rentrenz\nWeb Scraping\nWeb scraping is the process of extracting data from websites. This technique involves programmatically accessing web pages and then extracting useful information from them. We can then put that data into a dataframe.\n\n\n# Load libraries\n#install.packages(\"rvest\")\nlibrary(rvest)\nlibrary(ggplot2)\nlibrary(dplyr)\n\n# URL of the Wikipedia page\nurl <- \"https://en.wikipedia.org/wiki/List_of_tallest_buildings\"\n\n# Read HTML content from the URL\npage <- read_html(url)\n\n# Extract the second large table on the page:\n# - `html_nodes(\"table.wikitable\")` finds all nodes in the HTML that are tables with the class 'wikitable'\n# - `.[2]` selects the second node in the list (i.e., the second table)\n# - `html_table(fill = TRUE)` converts the HTML table into a data frame; `fill = TRUE` ensures that all rows have the same length\nbuildings_table <- page %>% \n html_nodes(\"table.wikitable\") %>% \n .[2] %>% # Select the second table\n html_table(fill = TRUE) %>% \n .[[1]]\n\n# Handle duplicate column names in the table:\n# - The code identifies columns named \"Height[14]\" and renames them to \"Height_m\" and \"Height_ft\"\n# - This is necessary because R does not allow duplicate column names in data frames\ncolnames(buildings_table)[colnames(buildings_table) == \"Height[14]\"] <- c(\"Height_m\", \"Height_ft\")\n\n# Process and clean the data:\n# - `select` is used to choose and rename certain columns from the table for further analysis\n# - `mutate` creates a new column `Height_m` where the height values are converted to numeric, removing any non-numeric characters\n# - `filter` removes rows where Height_m is NA (not available)\n# - Here, `Height_m` is intended to represent building heights in meters\nbuildings_data <- buildings_table %>%\n select(Rank, Building = Name, City, Country, Height_m, Floors, Year) %>%\n mutate(Height_m = as.numeric(gsub(\"[^0-9\\\\.]\", \"\", Height_m))) %>%\n filter(!is.na(Height_m))\n\n\n\n\nhead(buildings_data)\n\n# A tibble: 6 × 7\n Rank Building City Country Height_m Floors Year \n \n1 1 Burj Khalifa Dubai United… 828 163 (… 2010 \n2 2 Merdeka 118 Kual… Malays… 679. 118 (… 2023 \n3 3 Shanghai Tower Shan… China 632 128 (… 2015 \n4 4 Abraj Al-Bait Clock Tower Mecca Saudi … 601 120 (… 2012 \n5 5 Ping An International Fin… Shen… China 599. 115 (… 2017 \n6 6 Lotte World Tower Seoul South … 554. 123 (… 2017 \n\n\n\n# plot the heights of the tallest buildings\n\nggplot(buildings_data, aes(x = reorder(Building, Height_m), y = Height_m)) +\n geom_bar(stat = \"identity\") +\n coord_flip() +\n labs(title = \"Height of the World's Tallest Buildings\", x = \"Building\", y = \"Height (m)\")\n\n\n\n\n\n# distribution\nggplot(buildings_data, aes(x = Height_m)) +\n geom_histogram(bins = 30, fill = \"blue\", color = \"black\") +\n labs(title = \"Distribution of Building Heights\", x = \"Height (m)\", y = \"Count\")\n\n\n\n\n\n# height across years\nggplot(buildings_data, aes(x = Year, y = Height_m)) +\n geom_point() +\n labs(title = \"Building Height Over Years\", x = \"Year\", y = \"Height (m)\") +\n theme(axis.text.x = element_text(angle = 45, hjust = 1))\n\n\n#angle = 45 rotates the x-axis text labels by 45 degrees.\n#hjust = 1 aligns the labels at the end, which can help with readability after rotation.\n\n\n\n\n# heights by country\nggplot(buildings_data, aes(x = Country, y = Height_m)) +\n geom_boxplot() +\n coord_flip() + # for better readability of country names\n labs(title = \"Building Heights by Country\", x = \"Country\", y = \"Height (m)\")\n\n\n\n\n\n# cumulative height\nbuildings_data |>\n arrange(Year) |>\n mutate(CumulativeHeight = cumsum(Height_m)) |>\n ggplot(aes(x = Year, y = CumulativeHeight)) +\n geom_line(color = \"red\") +\n labs(title = \"Cumulative Height of Buildings Over Years\", x = \"Year\", y = \"Cumulative Height (m)\") +\n theme(axis.text.x = element_text(angle = 45, hjust = 1))\n\n\n\n\n\n# number by country\nbuildings_data |>\n count(Country) |>\n ggplot(aes(x = reorder(Country, n), y = n)) +\n geom_bar(stat = \"identity\", fill = \"blue\") +\n coord_flip() +\n labs(title = \"Number of Tall Buildings by Country\", x = \"Country\", y = \"Number of Buildings\")\n\n\n\nWorking with the NCBI Web APIs\nAccessing PubMed data in R can be efficiently done using the rentrez package, which is an R client for the NCBI Entrez API. Entrez is NCBI’s integrated database search system that covers various databases, including PubMed.\n\n\n#install.packages(\"rentrez\")\nlibrary(rentrez)\n\nentrez_dbs() #list of available databases\n\n [1] \"pubmed\" \"protein\" \"nuccore\" \n [4] \"ipg\" \"nucleotide\" \"structure\" \n [7] \"genome\" \"annotinfo\" \"assembly\" \n[10] \"bioproject\" \"biosample\" \"blastdbinfo\" \n[13] \"books\" \"cdd\" \"clinvar\" \n[16] \"gap\" \"gapplus\" \"grasp\" \n[19] \"dbvar\" \"gene\" \"gds\" \n[22] \"geoprofiles\" \"homologene\" \"medgen\" \n[25] \"mesh\" \"nlmcatalog\" \"omim\" \n[28] \"orgtrack\" \"pmc\" \"popset\" \n[31] \"proteinclusters\" \"pcassay\" \"protfam\" \n[34] \"pccompound\" \"pcsubstance\" \"seqannot\" \n[37] \"snp\" \"sra\" \"taxonomy\" \n[40] \"biocollections\" \"gtr\" \n\nPubmed Database\n\n\nentrez_db_summary(\"pubmed\")\n\n DbName: pubmed\n MenuName: PubMed\n Description: PubMed bibliographic record\n DbBuild: Build-2023.12.06.21.44\n Count: 36544432\n LastUpdate: 2023/12/06 21:44 \n\n\n\nentrez_db_searchable(\"pubmed\") \n\nSearchable fields for database 'pubmed'\n ALL All terms from all searchable fields \n UID Unique number assigned to publication \n FILT Limits the records \n TITL Words in title of publication \n MESH Medical Subject Headings assigned to publication \n MAJR MeSH terms of major importance to publication \n JOUR Journal abbreviation of publication \n AFFL Author's institutional affiliation and address \n ECNO EC number for enzyme or CAS registry number \n SUBS CAS chemical name or MEDLINE Substance Name \n PDAT Date of publication \n EDAT Date publication first accessible through Entrez \n VOL Volume number of publication \n PAGE Page number(s) of publication \n PTYP Type of publication (e.g., review) \n LANG Language of publication \n ISS Issue number of publication \n SUBH Additional specificity for MeSH term \n SI Cross-reference from publication to other databases \n MHDA Date publication was indexed with MeSH terms \n TIAB Free text associated with Abstract/Title \n OTRM Other terms associated with publication \n COLN Corporate Author of publication \n CNTY Country of publication \n PAPX MeSH pharmacological action pre-explosions \n GRNT NIH Grant Numbers \n MDAT Date of last modification \n CDAT Date of completion \n PID Publisher ID \n FAUT First Author of publication \n FULL Full Author Name(s) of publication \n FINV Full name of investigator \n TT Words in transliterated title of publication \n LAUT Last Author of publication \n PPDT Date of print publication \n EPDT Date of Electronic publication \n LID ELocation ID \n CRDT Date publication first accessible through Entrez \n BOOK ID of the book that contains the document \n ED Section's Editor \n ISBN ISBN \n PUBN Publisher's name \n AUCL Author Cluster ID \n EID Extended PMID \n DSO Additional text from the summary \n AUID Author Identifier \n PS Personal Name as Subject \n COIS Conflict of Interest Statements \n WORD Free text associated with publication \n P1DAT Date publication first accessible through Solr \n\n\n\nsearch_results <- entrez_search(db=\"pubmed\", term=\"RNA sequencing\", retmax = 20)\n\nsearch_results\n\nEntrez search result with 308466 hits (object contains 20 IDs and no web_history object)\n Search term (as translated): \"sequence analysis, rna\"[MeSH Terms] OR (\"sequence ... \n\n\n\nsearch_results$ids\n\n [1] \"38059393\" \"38059352\" \"38059347\" \"38059344\" \"38059330\" \"38059002\"\n [7] \"38059001\" \"38058845\" \"38058831\" \"38058713\" \"38058697\" \"38058280\"\n[13] \"38058092\" \"38058084\" \"38058065\" \"38058040\" \"38058007\" \"38057925\"\n[19] \"38057799\" \"38057716\"\n\n\n\n# Fetch details of the articles\narticle_summaries <- entrez_summary(db=\"pubmed\", id=search_results$ids[1:20])\narticle_summaries\n\nList of 20 esummary records. First record:\n\n $`38059393`\nesummary result with 42 items:\n [1] uid pubdate epubdate \n [4] source authors lastauthor \n [7] title sorttitle volume \n[10] issue pages lang \n[13] nlmuniqueid issn essn \n[16] pubtype recordstatus pubstatus \n[19] articleids history references \n[22] attributes pmcrefcount fulljournalname \n[25] elocationid doctype srccontriblist \n[28] booktitle medium edition \n[31] publisherlocation publishername srcdate \n[34] reportnumber availablefromurl locationlabel \n[37] doccontriblist docdate bookname \n[40] chapter sortpubdate sortfirstauthor \n\n\n\nrecords <- extract_from_esummary(article_summaries, c(\"pubdate\", \"pmcrefcount\", \"title\", \"fulljournalname\"))\n\nrecords_df <- as.data.frame(t(records))\ncolnames(records_df) <- c(\"pubdate\", \"pmcrefcount\", \"title\", \"fulljournalname\")\n\nrecords_df$pmcrefcount <- as.numeric(as.character(records_df$pmcrefcount))\nrecords_df$fulljournalname <- unlist(records_df$fulljournalname)\nrecords_df$pubdate <- unlist(records_df$pubdate)\nrecords_df$title <- unlist(records_df$title)\n\n# Remove rows with NA in 'pmcrefcount' or 'fulljournalname'\nrecords_df <- records_df %>% filter(!is.na(pmcrefcount) & !is.na(fulljournalname))\n\n\n\n\nrecords_df\n\n[1] pubdate pmcrefcount title fulljournalname\n<0 rows> (or 0-length row.names)\n\n\n\n# Plotting the total citations per journal\nggplot(records_df, aes(x = fulljournalname, y = pmcrefcount)) +\n geom_bar(stat = \"identity\") +\n theme(axis.text.x = element_text(angle = 45, hjust = 1)) +\n labs(title = \"Citations per Journal\", x = \"Journal\", y = \"Total Citations\") +\n coord_flip()\n\n\n\nNCBI Gene Database\n\n\nentrez_db_summary(\"gene\") \n\n DbName: gene\n MenuName: Gene\n Description: Gene database\n DbBuild: Build231204-1643.1\n Count: 72097485\n LastUpdate: 2023/12/06 12:32 \n\n\n\nentrez_db_searchable(\"gene\")\n\nSearchable fields for database 'gene'\n ALL All terms from all searchable fields \n UID Unique number assigned to a gene record \n FILT Limits the records \n TITL gene or protein name \n WORD Free text associated with record \n ORGN scientific and common names of organism \n MDAT The last date on which the record was updated \n CHR Chromosome number or numbers; also 'mitochondrial', 'unknown' properties \n MV Chromosomal map location as displayed in MapViewer \n GENE Symbol or symbols of the gene \n ECNO EC number for enzyme or CAS registry number \n MIM MIM number from OMIM \n DIS Name(s) of diseases associated with this gene. When available, OMIM name will be used \n ACCN Nucleotide or protein accession(s) associated with this gene \n UGEN UniGene cluster number for this gene \n PROP Properties of Gene record \n CDAT The date on which this record first appeared \n NCAC nucleotide accessions of sequences \n NUID nucleotide uids of sequences \n PACC protein accessions \n PUID protein uids \n PMID PubMed ids of accessions linked to the record \n TID taxonomy id \n GO Gene Ontology \n DOM Domain Name \n DDAT The date on which the record was discontinued \n CPOS Chromosome base position \n GFN Gene full name \n PFN Protein full name \n GL Gene length \n XC Exon count \n GRP Relationships for this gene \n PREF Preferred symbol of the gene \n AACC Assembly accession \n ASM Assembly name \n EXPR Gene expression \n\n\n\nsearch_results <- entrez_search(db=\"gene\",term = \"transcription factor\", retmax = 20)\n\nsearch_results$ids\n\n [1] \"7157\" \"1956\" \"7124\" \"7422\" \"3569\" \"7040\" \"22059\" \"4524\" \n [9] \"3091\" \"2064\" \"11816\" \"2099\" \"3586\" \"351\" \"6774\" \"672\" \n[17] \"3845\" \"627\" \"673\" \"4318\" \n\n\n\ngene_summaries <- entrez_summary(db=\"gene\", id=search_results$ids[1:20])\ngene_summaries\n\nList of 20 esummary records. First record:\n\n $`7157`\nesummary result with 20 items:\n [1] uid name description \n [4] status currentid chromosome \n [7] geneticsource maplocation otheraliases \n[10] otherdesignations nomenclaturesymbol nomenclaturename \n[13] nomenclaturestatus mim genomicinfo \n[16] geneweight summary chrsort \n[19] chrstart organism \n\n\n\nrecords <- extract_from_esummary(gene_summaries, c(\"name\", \"description\"))\n\nrecords_df <- as.data.frame(t(records))\ncolnames(records_df) <- c(\"name\", \"description\")\n\nrecords_df$name <- unlist(records_df$name)\nrecords_df$description <- unlist(records_df$description)\n\n\n\n\nrecords_df\n\n name description\n7157 TP53 tumor protein p53\n1956 EGFR epidermal growth factor receptor\n7124 TNF tumor necrosis factor\n7422 VEGFA vascular endothelial growth factor A\n3569 IL6 interleukin 6\n7040 TGFB1 transforming growth factor beta 1\n22059 Trp53 transformation related protein 53\n4524 MTHFR methylenetetrahydrofolate reductase\n3091 HIF1A hypoxia inducible factor 1 subunit alpha\n2064 ERBB2 erb-b2 receptor tyrosine kinase 2\n11816 Apoe apolipoprotein E\n2099 ESR1 estrogen receptor 1\n3586 IL10 interleukin 10\n351 APP amyloid beta precursor protein\n6774 STAT3 signal transducer and activator of transcription 3\n672 BRCA1 BRCA1 DNA repair associated\n3845 KRAS KRAS proto-oncogene, GTPase\n627 BDNF brain derived neurotrophic factor\n673 BRAF B-Raf proto-oncogene, serine/threonine kinase\n4318 MMP9 matrix metallopeptidase 9\n\nNCBI Protein Sequences\n\n\nsearch_results <- entrez_search(db=\"protein\", term=\"BRCA1[Gene Name] AND human[Organism]\")\nprotein_ids <- search_results$ids\n\nprotein_sequences <- entrez_fetch(db=\"protein\", id=protein_ids, rettype=\"fasta\")\n\nhead(protein_sequences)\n\n[1] \">WPF70740.1 BRCA1, partial [Homo sapiens]\\nDLILLKIPLIRQLIA\\n\\n>sp|P38398.2|BRCA1_HUMAN RecName: Full=Breast cancer type 1 susceptibility protein; AltName: Full=RING finger protein 53; AltName: Full=RING-type E3 ubiquitin transferase BRCA1\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPE\\nGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLE\\nESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAES\\nAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLI\\nTEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPK\\nRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNG\\nFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\\n\\n>WOV89523.1 BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPE\\nGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLE\\nESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAES\\nAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLI\\nTEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPK\\nRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGVSVVKELSSFTLGTGVHPIVVVQPDAWTEDNG\\nFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\\n\\n>WOV89522.1 BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPE\\nGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLE\\nESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAES\\nAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLI\\nTEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPK\\nRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQPCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNG\\nFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\\n\\n>WOV89521.1 BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPE\\nGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLE\\nESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAES\\nAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLI\\nTEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQFIKERKMLNEHDFEVRGDVVNGRNHQGPK\\nRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNG\\nFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\\n\\n>WOV89520.1 BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPE\\nGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLE\\nESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAES\\nAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLI\\nTEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFCVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPK\\nRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNG\\nFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\\n\\n>WOV89519.1 BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDNIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPE\\nGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLE\\nESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAES\\nAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLI\\nTEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPK\\nRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNG\\nFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\\n\\n>WOP78653.1 truncated BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDL\\n\\n>WOP78652.1 truncated BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHS\\nRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQ\\nGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRI\\nPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSS\\nTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTV\\nNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSP\\nSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENL\\nLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSES\\nQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTM\\nQHNLIKLQQKWLN\\n\\n>WOP78651.1 truncated BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDFLG\\n\\n>WOP78650.1 truncated BRCA1 [Homo sapiens]\\nMDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITK\\nRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQS\\nEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQ\\nGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHA\\nSSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCER\\nKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVD\\nEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEP\\nQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGD\\nSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRN\\nLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPEL\\nKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSIS\\nLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFMVVPKIIEMTQKALSIHWDMKLTTVGKQA\\n\\n>WOK84184.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQ\\nFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTIS\\nR\\n\\n>WOK84183.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQ\\nFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTIS\\nR\\n\\n>WOK84182.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQF\\nRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISR\\n\\n>WOK84181.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPK\\nLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMG\\n\\n>WOK84180.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGP\\nKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMG\\n\\n>WOK84179.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGR\\nNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMG\\n\\n>WOK84178.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCS\\nETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEEN\\n\\n>WOK84177.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKE\\nSSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQST\\nRHSTV\\n\\n>WOK84176.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\\nEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAV\\nFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHST\\nV\\n\\n\"\n\n\n\n# Splitting the fetched data into individual sequences\nsequences <- unlist(strsplit(protein_sequences, \">\"))\nsequences <- sequences[nzchar(sequences)] # Remove empty elements\n\n# Function to parse a single FASTA formatted sequence\nparse_fasta <- function(fasta) {\n lines <- strsplit(fasta, \"\\n\")[[1]]\n header <- lines[1]\n sequence <- paste(lines[-1], collapse = \"\")\n return(list(header = header, sequence = sequence))\n}\n\n# Parse all sequences\nparsed_sequences <- lapply(sequences, parse_fasta)\n\n# Extracting headers and sequences\nheaders <- sapply(parsed_sequences, function(x) x$header)\nsequences <- sapply(parsed_sequences, function(x) x$sequence)\n\n# Create a dataframe\ndf <- data.frame(Header = headers, Sequence = sequences)\n\ndf\n\n Header\n1 WPF70740.1 BRCA1, partial [Homo sapiens]\n2 sp|P38398.2|BRCA1_HUMAN RecName: Full=Breast cancer type 1 susceptibility protein; AltName: Full=RING finger protein 53; AltName: Full=RING-type E3 ubiquitin transferase BRCA1\n3 WOV89523.1 BRCA1 [Homo sapiens]\n4 WOV89522.1 BRCA1 [Homo sapiens]\n5 WOV89521.1 BRCA1 [Homo sapiens]\n6 WOV89520.1 BRCA1 [Homo sapiens]\n7 WOV89519.1 BRCA1 [Homo sapiens]\n8 WOP78653.1 truncated BRCA1 [Homo sapiens]\n9 WOP78652.1 truncated BRCA1 [Homo sapiens]\n10 WOP78651.1 truncated BRCA1 [Homo sapiens]\n11 WOP78650.1 truncated BRCA1 [Homo sapiens]\n12 WOK84184.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n13 WOK84183.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n14 WOK84182.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n15 WOK84181.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n16 WOK84180.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n17 WOK84179.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n18 WOK84178.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n19 WOK84177.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n20 WOK84176.1 breast cancer type 1 susceptibility protein isoform 1, partial [Homo sapiens]\n Sequence\n1 DLILLKIPLIRQLIA\n2 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPEGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLEESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAESAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLITEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPKRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNGFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\n3 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPEGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLEESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAESAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLITEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPKRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGVSVVKELSSFTLGTGVHPIVVVQPDAWTEDNGFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\n4 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPEGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLEESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAESAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLITEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPKRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQPCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNGFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\n5 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPEGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLEESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAESAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLITEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQFIKERKMLNEHDFEVRGDVVNGRNHQGPKRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNGFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\n6 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPEGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLEESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAESAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLITEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFCVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPKRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNGFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\n7 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDNIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDLRNPEQSTSEKAVLTSQKSSEYPISQNPEGLSADKFEVSADSSTSKNKEPGVERSSPSKCPSLDDRWYMHSCSGSLQNRNYPSQEELIKVVDVEEQQLEESGPHDLTETSYLPRQDLEGTPYLESGISLFSDDPESDPSEDRAPESARVGNIPSSTSALKVPQLKVAESAQSPAAAHTTDTAGYNAMEESVSREKPELTASTERVNKRMSMVVSGLTPEEFMLVYKFARKHHITLTNLITEETTHVVMKTDAEFVCERTLKYFLGIAGGKWVVSYFWVTQSIKERKMLNEHDFEVRGDVVNGRNHQGPKRARESQDRKIFRGLEICCYGPFTNMPTDQLEWMVQLCGASVVKELSSFTLGTGVHPIVVVQPDAWTEDNGFHAIGQMCEAPVVTREWVLDSVALYQCQELDTYLIPQIPHSHY\n8 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQEMAELEAVLEQHGSQPSNSYPSIISDSSALEDL\n9 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFENPKGLIHGCSKDNRNDTEGFKYPLGHEVNHSRETSIEMEESELDAQYLQNTFKVSKRQSFAPFSNPGNAEEECATFSAHSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTVATECLSKNTEENLLSLKNSLNDCSNQVILAKASQEHHLSEETKCSASLFSSQCSELEDLTANTNTQDPFLIGSSKQMRHQSESQGVGLSDKELVSDDEERGTGLEENNQEEQSMDSNLGEAASGCESETSVSEDCSGLSSQSDILTTQQRDTMQHNLIKLQQKWLN\n10 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDFLG\n11 MDLSALRVEEVQNVINAMQKILECPICLELIKEPVSTKCDHIFCKFCMLKLLNQKKGPSQCPLCKNDITKRSLQESTRFSQLVEELLKIICAFQLDTGLEYANSYNFAKKENNSPEHLKDEVSIIQSMGYRNRAKRLLQSEPENPSLQETSLSVQLSNLGTVRTLRTKQRIQPQKTSVYIELGSDSSEDTVNKATYCSVGDQELLQITPQGTRDEISLDSAKKAACEFSETDVTNTEHHQPSNNDLNTTEKRAAERHPEKYQGSSVSNLHVEPCGTNTHASSLQHENSSLLLTKDRMNVEKAEFCNKSKQPGLARSQHNRWAGSKETCNDRRTPSTEKKVDLNADPLCERKEWNKQKLPCSENPRDTEDVPWITLNSSIQKVNEWFSRSDELLGSDDSHDGESESNAKVADVLDVLNEVDEYSGSSEKIDLLASDPHEALICKSERVHSKSVESNIEDKIFGKTYRKKASLPNLSHVTENLIIGAFVTEPQIIQERPLTNKLKRKRRPTSGLHPEDFIKKADLAVQKTPEMINQGTNQTEQNGQVMNITNSGHENKTKGDSIQNEKNPNPIESLEKESAFKTKAEPISSSISNMELELNIHNSKAPKKNRLRRKSSTRHIHALELVVSRNLSPPNCTELQIDSCSSSEEIKKKKYNQMPVRHSRNLQLMEGKEPATGAKKSNKPNEQTSKRHDSDTFPELKLTNAPGSFTKCSNTSELKEFVNPSLPREEKEEKLETVKVSNNAEDPKDLMLSGERVLQTERSVESSSISLVPGTDYGTQESISLLEVSTLGKAKTEPNKCVSQCAAFMVVPKIIEMTQKALSIHWDMKLTTVGKQA\n12 HSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISR\n13 HSGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISR\n14 SGSLKKQSPKVTFECEQKEENQGKNESNIKPVQTVNITAGFPVVGQKDKPVDNAKCSIKGGSRFCLSSQFRGNETGLITPNKHGLLQNPYRIPPLFPIKSFVKTKCKKNLLEENFEEHSMSPEREMGNENIPSTVSTISR\n15 PEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMG\n16 SPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMG\n17 EHSMSPEREMGNENIPSTVSTISRNNIRENVFKEASSSNINEVGSSTNEVGSSINEIGSSDENIQAELGRNRGPKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMG\n18 PKLNAMLRLGVLQPEVYKQSLPGSNCKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEEN\n19 CKHPEIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTV\n20 EIKKQEYEEVVQTVNTDFSPYLISDNLEQPMGSSHASQVCSETPDDLLDDGEIKEDTSFAENDIKESSAVFSKSVQKGELSRSPSPFTHTHLAQGYRRGAKKLESSEENLSSEDEELPCFQHLLFGKVNNIPSQSTRHSTV\n\nShiny - Interactive Plots\nhttps://shiny.posit.co/\nhttps://shiny.posit.co/r/gallery/\nhttps://shiny.posit.co/r/getstarted/shiny-basics/lesson1/index.html\nExplain SessionInfo:\n## SessionInfo\n\n\nsessionInfo()\n\nR version 4.2.2 (2022-10-31)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.6\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRblas.0.dylib\nLAPACK: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRlapack.dylib\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods \n[7] base \n\nother attached packages:\n [1] rentrez_1.2.3 rvest_1.0.3 lubridate_1.9.2 forcats_1.0.0 \n [5] stringr_1.5.0 dplyr_1.1.2 purrr_1.0.1 readr_2.1.4 \n [9] tidyr_1.3.0 tibble_3.2.1 ggplot2_3.4.2 tidyverse_2.0.0\n\nloaded via a namespace (and not attached):\n [1] tidyselect_1.2.0 xfun_0.39 bslib_0.4.2 \n [4] colorspace_2.1-0 vctrs_0.6.2 generics_0.1.3 \n [7] htmltools_0.5.5 yaml_2.3.7 utf8_1.2.3 \n[10] XML_3.99-0.14 rlang_1.1.1 jquerylib_0.1.4 \n[13] pillar_1.9.0 glue_1.6.2 withr_2.5.0 \n[16] selectr_0.4-2 lifecycle_1.0.3 munsell_0.5.0 \n[19] gtable_0.3.3 memoise_2.0.1 evaluate_0.21 \n[22] labeling_0.4.2 knitr_1.43 tzdb_0.4.0 \n[25] fastmap_1.1.1 curl_5.0.0 fansi_1.0.4 \n[28] highr_0.10 scales_1.2.1 cachem_1.0.8 \n[31] jsonlite_1.8.4 farver_2.1.1 distill_1.6 \n[34] hms_1.1.3 digest_0.6.31 stringi_1.7.12 \n[37] grid_4.2.2 cli_3.6.1 tools_4.2.2 \n[40] magrittr_2.0.3 sass_0.4.6 pkgconfig_2.0.3 \n[43] downlit_0.4.3 xml2_1.3.4 timechange_0.2.0\n[46] rmarkdown_2.22 httr_1.4.6 rstudioapi_0.14 \n[49] R6_2.5.1 compiler_4.2.2 \n\n\n\n\n", "preview": "posts/2023-12-07-class-7-integrated-data-analysis/class-7-integrated-data-analysis_files/figure-html5/unnamed-chunk-3-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", + "input_file": {} + }, + { + "path": "posts/2023-12-08-class-8-matricies/", + "title": "Class 8: Introduction to matricies", + "description": {}, + "author": [ + { + "name": "Kristen Wells", + "url": "https://github.com/kwells4" + } + ], + "date": "2023-12-07", + "categories": [], + "contents": "\n\nContents\nGoals for this class\nLoad packages\nDownload files\nWhat is a matrix?\nMatrix opearions\nBasic operations\nMatrix multiplication\n\nPerforming functions on a matrix\nBasic functions\nSummary functions\nTransposition\nStatistical tests\n\nUsing dataframes and matricies\nI’m a biologist, why should I care?\nAcknowldgements and additional references\n\nThe Rmarkdown for this class is on github\nGoals for this class\nLearn what is a matrix\nDescribe difference between matrix and data frame\nPerform mathematical functions\nConvert between matrix and data frames\nLoad packages\n\n\nlibrary(tidyverse)\nlibrary(here)\n\n\nDownload files\nBefore we get started, let’s download all of the files you will need for the next three classes.\n\n\n# conditionally download all of the files used in rmarkdown from github \nsource(\"https://raw.githubusercontent.com/rnabioco/bmsc-7810-pbda/main/_posts/2023-12-08-class-8-matricies/download_files.R\")\n\n\nWhat is a matrix?\nA Matrix is an 2 dimensional object in R. We create a matrix using the matrix function\n\n\nM <- matrix(c(10:21), nrow = 4, byrow = TRUE)\nM\n\n [,1] [,2] [,3]\n[1,] 10 11 12\n[2,] 13 14 15\n[3,] 16 17 18\n[4,] 19 20 21\n\nWe can also use as.matrix on an existing dataframe\n\n\ndf <- data.frame(\"A\" = c(10:13), \"B\" = c(14:17), \"C\" = (18:21))\ndf\n\n A B C\n1 10 14 18\n2 11 15 19\n3 12 16 20\n4 13 17 21\n\n\n\nnew_mat <- as.matrix(df)\nnew_mat\n\n A B C\n[1,] 10 14 18\n[2,] 11 15 19\n[3,] 12 16 20\n[4,] 13 17 21\n\nJust like data frames, we can name the rows and columns of the Matrix\n\n\nrownames(new_mat) <- c(\"first\", \"second\", \"third\", \"forth\")\ncolnames(new_mat) <- c(\"D\", \"E\", \"F\")\n\nnew_mat\n\n D E F\nfirst 10 14 18\nsecond 11 15 19\nthird 12 16 20\nforth 13 17 21\n\nWe can look at the structure of the matrix using str\n\n\nstr(new_mat)\n\n int [1:4, 1:3] 10 11 12 13 14 15 16 17 18 19 ...\n - attr(*, \"dimnames\")=List of 2\n ..$ : chr [1:4] \"first\" \"second\" \"third\" \"forth\"\n ..$ : chr [1:3] \"D\" \"E\" \"F\"\n\nHere you can see that the type of this structure is int because it is a matrix consisting of integers. We can also see the row names and column names.\nAs with data frames, we can check the size of the matrix using nrow, ncol and dim\n\n\nnrow(new_mat)\n\n[1] 4\n\nncol(new_mat)\n\n[1] 3\n\ndim(new_mat)\n\n[1] 4 3\n\nWe can also access data using brackets[\nSelecting a single value:\n\n\nnew_mat[1,2]\n\n[1] 14\n\nSelecting a section of the matrix:\n\n\nnew_mat[1:3,2]\n\n first second third \n 14 15 16 \n\nIf we don’t provide an index for the row, R will return all rows:\n\n\nnew_mat[, 3]\n\n first second third forth \n 18 19 20 21 \n\nThe same is true for the columns\n\n\nnew_mat[3,]\n\n D E F \n12 16 20 \n\nBecause this matrix has row and column names, we can also pull out data based on those\n\n\nnew_mat[\"second\", \"D\"]\n\n[1] 11\n\nExercise\nWhat value is in row 2 and column 3 of new_mat?\n\n\n# TODO find the value in the matrix at row 2 and column 3\n\n\nIf we can make a matrix from a data frame, what’s the difference?\nMatrices can only have values of one type –> integer, boolean, character, while a dataframe can be a mix of types:\n\n\ndf <- data.frame(\"A\" = c(10:12),\n \"B\" = c(\"cat\", \"dog\", \"fish\"),\n \"C\" = c(TRUE, TRUE, FALSE))\n\ndf\n\n A B C\n1 10 cat TRUE\n2 11 dog TRUE\n3 12 fish FALSE\n\n\n\nM <- as.matrix(df)\n\nM\n\n A B C \n[1,] \"10\" \"cat\" \"TRUE\" \n[2,] \"11\" \"dog\" \"TRUE\" \n[3,] \"12\" \"fish\" \"FALSE\"\n\n\n\ntypeof(df[,1])\n\n[1] \"integer\"\n\ntypeof(M[,1])\n\n[1] \"character\"\n\nBut Matrices can take any type of input\n\n\nM <- matrix(rep(c(TRUE, FALSE), 4), nrow = 4, byrow = TRUE)\nM\n\n [,1] [,2]\n[1,] TRUE FALSE\n[2,] TRUE FALSE\n[3,] TRUE FALSE\n[4,] TRUE FALSE\n\n\n\ntypeof(M[,1])\n\n[1] \"logical\"\n\nMatrix opearions\nIf you’ve taken linear algebra, you’ve probably worked with matrices before. These same matrix operations can be done in R\nBasic operations\nWe can do any of the mathematical operations for a matrix and one value. For example, we can add 5 to all values in a matrix, or subtract 2, or divide by 10\n\n\nM <- matrix(c(10:21), nrow = 4, byrow = TRUE)\nM\n\n [,1] [,2] [,3]\n[1,] 10 11 12\n[2,] 13 14 15\n[3,] 16 17 18\n[4,] 19 20 21\n\nM + 1\n\n [,1] [,2] [,3]\n[1,] 11 12 13\n[2,] 14 15 16\n[3,] 17 18 19\n[4,] 20 21 22\n\nM + 2\n\n [,1] [,2] [,3]\n[1,] 12 13 14\n[2,] 15 16 17\n[3,] 18 19 20\n[4,] 21 22 23\n\nM - 5\n\n [,1] [,2] [,3]\n[1,] 5 6 7\n[2,] 8 9 10\n[3,] 11 12 13\n[4,] 14 15 16\n\nM / 3\n\n [,1] [,2] [,3]\n[1,] 3.333333 3.666667 4\n[2,] 4.333333 4.666667 5\n[3,] 5.333333 5.666667 6\n[4,] 6.333333 6.666667 7\n\nM * 10\n\n [,1] [,2] [,3]\n[1,] 100 110 120\n[2,] 130 140 150\n[3,] 160 170 180\n[4,] 190 200 210\n\nWe can also provide a vector or another matrix to perform element-wise functions with.\n\n\nvector <- c(2, 3, 4, 5)\n\nM + vector\n\n [,1] [,2] [,3]\n[1,] 12 13 14\n[2,] 16 17 18\n[3,] 20 21 22\n[4,] 24 25 26\n\nHere you can see that each element of the vector is added to a row ie element 1 is added to row 1, element 2 is added to row 2, etc.\nThe same is true for subtraction\n\n\nM - vector\n\n [,1] [,2] [,3]\n[1,] 8 9 10\n[2,] 10 11 12\n[3,] 12 13 14\n[4,] 14 15 16\n\nAnd multiplication and division\n\n\nM / vector\n\n [,1] [,2] [,3]\n[1,] 5.000000 5.500000 6.0\n[2,] 4.333333 4.666667 5.0\n[3,] 4.000000 4.250000 4.5\n[4,] 3.800000 4.000000 4.2\n\nM * vector\n\n [,1] [,2] [,3]\n[1,] 20 22 24\n[2,] 39 42 45\n[3,] 64 68 72\n[4,] 95 100 105\n\nWhat happens if there are a different number of rows as elements in the vector?\n\n\nvector <- c(2, 3, 4)\n\nM\n\n [,1] [,2] [,3]\n[1,] 10 11 12\n[2,] 13 14 15\n[3,] 16 17 18\n[4,] 19 20 21\n\nM + vector\n\n [,1] [,2] [,3]\n[1,] 12 14 16\n[2,] 16 18 17\n[3,] 20 19 21\n[4,] 21 23 25\n\nNote how the vector just gets reused, no error is thrown.\nWe can also perform these operations on two matrices\n\n\nM1 <- matrix(c(10:21), nrow = 4, byrow = TRUE)\nM2 <- matrix(c(110:121), nrow =4, byrow = TRUE)\n\nM1\n\n [,1] [,2] [,3]\n[1,] 10 11 12\n[2,] 13 14 15\n[3,] 16 17 18\n[4,] 19 20 21\n\nM2\n\n [,1] [,2] [,3]\n[1,] 110 111 112\n[2,] 113 114 115\n[3,] 116 117 118\n[4,] 119 120 121\n\nM1 + M2\n\n [,1] [,2] [,3]\n[1,] 120 122 124\n[2,] 126 128 130\n[3,] 132 134 136\n[4,] 138 140 142\n\nNote how elements in the same position of each matrix are added together\nNote this also is true of vectors\n\n\nv1 <- c(1,2,3)\nv2 <- c(4)\n\nv1 + v2\n\n[1] 5 6 7\n\n\n\nv3 <- c(5, 6)\nv1 + v3\n\n[1] 6 8 8\n\n\n\nv4 <- c(10, 11, 12, 13, 14, 15)\n\nv1 + v4\n\n[1] 11 13 15 14 16 18\n\nExercise\nMultiply, subtract, and divide the two matrices M1 and M2\n\n\n# TODO multiply, subtract and divide M1 and M2\n\n\nMatrix multiplication\nWe will only briefly touch on matrix multiplication, but one reason matrices are very important in R is that you can perform multiplication with them. Exactly how this is done is explained nicely in a math is fun tutorial.\nLet’s try to multiply two matrices together. Remember our first matrix has 4 rows and 3 columns:\n\n\ndim(M)\n\n[1] 4 3\n\nSo our new matrix must have 3 rows\n\n\nM2 <- matrix(c(5:19), nrow = 3, byrow = TRUE)\nM2\n\n [,1] [,2] [,3] [,4] [,5]\n[1,] 5 6 7 8 9\n[2,] 10 11 12 13 14\n[3,] 15 16 17 18 19\n\nLet’s perform matrix multiplication with these\n\n\nM %*% M2\n\n [,1] [,2] [,3] [,4] [,5]\n[1,] 340 373 406 439 472\n[2,] 430 472 514 556 598\n[3,] 520 571 622 673 724\n[4,] 610 670 730 790 850\n\nPerforming functions on a matrix\nSo far, a matrix has looked a lot like a dataframe with some limitations. One of the places where matrices become the most useful is performing statistical functions because all items in a matrix are of the same type.\nFor this next section, let’s use some data that I downloaded from the social security office. This has the top 100 boy and girl names by state for 2020.\nWe can now read in the data and convert it to a matrix\n\n\nnames_mat <- read.csv(here(\"class_8-10_data\", \"boy_name_counts.csv\"),\n row.names = 1) %>%\n as.matrix()\n\nnames_mat[1:5, 1:5]\n\n Alabama Alaska Arizona Arkansas California\nWilliam 366 36 174 152 1021\nJames 304 34 215 105 1148\nJohn 267 20 97 94 623\nElijah 254 42 284 143 1586\nNoah 243 35 397 138 2625\n\nAbove you can see that we have the number of males with each name in each state. Looking at the structure, we can see that it is an integer matrix.\n\n\nstr(names_mat)\n\n int [1:40, 1:20] 366 304 267 254 243 207 187 183 173 166 ...\n - attr(*, \"dimnames\")=List of 2\n ..$ : chr [1:40] \"William\" \"James\" \"John\" \"Elijah\" ...\n ..$ : chr [1:20] \"Alabama\" \"Alaska\" \"Arizona\" \"Arkansas\" ...\n\nWe can now explore this data set using many of the functions you have already learned such as rowSums and colSums\nBasic functions\nFirst, lets find the sum for all of the rows - how many total babies were named each name?\n\n\nrowSums(names_mat)\n\n William James John Elijah Noah Liam Mason \n 5067 5005 3295 5921 8118 8512 4040 \n Oliver Henry Jackson Samuel Jaxon Asher Grayson \n 5961 4121 3599 3549 2571 3212 2938 \n Levi Michael Carter Benjamin Charles Wyatt Thomas \n 3717 4061 3017 5265 2427 3255 2475 \n Aiden Luke David Owen Daniel Logan Joseph \n 3872 3230 3554 3369 4223 3933 3356 \n Lucas Joshua Jack Alexander Maverick Gabriel Ethan \n 4794 2645 3399 4530 2588 3091 4322 \n Eli Isaac Hunter Ezra Theodore \n 2202 3029 1940 2999 3409 \n\nAnd then the columns - how many babies were included from each state?\n\n\ncolSums(names_mat)\n\n Alabama Alaska Arizona Arkansas California \n 5687 986 7371 3428 41256 \n Colorado Connecticut Delaware Florida Georgia \n 6396 4211 1117 21661 11791 \n Hawaii Idaho Illinois Indiana Iowa \n 1111 2306 13407 8283 3508 \n Kansas Kentucky Louisiana Maine Maryland \n 3630 5518 4955 1372 6617 \n\nWhat if we want to find the percent of children with a given name across all states (divide the value by the row sum * 100) - what percent of total babies for each name came from each state:\n\n\npercent_mat <- names_mat / rowSums(names_mat) * 100\npercent_mat[1:5, 1:5]\n\n Alabama Alaska Arizona Arkansas California\nWilliam 7.223209 0.7104796 3.433985 2.999803 20.14999\nJames 6.073926 0.6793207 4.295704 2.097902 22.93706\nJohn 8.103187 0.6069803 2.943854 2.852807 18.90744\nElijah 4.289816 0.7093396 4.796487 2.415133 26.78602\nNoah 2.993348 0.4311407 4.890367 1.699926 32.33555\n\nRemember from above that division using a vector will divide every element of a row by one value, so we can only do this using rowSums. In a few minutes we will discuss how do do this on the columns.\nSummary functions\nWe can also find the minimum, maximum, mean, and median values of the whole matrix and any column. First, lets get summary data for the whole matrix using summary\n\n\nsummary(names_mat)[ , 1:3]\n\n Alabama Alaska Arizona \n Min. : 61.00 Min. :14.00 Min. : 84.0 \n 1st Qu.: 94.75 1st Qu.:18.00 1st Qu.:143.8 \n Median :125.00 Median :22.50 Median :174.5 \n Mean :142.18 Mean :24.65 Mean :184.3 \n 3rd Qu.:163.00 3rd Qu.:28.00 3rd Qu.:191.2 \n Max. :366.00 Max. :44.00 Max. :451.0 \n\nYou can see that this calculates the min, max, mean, median, and quartiles for the columns.\nWhat if we just want the minimum value for the “Alabama” names? We can run min while subsetting to just the column of interest\n\n\nmin(names_mat[, \"Alabama\"])\n\n[1] 61\n\nWe can do the same for the rows Lets try this for “William”\n\n\nmin(names_mat[\"William\",])\n\n[1] 29\n\nWhat if we wanted to find the smallest value in the whole matrix?\n\n\nmin(names_mat)\n\n[1] 13\n\nmax works the same as min\nExercise\nFind the maximum value in the for “Noah”\n\n\n# TODO Find the maximum value for Noah and the whole matrix\n\n\nWe can also find the mean, median, and standard deviation of any part of the matrix\nBy row:\n\n\nmean(names_mat[\"William\", ])\n\n[1] 253.35\n\n\n\nmedian(names_mat[\"William\", ])\n\n[1] 178\n\n\n\nsd(names_mat[\"William\", ])\n\n[1] 239.8189\n\nBy column:\n\n\nmean(names_mat[ , \"Alabama\"])\n\n[1] 142.175\n\n\n\nmedian(names_mat[ , \"Alabama\"])\n\n[1] 125\n\n\n\nsd(names_mat[ , \"Alabama\"])\n\n[1] 67.66012\n\nTransposition\nOne important quality of a matrix is being able to transpose it to interchange the rows and columns - here the rows become columns and columns become rows. We transpose using t() to the matrix. Let’s first look at this using the matrix we started with\n\n\nM <- matrix(c(10:21), nrow = 4, byrow = TRUE)\nM\n\n [,1] [,2] [,3]\n[1,] 10 11 12\n[2,] 13 14 15\n[3,] 16 17 18\n[4,] 19 20 21\n\n\n\nt(M)\n\n [,1] [,2] [,3] [,4]\n[1,] 10 13 16 19\n[2,] 11 14 17 20\n[3,] 12 15 18 21\n\nNote that the output of transposing either a matrix or a data frame will be a matrix (because the type within a column of a data frame must be the same).\n\n\ndf\n\n A B C\n1 10 cat TRUE\n2 11 dog TRUE\n3 12 fish FALSE\n\nt(df)\n\n [,1] [,2] [,3] \nA \"10\" \"11\" \"12\" \nB \"cat\" \"dog\" \"fish\" \nC \"TRUE\" \"TRUE\" \"FALSE\"\n\nstr(df)\n\n'data.frame': 3 obs. of 3 variables:\n $ A: int 10 11 12\n $ B: chr \"cat\" \"dog\" \"fish\"\n $ C: logi TRUE TRUE FALSE\n\nstr(t(df))\n\n chr [1:3, 1:3] \"10\" \"cat\" \"TRUE\" \"11\" \"dog\" \"TRUE\" \"12\" \"fish\" ...\n - attr(*, \"dimnames\")=List of 2\n ..$ : chr [1:3] \"A\" \"B\" \"C\"\n ..$ : NULL\n\nNote how after the transposition, all items in the original df are now characters and we no longer have a dataframe.\nNow let’s try this transposition on the names matrix we’ve been working with\n\n\ntransposed_mat <- t(names_mat)\n\ntransposed_mat[1:3,1:3]\n\n William James John\nAlabama 366 304 267\nAlaska 36 34 20\nArizona 174 215 97\n\nNote how the columns are now names and the rows are now states.\nRemember the note above where we could only divide by the rowSums? Now we can use this transposition to figure out the percent of children in each state with a given name (divide the value by the column sum * 100)\n\n\nstate_percents <- transposed_mat / rowSums(transposed_mat) * 100\n\nstate_percents <- t(state_percents)\n\nstate_percents[1:3, 1:3]\n\n Alabama Alaska Arizona\nWilliam 6.435731 3.651116 2.360602\nJames 5.345525 3.448276 2.916836\nJohn 4.694918 2.028398 1.315968\n\nAbove we did this in several steps, but we can also do in in one step:\n\n\nstate_percents_2 <- t(t(names_mat) / colSums(names_mat)) * 100\n\nidentical(state_percents, state_percents_2)\n\n[1] TRUE\n\nStatistical tests\nWe can also use matrices to perform statistical tests, like t-tests. For instance, are the names Oliver and Noah, or Oliver and Thomas used different amounts?\nFirst, let’s normalize the data to account for the fact that each state reported different numbers of births. To do this normalization, let’s first divide each value by the total number of children reported for that state. Remember, we need to first transpose the matrix to be able to divide by the colSums\n\n\nnormalized_mat <- t(t(names_mat) / colSums(names_mat))\n\n\nNow that we have normalized values, we can do a t-test.\n\n\nnormalized_mat[\"Oliver\", 1:3]\n\n Alabama Alaska Arizona \n0.03217865 0.04361055 0.04124271 \n\nnormalized_mat[\"Noah\", 1:3]\n\n Alabama Alaska Arizona \n0.04272903 0.03549696 0.05385972 \n\nnormalized_mat[\"Thomas\", 1:3]\n\n Alabama Alaska Arizona \n0.02092492 0.02028398 0.01329535 \n\n\n\nt.test(normalized_mat[\"Oliver\",], normalized_mat[\"Noah\",])\n\n\n Welch Two Sample t-test\n\ndata: normalized_mat[\"Oliver\", ] and normalized_mat[\"Noah\", ]\nt = -1.1861, df = 36.481, p-value = 0.2433\nalternative hypothesis: true difference in means is not equal to 0\n95 percent confidence interval:\n -0.010275411 0.002689617\nsample estimates:\n mean of x mean of y \n0.04133411 0.04512700 \n\nBetween Oliver and Noah, there does not seem to be a difference with the data we have. What about Oliver and Thomas?\n\n\nt.test(normalized_mat[\"Oliver\",], normalized_mat[\"Thomas\",])\n\n\n Welch Two Sample t-test\n\ndata: normalized_mat[\"Oliver\", ] and normalized_mat[\"Thomas\", ]\nt = 9.3268, df = 21.544, p-value = 5.121e-09\nalternative hypothesis: true difference in means is not equal to 0\n95 percent confidence interval:\n 0.01858464 0.02922945\nsample estimates:\n mean of x mean of y \n0.04133411 0.01742706 \n\nHere we can see that there is a difference between the mean values for Oliver and Thomas using a t.test\nUsing dataframes and matricies\nFor many of the tidyverse functions you’ve learned so far, a data frame is required. Fortunately, it is very easy to change between a data frame and a matrix.\n\n\nnormalized_dat <- data.frame(normalized_mat)\n\nstr(normalized_dat)\n\n'data.frame': 40 obs. of 20 variables:\n $ Alabama : num 0.0644 0.0535 0.0469 0.0447 0.0427 ...\n $ Alaska : num 0.0365 0.0345 0.0203 0.0426 0.0355 ...\n $ Arizona : num 0.0236 0.0292 0.0132 0.0385 0.0539 ...\n $ Arkansas : num 0.0443 0.0306 0.0274 0.0417 0.0403 ...\n $ California : num 0.0247 0.0278 0.0151 0.0384 0.0636 ...\n $ Colorado : num 0.0369 0.0364 0.0211 0.0317 0.0408 ...\n $ Connecticut: num 0.0356 0.0359 0.0302 0.024 0.0503 ...\n $ Delaware : num 0.0269 0.0322 0.0251 0.0403 0.0466 ...\n $ Florida : num 0.0244 0.0278 0.018 0.0436 0.0608 ...\n $ Georgia : num 0.0469 0.0369 0.0317 0.0446 0.0494 ...\n $ Hawaii : num 0.0261 0.0297 0.0252 0.0342 0.0513 ...\n $ Idaho : num 0.0412 0.0399 0.0173 0.0351 0.0308 ...\n $ Illinois : num 0.0345 0.0334 0.023 0.0309 0.053 ...\n $ Indiana : num 0.0333 0.0321 0.0164 0.0396 0.0391 ...\n $ Iowa : num 0.0425 0.0296 0.0154 0.0299 0.0371 ...\n $ Kansas : num 0.0342 0.0336 0.0226 0.0358 0.0372 ...\n $ Kentucky : num 0.0448 0.0399 0.0268 0.0419 0.0379 ...\n $ Louisiana : num 0.0367 0.0349 0.0367 0.0478 0.044 ...\n $ Maine : num 0.0277 0.0248 0.016 0.0255 0.035 ...\n $ Maryland : num 0.0328 0.0378 0.0213 0.0293 0.0533 ...\n\nOnce we can move between matrices and data frames, we can start to tidy our data for plotting purposes. Let’s plot the distribution of name usage as a violin plot. Here we want the counts to be the y axis and the names to be the y axis.\nThe first thing we need to do is make our matrix into a data frame\n\n\nnames_dat <- data.frame(names_mat)\n\n\nNext, we will want the names to be a column rather than the row names. We can do this using $ or tibble::rownames_to_column\n\n\nnames_dat <- rownames_to_column(names_dat, \"name\")\n\nnames_dat[1:3,1:3]\n\n name Alabama Alaska\n1 William 366 36\n2 James 304 34\n3 John 267 20\n\n# To set using $\n# names_dat$name <- rownames(names_dat)\n\n\nNext, we need to pivot_longer from tidyr. We want to take everything but the names column\n\n\npivot_columns <- colnames(names_dat)[colnames(names_dat) != \"name\"]\n\nnames_dat <- pivot_longer(names_dat, cols = all_of(pivot_columns),\n names_to = \"state\", values_to = \"count\")\n\n\nNote, we can use the pipe %>% from dplyr to put all of this into one statement.\n\n\n# Here we will specify the columns to keep first\npivot_columns <- colnames(names_mat)\n\nnames_dat <- names_mat %>% \n data.frame %>% \n rownames_to_column(\"name\") %>% \n pivot_longer(cols = all_of(pivot_columns), \n names_to = \"state\", values_to = \"count\")\n\n\nWith this new data frame, we can now plot the distribution of names\n\n\n# I first set the theme\ntheme_set(theme_classic(base_size = 10))\n\nggplot(names_dat, aes(x = name, y = count,\n fill = name)) + \n geom_violin() +\n theme(axis.text.x = element_text(angle = 90,\n vjust = 0.5, hjust=1)) # rotate x axis\n\n\n\nThere are a few outliers here, almost certainly California. As we discussed above, normalizing the data helps put everything onto the same scale.\nExercise\nCan you make the same plot as above but use our normalized values?\n\n\n# TODO make plot above with normalized values\n# Hint start with normalized_dat\n\n\nI’m a biologist, why should I care?\nMany of your datasets can be analyzed using matrices\nIf you analyze RNA-seq data, all of the processing and differential testing will be done in a matrix through DESeq2\nIf you analyze protein data, that is best done in a matrix\nSingle cell RNA-seq data is analyzed using matrices, especially sparse matrices\nEven just many measurements can be analyzed using a matrix\n\nMatrices are efficient object types for most types of analysis you would want to do\nAcknowldgements and additional references\nThe content of this class borrows heavily from previous tutorials:\nMatrices\nR example\nQuick tips\n\n\n\n\n", + "preview": "posts/2023-12-08-class-8-matricies/img/matrix_image.jpg", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -30,7 +47,7 @@ "categories": [], "contents": "\nThe Rmarkdown for this document is: https://github.com/rnabioco/bmsc-7810-pbda/blob/main/_posts/2023-12-06-class-6-intro-to-ggplot2-part2/class-6-intro-to-ggplot2-part2.Rmd\nGoals for today\nNew dataset: Iris\nPlotting the categorical data from iris measurements\nBox plots and violin plots\nFrequency and density plots\nUsing stat layers\nAdding additional annotations\nAxis, scales, and coordinate Systems\nNew dataset diamonds\nFaceting plots\nStoring plots as variables\nColor palettes\nApplying themes\nCombining plots with patchwork\nThe Iris Dataset\nFor this class we are going to use a new built in dataset that involves\nthe measurements of Iris flowers. In particular the measurements involve\nthe width and length of two structures of the flower: the petal and the\nsepal. Here is an overview of flower structure.\n\n\n\n\nThe Iris dataset is classically used in machine learning and\nclassification projects. Three species of iris were included in this\nstudy: iris setosa, iris versicolor, and iris virginica. Measurements\nwere taken in 1936 by famous statistician RA Fisher known for the\nStudent’s t-test and F-distribution.\nhttp://archive.ics.uci.edu/ml/datasets/Iris\n\n\n\n\nLet’s look at the this new dataset with head. You can see that it is\nin tidy format with each observation being a new row.\n\n\nhead(iris)\n\n Sepal.Length Sepal.Width Petal.Length Petal.Width Species\n1 5.1 3.5 1.4 0.2 setosa\n2 4.9 3.0 1.4 0.2 setosa\n3 4.7 3.2 1.3 0.2 setosa\n4 4.6 3.1 1.5 0.2 setosa\n5 5.0 3.6 1.4 0.2 setosa\n6 5.4 3.9 1.7 0.4 setosa\n\nTo get a list of the species in this study we can look at all the\nunique() entries in the Species column.\n\n\nunique(iris$Species)\n\n[1] setosa versicolor virginica \nLevels: setosa versicolor virginica\n\nEach one of the species is represented and now we have the exact names\nas written by each measurement. To get the number of measurements for\neach species we can use the summary() function.\n\n\nsummary(iris$Species)\n\n setosa versicolor virginica \n 50 50 50 \n\nWe can begin by looking at the relationships between some of the\nmeasurements by looking at a scatter plot. Here we have Sepal.Length on\nthe x-axis and Sepal.Width on the y-axis.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point()\n\n\n\nExercise: Despite this showing all the data points. How is this not very\ninformative? As a review of last class, add to this plot to make it more\ninformative?\n\n\n\nExercise: Remake this scatterplot but this time for Petal.Width and\nPetal.Length and plot ONLY the iris virginica species data points.\n\n\n\nPlotting the Categorical Data\nSpecies data points with geom_point\nTypically we can look at the distribution of a particular measurement\nvalue based on the category of the measurement, in this case the\nspecies. In this way we can make comparisons between the species. As\nbefore we can use a geom_point_() to plot the values for each species.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_point()\n\n\n\nWhile this does show a basic distribution of Sepal.Width for each\nSpecies, many of the points that have the same value are actually\nhidden! One way we can improve on this is by adding a bit of jitter or\nrandom horizontal position to each point.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter()\n\n\n\nNotice that if you rerun the plot the points are in different locations.\nThe space added by the jitter is randomly generated everytime. Don’t\nexpect them to look the same everytime!\nSide note: You can also use geom_point() geometry function with the\nposition = position_jitter() setting and it will generate the same\nplot as with geom_jitter()\nYou can also tighten the range of the jitter by specifying a width.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter(width=0.1)\n\n\n\nThe Boxplot\nA frequently used plot that is used to better descriptively show this\ntype of data is a boxplot. We can generate a box plot of this data\nsimply by adding a second geom layer called geom_boxplot(). This way\nwe keep the point layer but also have the boxplot.\n\n\n\n\nHere we can add a geom_boxplot layer to our existing jittered\nscatterplot.\n\n\nggplot(iris, (aes(x = Species, y = Sepal.Width))) +\n geom_jitter() +\n geom_boxplot()\n\n\n\nExercise: Many of the points are hidden behind the boxplot. Try changing\nthe order of the layers to see if it matters. What is another way you\ncould fix this?\n\n\n\nViolin Plot\nAnother type of frequently used plot is the violin plot. This plot shows\na continuous density distribution.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_violin() +\n geom_jitter()\n\n\n\nStats Layers\nStats or statistics layers allows us to calculate certain metrics about\nour data and potentially visualize them. First we will look at some of the geom that use stats in their plots.\nFrequency and Density Plots\nFor instance here is a new type of plot that calculates frequency of counts across all measurements of\nSepal.Width. It uses a stat to count the number of measurements at specific values. We could also show the color aes to visualize all the species.\n\n\nggplot(iris) +\n geom_freqpoly(aes(x = Sepal.Width))\n\n\n\ngeom_dotplot() is another way to visualize representative counts. Note that settings stackgroups = TRUE allows you to see all of the dots by stacking them vertically on top of one another without overlap. It uses a stat to count the number of measurements at specific values and represents them as a dot.\n\n\nggplot(iris) +\n geom_dotplot(aes(x = Sepal.Width, fill = Species), stackgroups = TRUE)\n\n\n\nDensity plots can overlap to show a comparison between groups and visualize distribution. It uses a stat to calculate a density metric.\n\n\nggplot(iris) +\n geom_density(aes(x = Sepal.Width, color = Species))\n\n\n\nFinally we have a traditional histogram representing the counts of specific measurement values as above but plotted as a bar plot. It also uses a stat to count the number of measurements at these specific values.\n\n\nggplot(iris) +\n geom_histogram(aes(x = Sepal.Width))\n\n\n\nUnderneath the hood the geom_histogram function is using a stat\nfunction called bin this essentially taking each measurement and\nplacing it in a specific sized category and calculating the frequency of\nthis occurrence. We can modify either the binwidth or the number of\nbins arguments to modify this behavior. For instance if there are 50\nmeasurements from say 1 to 4.5. This range would be divided by the\nnumber of bins. Each measurement value would fall into one of these bins\nand a count would be added for that bin.\n\n\nggplot(iris) +\n geom_histogram(aes(x = Sepal.Width), stat = \"bin\", bins = 10)\n\n\n\nStat Functions\nStats layers are additional information that we calculate and add to the\nplot. Essentially every geom_ function that we have been seen utilizes\ncalculations to produce the plots. Each of these geom_ functions has\nan equivalent stat_ function. It is beyond the scope of this class to\nget into the details of all of these stat functions. Here we will look\nat a particular function called stat_summary that we can use to plot\nsome summary statistics.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(fun = \"mean\",\n geom = \"point\",\n color = \"red\")\n\n\n\nSome of the other options for stat_summary:\ngeoms: point, errorbar, pointrange, linerange, crossbar\nfuns: mean, median, max, min\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(fun = \"mean\",\n geom = \"crossbar\",\n width = 0.5,\n color = \"red\")\n\n\n\nWe can combine multiple stat_summary layers to add additional\ninformation.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(fun = \"mean\",\n geom = \"crossbar\",\n width = 0.5,\n color = \"red\") +\n stat_summary(fun = \"median\",\n geom = \"crossbar\",\n width = 0.5,\n color = \"blue\")\n\n\n\nPlotting the standard error and the confidence intervals\nPlotting the standard error.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(geom = \"errorbar\",\n fun.data = mean_se)\n\n\n\nTo calculate the standard deviation and produce the confidence intervals\nyou can pass mean_cl_normal to the fun.data argument. Note you may\nneed to install the Hmisc package to get this working.\ninstall.packages(\"Hmisc\")\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(geom = \"errorbar\",\n fun.data = mean_cl_normal)\n\n\n\nAnnotations\nAnnotations are easy ways to add extra emphasis to your plots. It can be\nmuch more efficient to have them placed on your plots programatically\nrather than trying to add them later with Photoshop or Illustrator.\nUsing geom_text()\ngeom_text() is an easy way to play text on a plot to annotate. We can even use its aes() function to add column information to the plot like so.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point() +\n geom_text(aes(label=Species))\n\n\n\nNot very practical. Let’s look at the documentation to get some better ideas.\n\n\n?geom_text\n\n\nThere are several options we can add to make things a little neater.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point() +\n geom_text(aes(label=Species), nudge_y = .1, check_overlap = T, size = 3)\n\n\n\nWe can also manually place text anywhere we would like in the plot. This could be a way to annotate whole groups or parts of the visualization.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_text(aes(label=\"setosa\"), x=5, y=4, size = 5) +\n geom_text(aes(label=\"versicolor\"), x=5.5, y=2.25, size = 5) + \n geom_text(aes(label=\"virginica\"), x=7.5, y=3.5, size = 5)\n\n\n\nThe annotate function\nThe annotate function can be used to pass specific types of geometries\nthat you can manually draw on your plot.\n\n\n?annotate\n\n\nHere is an example of drawing a rectangle.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n annotate(\"rect\", xmin=5.5, xmax=6.5, ymin=2.5 , ymax=3.2, alpha=0.2, color=\"blue\")\n\n\n\nUsing a segment geom to produce an arrow. Notice how we need to add the\narrow function.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n annotate(\"segment\", x = 7, xend = 7, y = 4.5, yend = 3.25, color = \"pink\", size=3, alpha=0.6, arrow=arrow())\n\n\n\nDrawing intercept lines with geom_lines\nYou can add horizontal or vertical lines to show cut offs.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_hline(yintercept=4, color = \"orange\", size = 1)\n\n\n\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_vline(xintercept=7, color = \"orange\", size = 1)\n\n\n\nCan add a slope line.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_abline(slope = .5, intercept = 1)\n\n\n\nFiltering data as annotation\nYou can also filter your data during the annotation process and use that\nas a way to clearly highlight features of interest.\nHere by limiting the color to specific measurements.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point() + \n geom_point(data = filter(iris, Sepal.Width > 3.25), aes(color = Species))\n\n\n\nAnd here by limiting the text annotation to specific measurements.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color = Species)) + \n geom_text(data = filter(iris, Sepal.Width > 4), aes(label = Species), vjust = 1)\n\n\n\nExercise: Plot a scatter plot of the Petal.Length and Petal.Width and color by the species of iris. Place a rectangle around the group of points representing the data from the setosa species. Place text above the rectangle that displays “smallest flower”.\n\n\n\nAxis, Scales, and Coordinate Systems\nScales are ways of modifying how the data and the coordinates are shown. When you run this code below there are actually several default hidden scales functions being added.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point()\n\n\n\nNotice how there are three scale function layers added. These are actually being run above but are hidden by default. If you run this version you will get the same plot as above.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n scale_x_continuous() + \n scale_y_continuous() + \n scale_colour_discrete()\n\n\n\nBasically scale_x_ and scale_y_ functions can be used to modify the respective axis appearance and type. For instance we can change the x axis to be on a log scale by using scale_x_log10(). Great way to visualize without having to transform the actual data.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n scale_x_log10()\n\n\n\nYou can also reverse an axis.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n scale_x_reverse()\n\n\n\nYou can manually set the x and y axis range by using the xlim() and ylim() functions.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n xlim(0,10) +\n ylim(0,5)\n\n\n\nThe third default scale in the plot was scale_colour_discrete(). This type of scale modifies how the color can be mapped across the data.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width, color= Sepal.Length)) + \n geom_jitter() + \n scale_color_gradient(low = \"blue\", high = \"red\")\n\n\n\n\n\n#use autocomplete to all the scales options\n#scale_\n\n\nLast class I showed that you could quickly change the axis to swap the\ncoordinates. Here is another way to do that by interacting with the\ncoordinate layer using the coord_flip() function.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_violin() +\n geom_jitter() +\n coord_flip()\n\n\n\nDataset: Diamonds\n\n\n\n\nA dataset containing the prices and other attributes of almost 54,000\ndiamonds.\n\n\nhead(diamonds)\n\n# A tibble: 6 × 10\n carat cut color clarity depth table price x y z\n \n1 0.23 Ideal E SI2 61.5 55 326 3.95 3.98 2.43\n2 0.21 Premium E SI1 59.8 61 326 3.89 3.84 2.31\n3 0.23 Good E VS1 56.9 65 327 4.05 4.07 2.31\n4 0.29 Premium I VS2 62.4 58 334 4.2 4.23 2.63\n5 0.31 Good J SI2 63.3 58 335 4.34 4.35 2.75\n6 0.24 Very Good J VVS2 62.8 57 336 3.94 3.96 2.48\n\nA data frame with 53940 rows and 10 variables:\nprice = price in US dollars ($326–$18,823)\ncarat = weight of the diamond (0.2–5.01)\ncut = quality of the cut (Fair, Good, Very Good, Premium, Ideal)\ncolor = diamond color, from D (best) to J (worst)\nclarity = a measurement of how clear the diamond is (I1 (worst), SI2,\nSI1, VS2, VS1, VVS2, VVS1, IF (best))\nx = length in mm (0–10.74)\ny = width in mm (0–58.9)\nz = depth in mm (0–31.8)\ndepth = total depth percentage = z / mean(x, y) = 2 * z / (x + y)\n(43–79)\ntable = width of top of diamond relative to widest point (43–95)\n\n\nggplot(diamonds, aes(x=carat, y=price)) + \n geom_point()\n\n\n\nExercise: Review the last class. Make a histogram showing the\ndistribution of diamond prices. Color by the cut of the diamond. What\nstatements can you make about the relationships shown.\n\n\n\nExercise: More review. Create a freqpoly plot showing the frequency\ncount of the carat and the color as the cut of diamond. Does this help\nexplain the ideal cut price?\n\n\n\nThere are so many data points in this dataset as seen by our original\nscatterplot. Before moving on we can subset this dataset by using sample\nto grab a random selection of 1000 rows for downstream analysis.\n\n\nset.seed(1337) # set the random seed so that we get the same random rows everytime\n\nsubset_diamonds <- diamonds[sample(nrow(diamonds), 1000), ]\n\nggplot(subset_diamonds, aes(x=carat, y=price)) + \n geom_point()\n\n\n\nIntroducing the Facet\nOne way that we can take an attribute from your data and expand it to\nplot it into multiple plots, one for each level, letting you view them\nseparately. Just as a cut diamond has different flat edges called\nfacets, in ggplot this type of breaking out the levels of the data into\nmultiple plots is called “faceting”. One of the easiest ways to do this\nis by using the facet_wrap() function.\n\n\nggplot(subset_diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point() + \n facet_wrap(~cut, nrow = 1)\n\n\n\nThe second type of facet function is the facet_grid()\n\n\nggplot(subset_diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point() + \n facet_grid(clarity ~ cut)\n\n\n\nThis is a good time to introduce a way to modify the size of the figure\nbeing displayed in RMarkdown. We can edit the curly braces to give\nspecial instructions for the cell. Kent has previous showed this to you\nas well. Here we can add fig.width=20 to increase the width of the\nfigure. You can also try fig.height. There are numerous ways you can\ninfluence the plot using this format and most of them start with the\nfig. prefix.\n\n\nggplot(diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point() + \n facet_grid(clarity ~ cut)\n\n\n\nExercise: Use the dataset from last class iris. Make a scatterplot of\nSepal Width and Sepal Length and color by the Species. Use a\nfacet_wrap to break out the Species.\n\n\n\nStoring Plot Objects\nOne concept that can be useful is that you can assign ggplot plots to a\nvariable just like any other object in R. This can allow you to reuse\nthe plot over and over again simply by calling the variable name you\nsaved the plot. You can also continue to add layers to these plots and\ncan we a quick way to test and compare different versions of a plot.\n\n\np1 <- ggplot(subset_diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point()\n\n\nNotice that nothing was plotting when you run this code. Instead the\nplot is saved to the p1 variable. We can visualize this plot anytime\nsimply by calling the variable.\n\n\np1\n\n\n\nWe can add any additional layers just as we would when building the\nplot. Let’s look at a facet_wrap of the clarity.\n\n\np1 + facet_wrap(~clarity)\n\n\n\nWe changed our mind and now we want to compare this to the same base\nplot but use a facet_grid breaking out the diamond color.\n\n\np1 + facet_grid(clarity~color)\n\n\n\nColor Palettes\nYou can easily change the types and ranges of colors being used in your\nplots. Here is the default color palette:\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point()\n\n\n\nWe can use the scale_color_brewer() to set a different type of\npalette. There are many default options to choose from and maybe more\ncustom ones you can install.\nhttps://r-graph-gallery.com/38-rcolorbrewers-palettes.html\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point() +\n scale_color_brewer(palette = \"RdYlBu\")\n\n\n\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point() +\n scale_color_brewer(palette = \"Accent\")\n\n\n\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point() +\n scale_color_manual(values = c(\"red\", \"blue\", \"green\", \"yellow\", \"purple\", \"white\", \"black\", \"gray\"))\n\n\n\nThemes\nOne of the most fun aspects of ggplot is the ability to quickly change\nthe entire look of your plots with themes.\n\n\nptest <- ggplot(iris, aes(x=Sepal.Width, y=Sepal.Length, color = Species)) +\n geom_point() +\n facet_wrap(~ Species)\n\nptest\n\n\n\n\n\nptest + theme_dark()\n\n\n\n\n\nptest + theme_minimal()\n\n\n\n\n\nptest + theme_bw()\n\n\n\n\n\nptest + theme_classic()\n\n\n\n\n\nptest + theme_void()\n\n\n\nYou can install custom themes….\nhttps://ryo-n7.github.io/2019-05-16-introducing-tvthemes-package/\nhttps://github.com/Mikata-Project/ggthemr\nhttp://xkcd.r-forge.r-project.org/\nCombining multiple plots\nOne useful technique when assembling figures is to be able to stitch\nmultiple plots together into a single image. There is a special add on\npackage that allows us to do just that with simple syntax. This package\nis called patchwork and will need to be installed as it is not\nincluded in the tidyverse. It can be installed with\ninstall.packages(\"patchwork\"). More info at\nhttps://patchwork.data-imaginist.com/\n\n\nlibrary(patchwork)\n\n\nSave the plots as object variables.\n\n\np1 <- ggplot(mtcars) + \n geom_point(aes(mpg, disp))\n\np2 <- ggplot(mtcars) + \n geom_boxplot(aes(gear, disp, group = gear))\n\n\nTo use patchwork simply place the plus operator to “add” two plots\ntogether:\n\n\np1 + p2\n\n\n\nWhy stop at just two plots? We can keep adding more.\n\n\np3 <- ggplot(mtcars) + \n geom_smooth(aes(disp, qsec))\n\np4 <- ggplot(mtcars) + \n geom_bar(aes(carb))\n\n\nAnd use more complex ways of displaying them.\n\n\n(p1 + p2 + p3) / p4\n\n\n\nTo annotate the whole group we need to use a special plot_annotation()\nfunction:\n\n\n(p1 | p2 | p3) / p4 + \n plot_annotation(\n title = 'The surprising truth about mtcars',\n subtitle = 'These 3 plots will reveal yet-untold secrets about our beloved data-set',\n caption = 'Disclaimer: None of these plots are insightful')\n\n\n\nYou can even automatically add the subplot letter annotations. Publish\ntime!\n\n\n(p1 | p2 | p3) / p4 + \n plot_annotation(tag_levels = 'A')\n\n\n\n\n\n(p1 | p2 | p3) / p4 + \n plot_annotation(title = \"Figure 1: Motor Trend 1974 Car Stats\", tag_levels = 'A')\n\n\n\nExercise: Change the order of the plots combined with patchwork so that\np4 is in the middle of the top row and p2 is now on the bottom row. See\nhow the plot adapts.\n\n\n\nThanks for listening. Keep on plotting and exploring the world of\nggplot2!\n—\nSessionInfo\n\n\nsessionInfo()\n\nR version 4.2.2 (2022-10-31)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.6\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRblas.0.dylib\nLAPACK: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRlapack.dylib\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods \n[7] base \n\nother attached packages:\n [1] patchwork_1.1.2 lubridate_1.9.2 forcats_1.0.0 stringr_1.5.0 \n [5] dplyr_1.1.2 purrr_1.0.1 readr_2.1.4 tidyr_1.3.0 \n [9] tibble_3.2.1 ggplot2_3.4.2 tidyverse_2.0.0\n\nloaded via a namespace (and not attached):\n [1] lattice_0.20-45 digest_0.6.31 utf8_1.2.3 \n [4] R6_2.5.1 backports_1.4.1 evaluate_0.21 \n [7] highr_0.10 pillar_1.9.0 rlang_1.1.1 \n[10] rstudioapi_0.14 data.table_1.14.8 jquerylib_0.1.4 \n[13] Matrix_1.5-1 rpart_4.1.19 checkmate_2.3.1 \n[16] rmarkdown_2.22 labeling_0.4.2 splines_4.2.2 \n[19] foreign_0.8-83 htmlwidgets_1.6.2 munsell_0.5.0 \n[22] compiler_4.2.2 xfun_0.39 pkgconfig_2.0.3 \n[25] base64enc_0.1-3 mgcv_1.8-41 htmltools_0.5.5 \n[28] nnet_7.3-18 downlit_0.4.3 tidyselect_1.2.0 \n[31] gridExtra_2.3 htmlTable_2.4.2 Hmisc_5.1-1 \n[34] fansi_1.0.4 viridisLite_0.4.2 tzdb_0.4.0 \n[37] withr_2.5.0 grid_4.2.2 nlme_3.1-160 \n[40] jsonlite_1.8.4 gtable_0.3.3 lifecycle_1.0.3 \n[43] magrittr_2.0.3 scales_1.2.1 cli_3.6.1 \n[46] stringi_1.7.12 cachem_1.0.8 farver_2.1.1 \n[49] bslib_0.4.2 generics_0.1.3 vctrs_0.6.2 \n[52] distill_1.6 Formula_1.2-5 RColorBrewer_1.1-3\n[55] tools_4.2.2 glue_1.6.2 hms_1.1.3 \n[58] fastmap_1.1.1 yaml_2.3.7 timechange_0.2.0 \n[61] colorspace_2.1-0 cluster_2.1.4 memoise_2.0.1 \n[64] knitr_1.43 sass_0.4.6 \n\n\n\n\n", "preview": "posts/2023-12-06-class-6-intro-to-ggplot2-part2/class-6-intro-to-ggplot2-part2_files/figure-html5/unnamed-chunk-6-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -47,7 +64,7 @@ "categories": [], "contents": "\nThe Rmarkdown for this document is\nhttps://github.com/rnabioco/bmsc-7810-pbda/blob/main/_posts/2023-12-05-class-5-intro-to-ggplot2/class-5-intro-to-ggplot2.Rmd\nGoals for today\nIntroduction to plotting with the ggplot2 package\nThe grammar of graphics concept\nBasic plotting\nAdding additional information\nOther geometries\nMultiple geometries\nSaving plots\nAdditional Helpful Resources\nggplot2 package homepage :: https://ggplot2.tidyverse.org/\nggplot2 reference :: https://ggplot2.tidyverse.org/reference R for\nData Science 2e :: https://r4ds.hadley.nz/\nggplot2 Book :: https://ggplot2-book.org/\nGallery of Plots and Examples :: https://r-graph-gallery.com/\nData Visualization with ggplot2 :: Cheat sheet ::\nhttps://github.com/rstudio/cheatsheets/blob/main/data-visualization.pdf\nThe ggplot2 Package\n\n\n\n\nThis package allows you to declaratively create graphics by giving a set\nof variables to map to aesthetics and then layer graphical directives to\nproduce a plot. It’s part of the tidyverse of R packages for data\nscience and analysis, sharing in their design philosophy. It’s an\nalternative to the built in R graphics and plotting functions.Written by Hadley Wickham\nGrammar of Graphics\n\n\n\n\nGrammar gives languages rules.\nGrammar has a technical meaning.\nGrammar makes language expressive.\n-Leland Wilkinson 1945-2021\nLayers of logical command flow and readability.\nLayers of ggplot2\n\n\n\n\nBasic Grammar\nPlot = data + aesthetics + geometry\ndata = the dataset, typically a dataframeaesthetics = map variables x and y to axisgeometry = type of graphic or plot to be rendered\nfacets = multiple plotsstatistics = add calculationstheme = make the plot pretty or follow a particular style\n\n\n# ggplot(, aes()) + ()\n\n?ggplot # bring up the ggplot function help\n\n\nConsider the Type of Data you want to plot\n\n\n\n\nData to Plot\nTo begin plotting we need to start with some data to visualize. Here we\ncan use a built-in dataset regarding Motor Trend Car Road Tests called\nmtcars. This dataset is a dataframe which is a key format for using\nwith ggplot. We can preview the data structure using the head()\nfunction.\n\n\n#some built in data.\n\nhead(mtcars)\n\n mpg cyl disp hp drat wt qsec vs am gear carb\nMazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4\nMazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4\nDatsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1\nHornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1\nHornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2\nValiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1\n\nThe data was extracted from the 1974 Motor Trend US magazine, and\ncomprises fuel consumption and 10 aspects of automobile design and\nperformance for 32 automobiles (1973–74 models).\nA data frame with 32 observations on 11 (numeric) variables.\n[, 1] mpg = Miles/(US) gallon\n[, 2] cyl = Number of cylinders\n[, 3] disp = Displacement (cu.in.)\n[, 4] hp = Gross horsepower\n[, 5] dra = Rear axle ratio\n[, 6] wt = Weight (1000 lbs)\n[, 7] qsec = 1/4 mile time\n[, 8] vs = Engine (0 = V-shaped, 1 = straight)\n[, 9] am = Transmission (0 = automatic, 1 = manual)\n[,10] gear = Number of forward gears\n[,11] carb = Number of carburetors-R Documentation\nBasic Plot\nUsing the basic ggplot grammar of graphics template we can produce a\nscatterplot from the dataframe.\n\n\n# ggplot(, aes()) + ()\n\n\nThe first part of the expression calls the ggplot function and takes\nthe dataframe and the aes function which are the aesthetics\nmappings. In this case we are mapping the x-axis to be the wt variable\nand the y-axis to be the mpg variable . If you only evaluate the first\npart this is what you get:\n\n\nggplot(mtcars, aes(x=wt, y=mpg))\n\n\n\nNext we have to add the geometry layer to be able to actually see the\ndata. Here we are adding the geom_point geometry which allows you to\nvisualize the data as points. You use a plus sign to add these\nadditional layers.\n\n\nggplot(mtcars, aes(x=wt, y=mpg)) + geom_point()\n\n\n\nWe can change the data being plotted by picking a different column from\nthe dataframe. For instance here we are plotting the horsepower(hp)\nversus miles per gallon(mpg). Also note that we can make the code more\nreadable by placing proceeding layers on a different line after the plus\nsign. A common error is misplacing the plus sign. It must be trailing on\nthe line before the next layer.\n\n\nggplot(mtcars, aes(x=hp, y=mpg)) + \n geom_point()\n\n\n\nExercise: Try building a scatterplot on your own. This time plot the\nvariables corresponding to the number of cylinders and the type of\ntransmission.\n\n\n\nExercise: Modify the scatterplot to plot horsepower instead of the type\nof transmission. Can you start to see a relationship with the data?\nAdding Additional Information to the Plot\nTitle\nWe can add a title to the plot simply by adding another layer and the\nggtitle() function.\n\n\nggplot(mtcars, aes(x=hp, y=mpg)) + \n geom_point() +\n ggtitle(\"1974 Cars: Horsepower vs Miles Per Gallon\")\n\n\n\nX and Y axis Labels\nWe can overwrite the default labels and add our own to the x and y axis\nby using the xlab() and ylab() functions respectively.\n\n\nggplot(mtcars, aes(x=hp, y=mpg)) + \n geom_point() +\n ggtitle(\"1974 Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\")\n\n\n\nSet title and axis labels in one layer\n\n\nggplot(mtcars, aes(x=hp, y=mpg, alpha = 0.5)) + \n geom_point() +\n labs(x = \"Horepower\", \n y = \"Miles Per Gallon\", \n title = \"Horsepower vs Miles Per Gallon Scatterplot\",\n subtitle = \"Motor Trend Car Road Tests - 1974\",\n caption = \"Smith et al. 1974\")\n\n\n\nNotice that we also added an alpha aesthetic which helps us visualize\noverlapping points. We can add a show.legend = FALSE argument to the\ngeom_point function to remove the alpha legend and clean up the plot\nfigure. Let’s try it. You can also specify a vector of aesthetics to\ndisplay.\nCheck the documentation ?geom_point.\nGetting Geometry Specific Help\nWe can easily add a third bit of information to the plot by using the\ncolor aesthetic. Each geometry has its own list of aesthetics that you\ncan add and modify. Consult the help page for each one.\n\n\n?geom_point() # bring up the help page for geom_point()\n\n\nAdding the Color Aesthetic\nHere we are adding the color aesthetic.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\")\n\n\n\nAnd we can relabel the legend title for the new color aesthetic to make\nit more readable.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nA Fourth Aesthetic\nYou can even continue to add even more information to the plot through\nadditional aesthetics. Though this might be a bit much.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl, size = wt)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\", size=\"weight (x1000lb)\")\n\n\n\nInstead we can use a specific value instead of the wt variable to\nadjust the size of the dots.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl, size = 3)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nOther Geometries\nThere are many other geometries that you can use in your plots.\nhttps://ggplot2.tidyverse.org/reference\nHere is a short list:\ngeom_point(): scatterplot\ngeom_line(): lines connecting points by increasing value of x\ngeom_path(): lines connecting points in sequence of appearance\ngeom_boxplot(): box and whiskers plot for categorical variables\ngeom_bar(): bar charts for categorical x axis\ngeom_col(): bar chart where heights of the bars represent values in the\ndata\ngeom_histogram(): histogram for continuous x axis\ngeom_violin(): distribution kernel of data dispersion\ngeom_smooth(): function line based on data\ngeom_bin2d(): heatmap of 2d bin counts\ngeom_contour(): 2d contours of a 3d surface\ngeom_count(): count overlapping points\ngeom_density(): smoothed density estimates\ngeom_dotplot(): dot plot\ngeom_hex(): hexagonal heatmap of 2d bin counts\ngeom_freqpoly(): histogram and frequency polygons\ngeom_jitter(): jittered point plot geom_polygon(): polygons\ngeom_line()\nBut utilizing the right plot to efficiently show your data is key. Here\nwe swapped the geom_point for geom_line to see what would happen. You\ncould also try something like geom_bin2d()\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl)) + \n geom_line() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nPlotting the Categories as a Bar Chart with geom_col()\nThe geom_col() geometry is a type of bar plot that uses the heights of\nthe bars to represent values in the data. Let’s look at plotting this\ntype of data for the cars in this dataset.\n\n\n?geom_col()\n\n\n\n\nhead(mtcars)\n\n mpg cyl disp hp drat wt qsec vs am gear carb\nMazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4\nMazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4\nDatsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1\nHornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1\nHornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2\nValiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1\n\nLooking back at the data structure of mtcars, we see that the names of\nthe cars are stored as the row names of the data frame. We can access\nthis using the rownames()function and use it in subsequent plots.\nQ: What was another way to address this issue, discussed in the first\nblock?\n\n\nrownames(mtcars)\n\n [1] \"Mazda RX4\" \"Mazda RX4 Wag\" \"Datsun 710\" \n [4] \"Hornet 4 Drive\" \"Hornet Sportabout\" \"Valiant\" \n [7] \"Duster 360\" \"Merc 240D\" \"Merc 230\" \n[10] \"Merc 280\" \"Merc 280C\" \"Merc 450SE\" \n[13] \"Merc 450SL\" \"Merc 450SLC\" \"Cadillac Fleetwood\" \n[16] \"Lincoln Continental\" \"Chrysler Imperial\" \"Fiat 128\" \n[19] \"Honda Civic\" \"Toyota Corolla\" \"Toyota Corona\" \n[22] \"Dodge Challenger\" \"AMC Javelin\" \"Camaro Z28\" \n[25] \"Pontiac Firebird\" \"Fiat X1-9\" \"Porsche 914-2\" \n[28] \"Lotus Europa\" \"Ford Pantera L\" \"Ferrari Dino\" \n[31] \"Maserati Bora\" \"Volvo 142E\" \n\n\n\nggplot(mtcars, aes(x=rownames(mtcars), y=mpg)) + \n geom_col() +\n ggtitle(\"1974 Cars: Miles Per Gallon\")\n\n\n\nYou will learn other ways to make this more legible later. For a quick\nfix we can swap the x and y mappings.\n\n\nggplot(mtcars, aes(y=rownames(mtcars), x=mpg)) + \n geom_col() +\n ggtitle(\"1974 Cars: Miles Per Gallon\")\n\n\n\nWe can reorder the data to make it easier to visualize important\ninformation.\n\n\nggplot(mtcars, aes(y=reorder(rownames(mtcars), mpg), x=mpg)) + \n geom_col() +\n ggtitle(\"1974 Cars: Ranked by Miles Per Gallon\")\n\n\n\nExercise: Plot a bar chart using geom_col() with the mtcar dataset. Plot\nthe names of the cars ranked by the weight of each car. Try adding a\nthird aesthetic color for horsepower.\n\n\n\nMultiple Geometries\nYou can also add another layer of geometry to the same ggplot. Notice\nyou can have two separate aesthetic declarations and they have moved\nfrom the ggplot function to their respective geom_ functions.\n\n\n# ggplot(data = , mapping = aes()) + \n# () + \n# () \n\n# OR\n\n# ggplot(data = ) + \n# (mapping = aes()) + \n# (mapping = aes()) \n\nggplot(mtcars) +\n geom_point(aes(x=hp, y=mpg)) +\n geom_line(aes(x=hp, y=mpg, color=cyl)) +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nThis particular geometry addition isn’t very useful.\nExercise: Try adding geom_smooth() instead of geom_line().\nSaving Plots\nSaving these plots is easy! Simply call the ggsave() function to save\nthe last plot that you created. You can specify the file format by\nchanging the extension after the filename.\n\n\nggsave(\"plot.png\") # saves the last plot to a PNG file in the current working directory\n\n\nYou can also specify the dots per inch and the width of height of the\nimage to ensure publication quality figures upon saving.\n\n\nggsave(\"plot-highres.png\", dpi = 300, width = 8, height = 4) # you can specify the dots per inch (dpi) and the width and height parameters\n\n\nExercise: Try saving the last plot that we produced as a jpg. Can you\nnavigate to where it saved and open it on your computer?\nCheatsheet\nData Visualization with ggplot2 :: Cheat sheet ::\nhttps://github.com/rstudio/cheatsheets/blob/main/data-visualization.pdf\nMore Examples\nLets take a look at gallery resource to preview different plot types and\nget ideas for our own plots.\nhttps://r-graph-gallery.com/\nNote about LLMs and ChatGPT\nSessionInfo\n\n\nsessionInfo()\n\nR version 4.2.2 (2022-10-31)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.6\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRblas.0.dylib\nLAPACK: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRlapack.dylib\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods \n[7] base \n\nother attached packages:\n [1] lubridate_1.9.2 forcats_1.0.0 stringr_1.5.0 dplyr_1.1.2 \n [5] purrr_1.0.1 readr_2.1.4 tidyr_1.3.0 tibble_3.2.1 \n [9] ggplot2_3.4.2 tidyverse_2.0.0\n\nloaded via a namespace (and not attached):\n [1] highr_0.10 bslib_0.4.2 compiler_4.2.2 \n [4] pillar_1.9.0 jquerylib_0.1.4 tools_4.2.2 \n [7] digest_0.6.31 downlit_0.4.3 timechange_0.2.0 \n[10] jsonlite_1.8.4 evaluate_0.21 memoise_2.0.1 \n[13] lifecycle_1.0.3 gtable_0.3.3 pkgconfig_2.0.3 \n[16] rlang_1.1.1 cli_3.6.1 rstudioapi_0.14 \n[19] distill_1.6 yaml_2.3.7 xfun_0.39 \n[22] fastmap_1.1.1 withr_2.5.0 knitr_1.43 \n[25] systemfonts_1.0.4 hms_1.1.3 generics_0.1.3 \n[28] sass_0.4.6 vctrs_0.6.2 grid_4.2.2 \n[31] tidyselect_1.2.0 glue_1.6.2 R6_2.5.1 \n[34] textshaping_0.3.6 fansi_1.0.4 rmarkdown_2.22 \n[37] farver_2.1.1 tzdb_0.4.0 magrittr_2.0.3 \n[40] scales_1.2.1 htmltools_0.5.5 colorspace_2.1-0 \n[43] ragg_1.2.5 labeling_0.4.2 utf8_1.2.3 \n[46] stringi_1.7.12 munsell_0.5.0 cachem_1.0.8 \n\n\n\n\n", "preview": "posts/2023-12-05-class-5-intro-to-ggplot2/class-5-intro-to-ggplot2_files/figure-html5/unnamed-chunk-8-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -64,7 +81,7 @@ "categories": [], "contents": "\n\n\n\nThe Rmarkdown for this class is on github\nGoals for today\nDiscuss wide and long (tidy) data representations for analysis\nIntroduce the tidyr package for “tidying” rectangular data\nJoining related tables with dplyr\nStrategies for missing data\n\n“Data Scientists spend up to 80% of the time on data cleaning and 20 percent of their time on actual data analysis.”\n– Exploratory Data Mining and Data Cleaning. Dasu and Johnson\n\nWide versus long data formats\nData can be represented in multiple formats. Today we will discuss two common tabular formats for organizing data for analysis.\nConsider the following dataset, which contains population estimates for countries throughout history. This representation of data is commonly referred to as ‘wide’ data format, which is a matrix-like format containing samples as rows and features as columns, with values associated with each observation of a sample and feature.\n\n\nlibrary(readr)\npop_wide <- read_csv(\"data/country_population.csv\")\npop_wide\n\n# A tibble: 197 × 302\n country `1800` `1801` `1802` `1803` `1804` `1805` `1806` `1807` `1808` `1809`\n \n 1 Afghan… 3.28e6 3.28e6 3.28e6 3.28e6 3.28e6 3.28e6 3.28e6 3.28e6 3.28e6 3.28e6\n 2 Angola 1.57e6 1.57e6 1.57e6 1.57e6 1.57e6 1.57e6 1.57e6 1.57e6 1.57e6 1.57e6\n 3 Albania 4 e5 4.02e5 4.04e5 4.05e5 4.07e5 4.09e5 4.11e5 4.13e5 4.14e5 4.16e5\n 4 Andorra 2.65e3 2.65e3 2.65e3 2.65e3 2.65e3 2.65e3 2.65e3 2.65e3 2.65e3 2.65e3\n 5 UAE 4.02e4 4.02e4 4.02e4 4.02e4 4.02e4 4.02e4 4.02e4 4.02e4 4.02e4 4.02e4\n 6 Argent… 5.34e5 5.20e5 5.06e5 4.92e5 4.79e5 4.66e5 4.53e5 4.41e5 4.29e5 4.17e5\n 7 Armenia 4.13e5 4.13e5 4.13e5 4.13e5 4.13e5 4.13e5 4.13e5 4.13e5 4.13e5 4.13e5\n 8 Antigu… 3.7 e4 3.7 e4 3.7 e4 3.7 e4 3.7 e4 3.7 e4 3.7 e4 3.7 e4 3.7 e4 3.7 e4\n 9 Austra… 2 e5 2.05e5 2.11e5 2.16e5 2.22e5 2.27e5 2.33e5 2.39e5 2.46e5 2.52e5\n10 Austria 3 e6 3.02e6 3.04e6 3.05e6 3.07e6 3.09e6 3.11e6 3.12e6 3.14e6 3.16e6\n# ℹ 187 more rows\n# ℹ 291 more variables: `1810` , `1811` , `1812` , `1813` ,\n# `1814` , `1815` , `1816` , `1817` , `1818` ,\n# `1819` , `1820` , `1821` , `1822` , `1823` ,\n# `1824` , `1825` , `1826` , `1827` , `1828` ,\n# `1829` , `1830` , `1831` , `1832` , `1833` ,\n# `1834` , `1835` , `1836` , `1837` , `1838` , …\n\nThe wide matrix-like format is very useful and a common format used for statistics and machine learning. Matrices can take advantage of optimized numerical routines and are the data representation of mathematical matrices. We will work with matrices later in class, particularly with their use to generate heatmaps.\nRepresenting data in a matrix however has a few practical implications:\nThere is only 1 type of data stored in a matrix-like representation (e.g. each cell is the same unit of observation, the population per country). To store additional related data types (e.g. the countries GDP each year) you need to place each new value in an independent matrix.\nThe matrix-like format does not easily lend itself to more complicated summaries. For example, what if we wanted to average the GDP values for each decade or century? We would have to write rather complicated code to parse out subsets of columns for each time period, average them, then merge them into a summary matrix.\nData in a matrix can be instead formatted into a long (also called “tidy”) format.\n#> # A tibble: 10 × 3\n#> country year population\n#> \n#> 1 Afghanistan 1800 3280000\n#> 2 Afghanistan 1801 3280000\n#> 3 Afghanistan 1802 3280000\n#> 4 Afghanistan 1803 3280000\n#> 5 Afghanistan 1804 3280000\n#> 6 Afghanistan 1805 3280000\n#> 7 Afghanistan 1806 3280000\n#> 8 Afghanistan 1807 3280000\n#> 9 Afghanistan 1808 3280000\n#> 10 Afghanistan 1809 3280000\nThe long format of this data convert the many columns of a matrix into a 3 column data.frame containing 3 variables (country, year, and population).\nTidy data format\n\n“Tidy datasets are all alike, but every messy dataset is messy in its own way.” –– Hadley Wickham\n\nA tidy dataset is structured in a manner to be most effectively processed in R using the tidyverse. For example, with the population dataset, instead of having to provide logic to process 100s of columns, instead there are only 3 columns.\nMost data tables that you’ve worked with are probably not tidy. It takes experience to understand the best way to format the data for data processing. As you work more in R and the tidyverse this will become more natural.\nTidy data has the following attributes:\nEach variable must have its own column.\nEach observation must have its own row.\nEach value must have its own cell.\nWhat is a variable, what is an observation, and what is a value?\nA value is a number or word, e.g. the population.\nEvery value belongs to a variable and an observation, e.g. the population value observed in Austria in the year 1910.\nA variable contains all values that measure the same attribute (e.g. height, temperature, duration, magnitude) across units. (e.g. Austria is a value of the country variable, 1910 is a value of the year variable).\nAn observation contains all values measured on the same unit across attributes (e.g observations about Austria in 1910).\n\n\n\nShown below is a simplified data table in a tidy format, provided by the tidyr package. This data table shows the # of TB cases documented by the WHO in a few countries in the years 1999 and 2000.\n\n\nlibrary(tidyr)\ntable1\n\n# A tibble: 6 × 4\n country year cases population\n \n1 Afghanistan 1999 745 19987071\n2 Afghanistan 2000 2666 20595360\n3 Brazil 1999 37737 172006362\n4 Brazil 2000 80488 174504898\n5 China 1999 212258 1272915272\n6 China 2000 213766 1280428583\n\nThe same data, represented in wide, matrix-like format, would require 2 tables:\ne.g a table with the cases values per country.\n\n\ntable4a\n\n# A tibble: 3 × 3\n country `1999` `2000`\n \n1 Afghanistan 745 2666\n2 Brazil 37737 80488\n3 China 212258 213766\n\ne.g a table with the population values per country\n\n\ntable4b\n\n# A tibble: 3 × 3\n country `1999` `2000`\n \n1 Afghanistan 19987071 20595360\n2 Brazil 172006362 174504898\n3 China 1272915272 1280428583\n\nWhat advantages does the tidy format provide?\nEasy to generate summaries of the data.\ne.g. via group_by() -> summarize()\nEasy to plot the data using the ggplot2 framework (more on that in later classes)\nVery easy to join multiple related data frames based on key values.\nSome disadvantages:\nNot space efficient\nNot intuitive\nDoesn’t interface well with traditional machine learning and statistical approaches.\nConverting between long and wide formats using tidyr\nThe tidyr package provides functionality to convert datasets into tidy formats.\npivot_longer(): convert wide data to long data\npivot_wider(): convert long data to wide data\nseparate(): split a single column into multiple columns\nReshaping wide data to long\nThe pivot_longer function requires specifying the columns to pivot using the tidyselect syntax. This syntax is used elsewhere in the tidyverse and is a useful shorthand to avoid listing all columns of interest.\npivot_longer(tbl, cols = <...>)\n\n\n\nFigure 1: Tables from tidyr cheatsheet from https://posit.co/wp-content/uploads/2022/10/tidyr.pdf\n\n\n\n\n\ntable4a\n\n# A tibble: 3 × 3\n country `1999` `2000`\n \n1 Afghanistan 745 2666\n2 Brazil 37737 80488\n3 China 212258 213766\n\n\n\npivot_longer(table4a, cols = `1999`:`2000`) # pivot columns from 1999 -> 2000\n\n# A tibble: 6 × 3\n country name value\n \n1 Afghanistan 1999 745\n2 Afghanistan 2000 2666\n3 Brazil 1999 37737\n4 Brazil 2000 80488\n5 China 1999 212258\n6 China 2000 213766\n\npivot_longer(table4a, cols = -country) # pivot all columns not matching country\n\n# A tibble: 6 × 3\n country name value\n \n1 Afghanistan 1999 745\n2 Afghanistan 2000 2666\n3 Brazil 1999 37737\n4 Brazil 2000 80488\n5 China 1999 212258\n6 China 2000 213766\n\nLet’s try it out on the pop_wide population data\n\n\npop_long <- pivot_longer(pop_wide, cols = -country)\n\npop_long <- pivot_longer(pop_wide, \n cols = -country, \n names_to = \"year\",\n values_to = \"population\")\n\n\nWhy is the useful? Well now we can quickly use dplyr to answer questions, such\nas what is the average population per country across all years?\n\n\nlibrary(dplyr)\ngroup_by(pop_long, country) |> \n summarize(mean_population = mean(population))\n\n# A tibble: 197 × 2\n country mean_population\n \n 1 Afghanistan 28038306.\n 2 Albania 1530495.\n 3 Algeria 23736578.\n 4 Andorra 31687.\n 5 Angola 27240465.\n 6 Antigua and Barbuda 58430.\n 7 Argentina 22730847.\n 8 Armenia 1637548.\n 9 Australia 13964223.\n10 Austria 6573422.\n# ℹ 187 more rows\n\nReshaping long data to wide\npivot_wider(tbl, names_from = <...>, values_from = <...>)\nnames_from: the column whose values will become new columns in the result.values_from: the column whose values will be in the new columns.\n\n\n\n\n\ntable2\n\n# A tibble: 12 × 4\n country year type count\n \n 1 Afghanistan 1999 cases 745\n 2 Afghanistan 1999 population 19987071\n 3 Afghanistan 2000 cases 2666\n 4 Afghanistan 2000 population 20595360\n 5 Brazil 1999 cases 37737\n 6 Brazil 1999 population 172006362\n 7 Brazil 2000 cases 80488\n 8 Brazil 2000 population 174504898\n 9 China 1999 cases 212258\n10 China 1999 population 1272915272\n11 China 2000 cases 213766\n12 China 2000 population 1280428583\n\n\n\npivot_wider(table2, names_from = type, values_from = count)\n\n# A tibble: 6 × 4\n country year cases population\n \n1 Afghanistan 1999 745 19987071\n2 Afghanistan 2000 2666 20595360\n3 Brazil 1999 37737 172006362\n4 Brazil 2000 80488 174504898\n5 China 1999 212258 1272915272\n6 China 2000 213766 1280428583\n\nTry it out with the pop_long population data.\n\n\n\nSeparate\nseparate is useful for dealing with data in which a single column contains multiple variables.\nseperate(tbl, col = <...>, into = c(<..., ..., ...>), sep = \"...\")\ncol: column to split into multiple columnsinto: column names of new columns to be generated, supplied as a character vector (use quotes).sep: the separator used to split values in the col column. Can be a character (_) or a integer to indicate the character position to split (2).\n\n\n\n\n\ntable3\n\n# A tibble: 6 × 3\n country year rate \n \n1 Afghanistan 1999 745/19987071 \n2 Afghanistan 2000 2666/20595360 \n3 Brazil 1999 37737/172006362 \n4 Brazil 2000 80488/174504898 \n5 China 1999 212258/1272915272\n6 China 2000 213766/1280428583\n\n\n\nseparate(table3, col = rate, into = c(\"cases\", \"pop\"), sep = \"/\")\n\n# A tibble: 6 × 4\n country year cases pop \n \n1 Afghanistan 1999 745 19987071 \n2 Afghanistan 2000 2666 20595360 \n3 Brazil 1999 37737 172006362 \n4 Brazil 2000 80488 174504898 \n5 China 1999 212258 1272915272\n6 China 2000 213766 1280428583\n\nExercises\nUse the gapminder population dataset (pop_long) to perform the following tasks and answer the following questions:\nWhich country had the highest population in 1810?\n\n\n\nWhat was the world population in the year 1840?\n\n\n\nWhich country had the lowest average population in the 19th century (years 1800-1899)?\n\n\n\nUsing binds and joins to aggregate multiple data.frames\ncolumn binds\n\n\n\nFigure 2: from the dplyr cheatsheet at https://posit.co/wp-content/uploads/2022/10/data-transformation-1.pdf\n\n\n\nbind_cols(tbl_1, tbl_2, ...)\nbind_cols will bind the columns from 2 or more tables into 1 table. Note that with column binds you need to ensure that each table has the same number of rows, and that the rows correspond to the same observations.\n\n\nlibrary(dplyr)\ntbl1 <- data.frame(x = 1:3)\ntbl2 <- data.frame(y = 3:5)\nbind_cols(tbl1, tbl2)\n\n x y\n1 1 3\n2 2 4\n3 3 5\n\nrow binds\nbind_rows binds rows from multiple tables into one table. Similarly to bind_cols you will want the columns to match between the tables, so that the observations are consistent with the variables.\nbind_rows(tbl_1, tbl_2, ..., .id = NULL)\n\n\n\n\n\ndf_1 <- data.frame(x = 1:5, y = LETTERS[1:5])\ndf_2 <- data.frame(x = 11:15, y = LETTERS[6:10])\n\nbind_rows(df_1, df_2)\n\n x y\n1 1 A\n2 2 B\n3 3 C\n4 4 D\n5 5 E\n6 11 F\n7 12 G\n8 13 H\n9 14 I\n10 15 J\n\nYou can also use a list of data.frames with bind_rows. If the list is named, you can use the .id argument to store a column specifying the name of the data.frame in the output.\n\n\nlst_of_dfs <- list(one = df_1,\n two = df_2)\n\nbind_rows(lst_of_dfs)\n\n x y\n1 1 A\n2 2 B\n3 3 C\n4 4 D\n5 5 E\n6 11 F\n7 12 G\n8 13 H\n9 14 I\n10 15 J\n\nbind_rows(lst_of_dfs, .id = \"source_table\")\n\n source_table x y\n1 one 1 A\n2 one 2 B\n3 one 3 C\n4 one 4 D\n5 one 5 E\n6 two 11 F\n7 two 12 G\n8 two 13 H\n9 two 14 I\n10 two 15 J\n\nJoins\nJoin operations are used to join one table with another table by matching the values shared in particular columns. Join operations enable linking of multiple datasets that contain shared values.\nThere are multiple way to join two tables, depending on how you want to handle different combinations of values present or missing in two tables.\nAssume we have two data.frames called x and y\nThe following joins add columns from y to x, matching rows based on the matching values in shared columns.\ninner_join(x, y): includes all rows in x and y.\nleft_join(x, y): includes all rows in x.\nright_join(x, y): includes all rows in y.\nfull_join(x, y): includes all rows in x or y.\nIf a row in x matches multiple rows in y, all the rows in y will\nbe returned once for each matching row in x.\nConsider our pop_long data.frame. What if we wanted to add additional variables to the data.frame, such as the estimated GDP?\n\n\npop_long[1:5, ]\n\n# A tibble: 5 × 3\n country year population\n \n1 Afghanistan 1800 3280000\n2 Afghanistan 1801 3280000\n3 Afghanistan 1802 3280000\n4 Afghanistan 1803 3280000\n5 Afghanistan 1804 3280000\n\nFirst we’ll read in an additional dataset from Gapminder that contains GDP estimates per country over time. Note that these datafiles have been preprocessed using code here\n\n\n# read in and convert to long format\ngdp_wide <- read_csv(\"data/income_per_person.csv\")\ngdp_long <- pivot_longer(gdp_wide, \n -country, \n names_to = \"year\",\n values_to = \"GDP\")\ngdp_long\n\n# A tibble: 48,945 × 3\n country year GDP\n \n 1 Afghanistan 1799 683\n 2 Afghanistan 1800 683\n 3 Afghanistan 1801 683\n 4 Afghanistan 1802 683\n 5 Afghanistan 1803 683\n 6 Afghanistan 1804 683\n 7 Afghanistan 1805 683\n 8 Afghanistan 1806 683\n 9 Afghanistan 1807 683\n10 Afghanistan 1808 683\n# ℹ 48,935 more rows\n\nNow we can use various joins to merge these data.frames into 1 data.frame.\n\n\n# join on country and year columns, keeping rows with values present in both tables\ninner_join(gdp_long, pop_long)\n\n# A tibble: 48,000 × 4\n country year GDP population\n \n 1 Afghanistan 1800 683 3280000\n 2 Afghanistan 1801 683 3280000\n 3 Afghanistan 1802 683 3280000\n 4 Afghanistan 1803 683 3280000\n 5 Afghanistan 1804 683 3280000\n 6 Afghanistan 1805 683 3280000\n 7 Afghanistan 1806 683 3280000\n 8 Afghanistan 1807 683 3280000\n 9 Afghanistan 1808 683 3280000\n10 Afghanistan 1809 684 3280000\n# ℹ 47,990 more rows\n\nThe Joining, by = join_by(country, year) message indicates that the “country” and “year” columns were used to determine matching rows between the two tables. This is auto-detected based on shared column names in the two data.frames.\nYou can use the by argument to explicitly specify the columns you’d like to join, which is useful if the columns of interest have different names in the two tables.\n\n\n# same as above, but being explicit about the columns to use for joining.\n\n# note that for joins you DO need to use quotes for the columns\ninner_join(gdp_long, pop_long, by = c(\"country\", \"year\"))\n\n# A tibble: 48,000 × 4\n country year GDP population\n \n 1 Afghanistan 1800 683 3280000\n 2 Afghanistan 1801 683 3280000\n 3 Afghanistan 1802 683 3280000\n 4 Afghanistan 1803 683 3280000\n 5 Afghanistan 1804 683 3280000\n 6 Afghanistan 1805 683 3280000\n 7 Afghanistan 1806 683 3280000\n 8 Afghanistan 1807 683 3280000\n 9 Afghanistan 1808 683 3280000\n10 Afghanistan 1809 684 3280000\n# ℹ 47,990 more rows\n\n# unless you use the `join_by` helper\ninner_join(gdp_long, pop_long, by = join_by(country, year))\n\n# A tibble: 48,000 × 4\n country year GDP population\n \n 1 Afghanistan 1800 683 3280000\n 2 Afghanistan 1801 683 3280000\n 3 Afghanistan 1802 683 3280000\n 4 Afghanistan 1803 683 3280000\n 5 Afghanistan 1804 683 3280000\n 6 Afghanistan 1805 683 3280000\n 7 Afghanistan 1806 683 3280000\n 8 Afghanistan 1807 683 3280000\n 9 Afghanistan 1808 683 3280000\n10 Afghanistan 1809 684 3280000\n# ℹ 47,990 more rows\n\n\n\n# join on country and year columns, keeping values all values from gdp_long data.frame\nleft_join(gdp_long, pop_long)\n\n# A tibble: 48,945 × 4\n country year GDP population\n \n 1 Afghanistan 1799 683 NA\n 2 Afghanistan 1800 683 3280000\n 3 Afghanistan 1801 683 3280000\n 4 Afghanistan 1802 683 3280000\n 5 Afghanistan 1803 683 3280000\n 6 Afghanistan 1804 683 3280000\n 7 Afghanistan 1805 683 3280000\n 8 Afghanistan 1806 683 3280000\n 9 Afghanistan 1807 683 3280000\n10 Afghanistan 1808 683 3280000\n# ℹ 48,935 more rows\n\n\n\n# join on country and year columns, keeping values all values from gdp_long and pop_long data.frame\nfull_join(gdp_long, pop_long)\n\n# A tibble: 60,242 × 4\n country year GDP population\n \n 1 Afghanistan 1799 683 NA\n 2 Afghanistan 1800 683 3280000\n 3 Afghanistan 1801 683 3280000\n 4 Afghanistan 1802 683 3280000\n 5 Afghanistan 1803 683 3280000\n 6 Afghanistan 1804 683 3280000\n 7 Afghanistan 1805 683 3280000\n 8 Afghanistan 1806 683 3280000\n 9 Afghanistan 1807 683 3280000\n10 Afghanistan 1808 683 3280000\n# ℹ 60,232 more rows\n\nMissing data\nJoin operations will often generate missing data (e.g. NA values).\nZeroes, NA, NaN and NULL\nDon’t use use zeroes to represent missing data. 0 is valid observed value.\nNA (Not Available) is most often use to represent missing data.\nNaN (Not a Number) is the result of an undefined operation, e.g. 0 / 0.\nNULL means “undefined” and is only used in a programming context (i.e., a function that returns NULL). You can’t put NULL values in a data frame.\nLet’s examine the output from the full_join() operation above which generated NA values.\n\n\ncountry_stats <- full_join(gdp_long, pop_long)\ncountry_stats\n\n# A tibble: 60,242 × 4\n country year GDP population\n \n 1 Afghanistan 1799 683 NA\n 2 Afghanistan 1800 683 3280000\n 3 Afghanistan 1801 683 3280000\n 4 Afghanistan 1802 683 3280000\n 5 Afghanistan 1803 683 3280000\n 6 Afghanistan 1804 683 3280000\n 7 Afghanistan 1805 683 3280000\n 8 Afghanistan 1806 683 3280000\n 9 Afghanistan 1807 683 3280000\n10 Afghanistan 1808 683 3280000\n# ℹ 60,232 more rows\n\nQuick check for NA values\n\n\nsum(is.na(country_stats))\n\n[1] 12342\n\nany(is.na(country_stats))\n\n[1] TRUE\n\nfilter with is.na()\nYou can identify variables with NA values by combining filter() and is.na().\n\n\n# find rows where GDP is NA\nfilter(country_stats, is.na(GDP))\n\n# find rows where GDP is *not* NA\nfilter(country_stats, !is.na(GDP))\n\n\nna.omit()\nYou can remove all rows containing NA values with na.omit().\n\n\nna.omit(country_stats)\n\n\nComputing with NA values\nInstead of removing NA values we can instead just exclude NA values from operations with a common optional argument na.rm = TRUE.\n\n\nx <- c(1, NA, 3)\nsum(x)\nsum(x, na.rm = TRUE)\n\n# if NAs are present, the result is NA\nsum(country_stats$GDP)\n\n# solution: exclude NAs from the calculation\nsum(country_stats$GDP, na.rm = TRUE)\n\n\n\n\ngroup_by(country_stats, country) %>% \n summarize(avg_GDP = mean(GDP, na.rm = TRUE))\n\n\nAlso you can remove NaN values by detecting for their presence using is.nan(). These values often occur when a summary operation (e.g. mean or sum) is performed on a vector with 0 elements.\n\n\nx <- 1:10\n# none are TRUE\nx <- x[x > 100]\nx\n\ninteger(0)\n\nlength(x)\n\n[1] 0\n\nmean(x)\n\n[1] NaN\n\nmean(c(1, NaN), na.rm = TRUE)\n\n[1] 1\n\nReplacing NA values\nLet’s replace the NA values in the population column with a number, such as -1234.\nThis is an operation that is easy to do with base R [] approach.\n\n\n# use is.na to identify NA values to replace with -1234\ncountry_stats$population[is.na(country_stats$population)] <- -1234\n\ncountry_stats[1:10, ]\n\n# A tibble: 10 × 4\n country year GDP population\n \n 1 Afghanistan 1799 683 -1234\n 2 Afghanistan 1800 683 3280000\n 3 Afghanistan 1801 683 3280000\n 4 Afghanistan 1802 683 3280000\n 5 Afghanistan 1803 683 3280000\n 6 Afghanistan 1804 683 3280000\n 7 Afghanistan 1805 683 3280000\n 8 Afghanistan 1806 683 3280000\n 9 Afghanistan 1807 683 3280000\n10 Afghanistan 1808 683 3280000\n\nAlternatively you can use the ifelse() base R function.\n\n\nx <- 1:10\n\nifelse(x < 5, # an expression producing a logical vector \n 5, # if TRUE, replace with this expression\n x) # if FALSE replace with this expression\n\n [1] 5 5 5 5 5 6 7 8 9 10\n\nReplace -1234 with NA using base R $ notation to identify columns.\n\n\ncountry_stats$population <- ifelse(country_stats$population == -1234,\n NA,\n country_stats$population)\ncountry_stats[1:10, ]\n\n# A tibble: 10 × 4\n country year GDP population\n \n 1 Afghanistan 1799 683 NA\n 2 Afghanistan 1800 683 3280000\n 3 Afghanistan 1801 683 3280000\n 4 Afghanistan 1802 683 3280000\n 5 Afghanistan 1803 683 3280000\n 6 Afghanistan 1804 683 3280000\n 7 Afghanistan 1805 683 3280000\n 8 Afghanistan 1806 683 3280000\n 9 Afghanistan 1807 683 3280000\n10 Afghanistan 1808 683 3280000\n\nThe same can also be done with dplyr, in this case replacing NA values again with -1234.\n\n\nmutate(country_stats, \n population = ifelse(is.na(population), \n -1234,\n population)) \n\n# A tibble: 60,242 × 4\n country year GDP population\n \n 1 Afghanistan 1799 683 -1234\n 2 Afghanistan 1800 683 3280000\n 3 Afghanistan 1801 683 3280000\n 4 Afghanistan 1802 683 3280000\n 5 Afghanistan 1803 683 3280000\n 6 Afghanistan 1804 683 3280000\n 7 Afghanistan 1805 683 3280000\n 8 Afghanistan 1806 683 3280000\n 9 Afghanistan 1807 683 3280000\n10 Afghanistan 1808 683 3280000\n# ℹ 60,232 more rows\n\ncase_when()\nIf you want to perform more complex operations use case_when() from dplyr. case_when() is equivalent to performing multiple nested ifelse() operations, whereby if the first operation is not TRUE, then check for the second condition, repeating for each condition until there are no more statements.\nthe syntax for case when is :\n`case_when(conditional statement ~ \"value in result if TRUE\",\n conditional statement #2 ~ \"another value in result if\",\n TRUE ~ \"default if neither conditional statement 1 or 2 are TRUE\")`\nFor a use case, imagine that we wanted to add a new column called era, which signified if the year was in the past, present or future.\n\n\ncountry_stats |>\n mutate(\n era = case_when(year < 2023 ~ \"past\",\n year == 2023 ~ \"present\",\n year > 2023 ~ \"future\")\n )\n\n# A tibble: 60,242 × 5\n country year GDP population era \n \n 1 Afghanistan 1799 683 NA past \n 2 Afghanistan 1800 683 3280000 past \n 3 Afghanistan 1801 683 3280000 past \n 4 Afghanistan 1802 683 3280000 past \n 5 Afghanistan 1803 683 3280000 past \n 6 Afghanistan 1804 683 3280000 past \n 7 Afghanistan 1805 683 3280000 past \n 8 Afghanistan 1806 683 3280000 past \n 9 Afghanistan 1807 683 3280000 past \n10 Afghanistan 1808 683 3280000 past \n# ℹ 60,232 more rows\n\n# same as above, using TRUE on the left side provides a default value.\ncountry_stats |>\n mutate(\n era = case_when(year < 2023 ~ \"past\",\n year == 2023 ~ \"present\",\n TRUE ~ \"future\")\n ) \n\n# A tibble: 60,242 × 5\n country year GDP population era \n \n 1 Afghanistan 1799 683 NA past \n 2 Afghanistan 1800 683 3280000 past \n 3 Afghanistan 1801 683 3280000 past \n 4 Afghanistan 1802 683 3280000 past \n 5 Afghanistan 1803 683 3280000 past \n 6 Afghanistan 1804 683 3280000 past \n 7 Afghanistan 1805 683 3280000 past \n 8 Afghanistan 1806 683 3280000 past \n 9 Afghanistan 1807 683 3280000 past \n10 Afghanistan 1808 683 3280000 past \n# ℹ 60,232 more rows\n\n\nShow session info\n\n\nsessionInfo()\n\nR version 4.3.1 (2023-06-16)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.2.1\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.3-arm64/Resources/lib/libRblas.0.dylib \nLAPACK: /Library/Frameworks/R.framework/Versions/4.3-arm64/Resources/lib/libRlapack.dylib; LAPACK version 3.11.0\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\ntime zone: America/Denver\ntzcode source: internal\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods base \n\nother attached packages:\n[1] dplyr_1.1.3 tidyr_1.3.0 readr_2.1.4\n\nloaded via a namespace (and not attached):\n [1] bit_4.0.5 jsonlite_1.8.7 compiler_4.3.1 highr_0.10 \n [5] crayon_1.5.2 tidyselect_1.2.0 parallel_4.3.1 jquerylib_0.1.4 \n [9] yaml_2.3.7 fastmap_1.1.1 R6_2.5.1 generics_0.1.3 \n[13] knitr_1.45 tibble_3.2.1 distill_1.6 bslib_0.5.1 \n[17] pillar_1.9.0 tzdb_0.4.0 rlang_1.1.2 utf8_1.2.4 \n[21] cachem_1.0.8 xfun_0.41 sass_0.4.7 bit64_4.0.5 \n[25] memoise_2.0.1 cli_3.6.1 withr_2.5.2 magrittr_2.0.3 \n[29] digest_0.6.33 vroom_1.6.4 rstudioapi_0.15.0 hms_1.1.3 \n[33] lifecycle_1.0.4 vctrs_0.6.4 downlit_0.4.3 evaluate_0.23 \n[37] glue_1.6.2 fansi_1.0.5 purrr_1.0.2 rmarkdown_2.25 \n[41] tools_4.3.1 pkgconfig_2.0.3 htmltools_0.5.7 \n\nAcknowledgements and additional references\nThe content of this class borrows heavily from previous tutorials:\nTutorial organization:\nhttps://github.com/sjaganna/molb7910-2019\nR tutorials and documentation:\nhttps://github.com/tidyverse/dplyrhttps://r4ds.had.co.nz/index.html\n\n\n\n", "preview": {}, - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -81,7 +98,7 @@ "categories": [], "contents": "\n\nContents\nUsing R scripts\nUsing Rmarkdown to conduct data analysis\nMore on vectors\nLogical operations\nNegation\nany and all\nFactors\n\nNames\nAdditional data structures in R\nmatrix\nlist\ndata.frame\n\nSubsetting and working with data.frames\nExercises:\n\nFunctions in R\nChaining operations with the pipe operator |>\nErrors, warnings, and messages\nWorkspaces\nOrganizing analyses\nOrganizing your code\nAcknowledgements and additional references\n\nThe Rmarkdown for this class is on github\nUsing R scripts\nR code can be executed using R scripts, which have the .R extension. R scripts can only contain R code, not plain text or markdown. Scripts are executed line by line starting at the top of the document.\nR scripts are useful if you have code that you want to run but don’t need the additional functionality of an Rmarkdown. You can also put custom R functions or R expression into an .R script and then use them in another document. The source() function will execute the R code in a Rscript.\n\n\n# can be a path to a .R file or a URL\nsource(\"https://raw.githubusercontent.com/rnabioco/bmsc-7810-pbda/main/_posts/2023-11-27-class-2/custom-functions.R\")\n\n# defined in script at URL\ngreeting(\"class\")\n\nimportant_list\n\n\nAs an aside, on the command line (e.g. terminal) you can run a R script (or expression):\n\nR -e 'print(\"Hello World\")'\n\n\nRscript your_awesome_code.R \n\nUsing Rmarkdown to conduct data analysis\nRmarkdown is a reproducible framework to create, collaborate, and communicate your work.\nRmarkdown supports a number of output formats including pdfs, word documents, slide shows, html, etc.\nAn Rmarkdown document is a plain text file with the extension .Rmd and contains the following basic components:\nAn (optional) YAML header surrounded by —s.\nChunks of R code surrounded by ```.\nText mixed with simple text formatting like # heading and italics.\n\nRmarkdown documents are executable documents. You can execute the code and render the markdown into html using the render() function, or alternatively by clicking the knit button in Rstudio.\n\n\nlibrary(rmarkdown)\nrender(\"your-rmarkdown.Rmd\")\n\n\nMore on vectors\nWe have spent a large amount of time focused on vectors because these are the fundamental building blocks of more complex data structures.\nLogical operations\nAs we have seen we can use relational operators (e.g. ==, >, <=) to compare values in a vector.\nReturning to our state data, say we wanted to identify states that are located in the south or in the west. How might we approach this?\nThere are a few approaches:\nWe can combine relational operators with logical operators, such as the or operator |, similarly we can use the and operator &.\n\n\n# return TRUE if state is in the South or the West\nstate.region == \"South\" | state.region == \"West\"\n\n [1] TRUE TRUE TRUE TRUE TRUE TRUE FALSE TRUE TRUE TRUE TRUE\n[12] TRUE FALSE FALSE FALSE FALSE TRUE TRUE FALSE TRUE FALSE FALSE\n[23] FALSE TRUE FALSE TRUE FALSE TRUE FALSE FALSE TRUE FALSE TRUE\n[34] FALSE FALSE TRUE TRUE FALSE FALSE TRUE FALSE TRUE TRUE TRUE\n[45] FALSE TRUE TRUE TRUE FALSE TRUE\n\n# states can't be in two regions, so these are all FALSE\nstate.region == \"South\" & state.region == \"West\"\n\n [1] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE\n[12] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE\n[23] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE\n[34] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE\n[45] FALSE FALSE FALSE FALSE FALSE FALSE\n\nWhat if we wanted to ask if the state is in the South, West, or Northeast?\nWe could add another or statement with |\n\n\nstate.region == \"South\" | state.region == \"West\" | state.region == \"Northeast\"\n\n [1] TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE\n[12] TRUE FALSE FALSE FALSE FALSE TRUE TRUE TRUE TRUE TRUE FALSE\n[23] FALSE TRUE FALSE TRUE FALSE TRUE TRUE TRUE TRUE TRUE TRUE\n[34] FALSE FALSE TRUE TRUE TRUE TRUE TRUE FALSE TRUE TRUE TRUE\n[45] TRUE TRUE TRUE TRUE FALSE TRUE\n\nA more efficient approach when testing for the presence of multiple values is to use the %in% operator. This operator tests if an element in a vector on the left is present in the vector on the right.\n\n\nstate.region %in% c(\"South\", \"West\", \"Northeast\")\n\n [1] TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE\n[12] TRUE FALSE FALSE FALSE FALSE TRUE TRUE TRUE TRUE TRUE FALSE\n[23] FALSE TRUE FALSE TRUE FALSE TRUE TRUE TRUE TRUE TRUE TRUE\n[34] FALSE FALSE TRUE TRUE TRUE TRUE TRUE FALSE TRUE TRUE TRUE\n[45] TRUE TRUE TRUE TRUE FALSE TRUE\n\nThis is a very common operation used to select particular subsets of a vector.\nNegation\nWhat we want to find states not in the west or the south?\nAgain there are multiple approaches. We could use the != operator to ask if\na vector does not equal a value. We then combine this with the & operator to find values that do not satisfy either condition.\n\n\n# TRUE if state is not in the south AND the state is not in the WEST\nstate.region != \"South\" & state.region != \"West\"\n\n [1] FALSE FALSE FALSE FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE\n[12] FALSE TRUE TRUE TRUE TRUE FALSE FALSE TRUE FALSE TRUE TRUE\n[23] TRUE FALSE TRUE FALSE TRUE FALSE TRUE TRUE FALSE TRUE FALSE\n[34] TRUE TRUE FALSE FALSE TRUE TRUE FALSE TRUE FALSE FALSE FALSE\n[45] TRUE FALSE FALSE FALSE TRUE FALSE\n\nAlternatively we can use the ! operator, which inverts TRUE to FALSE and vice versa.\ne.g.:\n\n\nx <- c(TRUE, FALSE, TRUE)\n!x\n\n[1] FALSE TRUE FALSE\n\n\n\n!(state.region == \"South\" | state.region == \"West\")\n\n [1] FALSE FALSE FALSE FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE\n[12] FALSE TRUE TRUE TRUE TRUE FALSE FALSE TRUE FALSE TRUE TRUE\n[23] TRUE FALSE TRUE FALSE TRUE FALSE TRUE TRUE FALSE TRUE FALSE\n[34] TRUE TRUE FALSE FALSE TRUE TRUE FALSE TRUE FALSE FALSE FALSE\n[45] TRUE FALSE FALSE FALSE TRUE FALSE\n\nAlso we can use the ! operator with %in%:\n\n\n!(state.region %in% c(\"South\", \"West\"))\n\n [1] FALSE FALSE FALSE FALSE FALSE FALSE TRUE FALSE FALSE FALSE FALSE\n[12] FALSE TRUE TRUE TRUE TRUE FALSE FALSE TRUE FALSE TRUE TRUE\n[23] TRUE FALSE TRUE FALSE TRUE FALSE TRUE TRUE FALSE TRUE FALSE\n[34] TRUE TRUE FALSE FALSE TRUE TRUE FALSE TRUE FALSE FALSE FALSE\n[45] TRUE FALSE FALSE FALSE TRUE FALSE\n\nany and all\nWhat if we want to test if all values are TRUE?\n\n\nis_in_regions <- state.region %in% c(\"South\", \"West\", \"Northeast\", \"North Central\")\nis_in_regions\n\n [1] TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE\n[14] TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE\n[27] TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE\n[40] TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE TRUE\n\nall(is_in_regions)\n\n[1] TRUE\n\nWhat if we want to test if any values are TRUE?\n\n\nany(state.region %in% c(\"Mountain\", \"Alpine\"))\n\n[1] FALSE\n\nany(state.region == \"West\")\n\n[1] TRUE\n\n# useful to quickly check for problematic data points\nany(is.na(state.region))\n\n[1] FALSE\n\nFactors\nWhen printing the state.region object you may have noticed the Levels: Northeast South North Central West. What is this?\nstate.region is a special type of integer vector called a factor. These are commonly used to represent categorical data, and allow one to define a custom order for a category. In various statistical models factors are treated differently from numeric data. In our class you will use them mostly when you are plotting.\nInternally they are represented as integers, with levels that map a value to each integer value.\n\n\ntypeof(state.region)\n\n[1] \"integer\"\n\nclass(state.region)\n\n[1] \"factor\"\n\nlevels(state.region)\n\n[1] \"Northeast\" \"South\" \"North Central\" \"West\" \n\nYou can convert a vector into a factor using factor().\n\n\nanimals <- c(\"cat\", \"fish\", \"fish\", \"bear\", \"bear\")\nanimals <- factor(animals)\nanimals\n\n[1] cat fish fish bear bear\nLevels: bear cat fish\n\nNote that the levels are sorted lexicographically by default\n\n\nlevels(animals)\n\n[1] \"bear\" \"cat\" \"fish\"\n\nWe can add custom ordering by setting the levels\n\n\nanimals <- factor(animals, levels = c(\"cat\", \"bear\", \"fish\"))\nanimals\n\n[1] cat fish fish bear bear\nLevels: cat bear fish\n\n\n\n# sorting will reorder based on the levels\nsort(animals)\n\n[1] cat bear bear fish fish\nLevels: cat bear fish\n\nNames\nVectors in R can also have names, which provide additional information about elements in an object and provide a convenient method to identify elements by name, rather than by position.\nA use case: what if we wanted to determine a state name corresponding to a\nstate abbreviation?\nWe can set the names() of the state.name vector to be the abbreviations.\n\n\nnames(state.name) <- state.abb\nstate.name[1:5]\n\n AL AK AZ AR CA \n \"Alabama\" \"Alaska\" \"Arizona\" \"Arkansas\" \"California\" \n\nNow the names are displayed above each element of the vector.\nWith names, now we query the vector by the abbreviations, which will then return the state names.\n\n\nstate.name[c(\"UT\", \"CO\")]\n\n UT CO \n \"Utah\" \"Colorado\" \n\nNames will become more important next when we start to discuss data.frames and matrices, which can have names corresponding to rows and columns.\nAdditional data structures in R\n\n\n\nFigure 1: Ceballos, Maite and Nicolás Cardiel. 2013. Data structure. First Steps in R. https://web.archive.org/web/20200621022950/http://venus.ifca.unican.es/Rintro/dataStruct.html\n\n\n\nmatrix\nA matrix is a 2 dimensional rectangular data structure, where all values have the same type. It is at is core just a vector, but with a special attribute called dim which specifies the number of rows and columns.\nA matrix is used to store a collection of vectors of the same type and same length.\n\n\nm <- matrix(1:25, nrow = 5, ncol = 5)\ntypeof(m)\n\n[1] \"integer\"\n\nm\n\n [,1] [,2] [,3] [,4] [,5]\n[1,] 1 6 11 16 21\n[2,] 2 7 12 17 22\n[3,] 3 8 13 18 23\n[4,] 4 9 14 19 24\n[5,] 5 10 15 20 25\n\nWe can subset or assign values to specific rows or columns using bracket notation, with values denoting rows and/or columns to keep.\nmatrix[rows to keep, columns to keep].\n\n\n# keep first two rows\nm[1:2, ] \n\n [,1] [,2] [,3] [,4] [,5]\n[1,] 1 6 11 16 21\n[2,] 2 7 12 17 22\n\n# keep first two columns\nm[, 1:2]\n\n [,1] [,2]\n[1,] 1 6\n[2,] 2 7\n[3,] 3 8\n[4,] 4 9\n[5,] 5 10\n\n# keep first two rows and first 3 columns\nm[1:2, 1:3]\n\n [,1] [,2] [,3]\n[1,] 1 6 11\n[2,] 2 7 12\n\n# replace values\nm[1, 1] <- 1000\n\n\nMatrices can have column names and row names that identify the columns. These names can also be used to subset the matrix by row name or column name.\n\n\ncolnames(m) <- LETTERS[1:5]\nrownames(m) <- letters[1:5]\nm\n\n A B C D E\na 1000 6 11 16 21\nb 2 7 12 17 22\nc 3 8 13 18 23\nd 4 9 14 19 24\ne 5 10 15 20 25\n\n\n\nm[c(\"a\", \"b\", \"c\"), c(\"C\", \"D\")]\n\n C D\na 11 16\nb 12 17\nc 13 18\n\nMany functions that operate on vectors also operate on matrices:\n\n\n# total values in m\nsum(m)\nmean(m)\nmax(m)\n\n# add 100 to every value\nm + 100\n# element-wise addition or division\nm + m\nm / m\n\n# replace specific values\nm[m > 10] <- 123455\nm\n\n\nMatrices are a very commonly used data structure, used in many statistics and genomic packages. We will use matrices later in the course as part of a discussion of clustering and heatmaps.\nlist\nA list is similar to a vector, in that it is a container for multiple elements, however it can contain elements from different classes or types. Each element can have a different length or type and can even be a list to generate a nested list of lists.\n\n\nlst <- list(vals = 1:4, \n ids = c(\"bear\", \"dog\"),\n is_valid = TRUE,\n aux = m)\nlst\n\n$vals\n[1] 1 2 3 4\n\n$ids\n[1] \"bear\" \"dog\" \n\n$is_valid\n[1] TRUE\n\n$aux\n A B C D E\na 1000 6 11 16 21\nb 2 7 12 17 22\nc 3 8 13 18 23\nd 4 9 14 19 24\ne 5 10 15 20 25\n\nWe can subset a list using [] and select elements with [[.\nlst[1] # list of length 1\n\nlst[[1]] # first element of list\n\nlst[[1]][1] # first value in first element of list\nIf the list has names we can also use the $ operator or [[ to extract an element by name or subset the list to contain only certain elements based on position.\nA single [ operator when used on a list, returns a list, whereas [[ operators returns the entry in the list. The [[ operator only returns 1 element, whereas [ can return multiple elements.\n\n\n# extract ids element, these are all equivalent\nlst$ids # by name\n\n[1] \"bear\" \"dog\" \n\nlst[[2]] # by position\n\n[1] \"bear\" \"dog\" \n\nlst[[\"ids\"]] # by name, with double bracket notation\n\n[1] \"bear\" \"dog\" \n\n\n\n# subset to first two list elements, returns a list of length 2\n# these are equivalent\nlst[1:2] \n\n$vals\n[1] 1 2 3 4\n\n$ids\n[1] \"bear\" \"dog\" \n\nlst[c(\"vals\", \"ids\")] # using names to subset list\n\n$vals\n[1] 1 2 3 4\n\n$ids\n[1] \"bear\" \"dog\" \n\nlst[c(TRUE, TRUE, FALSE, FALSE)] # using a logical vector\n\n$vals\n[1] 1 2 3 4\n\n$ids\n[1] \"bear\" \"dog\" \n\nSimilar to vectors, we can also add or replace elements in lists. In this case using the $ operator adds an entry to the list with a name (e.g. new_entry). Using the [ approach (with two [[)\n\n\nlst$new_entry <- c(\"hello\", \"world!\")\nlst[[6]] <- c(\"hello\", \"again!\")\n\n\nLists are a very useful data structure that is commonly used as a foundation for storing many different data types in a single object.\nFor example many statistical tests return lists that store various information about the test results.\n\n\nres <- t.test(x = 1:100, y = 100:200)\ntypeof(res)\n\n[1] \"list\"\n\nnames(res)\n\n [1] \"statistic\" \"parameter\" \"p.value\" \"conf.int\" \n [5] \"estimate\" \"null.value\" \"stderr\" \"alternative\"\n [9] \"method\" \"data.name\" \n\nres$p.value\n\n[1] 3.574345e-61\n\ndata.frame\nA data.frame is similar to a matrix, but each column can have a different type. This property makes the data.frame a very useful data structure to store multiple types of related information about an observation.\nA data.frame can be generated using data.frame() or by coercing a matrix or other data structure (as.data.frame()).\n\n\ndf <- data.frame(vals = 1:4, \n animal = c(\"cat\", \"fish\", \"bear\", \"dog\"),\n is_mammal = c(TRUE, FALSE, TRUE, TRUE))\ndf\n\n vals animal is_mammal\n1 1 cat TRUE\n2 2 fish FALSE\n3 3 bear TRUE\n4 4 dog TRUE\n\nIndividual columns (vectors) can be accessed using the $ symbol and treated like regular vectors.\n\n\ndf$animal\n\n[1] \"cat\" \"fish\" \"bear\" \"dog\" \n\nsum(df$is_mammal)\n\n[1] 3\n\nA data.frame is actually a specialized form of a list, whereby each list entry is a vector, and all the vectors have the same length. This is why the syntax is somewhat similar to a list.\n\n\n# convert df to a list, then back to a data.frame\ndf_lst <- as.list(df)\ndf_lst\nas.data.frame(df_lst)\n\n# you can also use the double brackets to extract a column, similar to extracting an element from a list\ndf$is_mammal\ndf[[\"is_mammal\"]] \ndf[[3]]\n\n\nSubsetting and working with data.frames\nJust like with vectors and matrices we can also subset data.frames using logical vectors, positions, and names if they have column and row names.\nFor the next exercises we will use the mtcars dataset built into R. It is data.frame with information about various vehicles from the 1970s. see ?mtcars for a description.\nHere I am using the head() function to print only the first 6 rows (there is also a tail() function).\n\n\nhead(mtcars)\n\n mpg cyl disp hp drat wt qsec vs am gear carb\nMazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4\nMazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4\nDatsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1\nHornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1\nHornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2\nValiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1\n\nWe can subset or select data in the data.frame using the [ notation, similar to matrices.\ndf[rows to keep, columns to keep]\n\n\n# mimic the head() function, keep first 6 rows\nmtcars[1:6, ]\n\n# first row, columns 2 and 3\nmtcars[1, 2:3]\n\n# all data from rows 2 and 4\nmtcars[c(2, 4), ]\n\n# all data from columns 1 and 3\nmtcars[, c(1, 3)]\n\n# extract first 2 columns with logical vector (rep() repeats elements)\nlgl_vec <- c(TRUE, TRUE, rep(FALSE, 9))\nmtcars[, lgl_vec]\n\n\nThis data.frame has row names, which are names that denote individual rows and column names that indicate columns. The rownames are in a column on the far left with no column name. We can subset columns and rows using these names.\n\n\nrownames(mtcars)[1:5]\n\n[1] \"Mazda RX4\" \"Mazda RX4 Wag\" \"Datsun 710\" \n[4] \"Hornet 4 Drive\" \"Hornet Sportabout\"\n\ncolnames(mtcars)[1:5]\n\n[1] \"mpg\" \"cyl\" \"disp\" \"hp\" \"drat\"\n\nmtcars[c(\"Duster 360\", \"Datsun 710\"), c(\"cyl\", \"hp\")]\n\n cyl hp\nDuster 360 8 245\nDatsun 710 4 93\n\nExercises:\nFor cars with miles per gallon (mpg) of at least 30, how many cylinders (cyl) do they have?\n\n\nn_cyl <- mtcars[mtcars$mpg > 30, \"cyl\"]\nn_cyl\n\n[1] 4 4 4 4\n\nunique(n_cyl)\n\n[1] 4\n\nWhich car has the highest horsepower (hp)?\n\n\ntop_hp_car <- mtcars[mtcars$hp == max(mtcars$hp), ]\nrownames(top_hp_car)\n\n[1] \"Maserati Bora\"\n\nThe data.frame and related variants (e.g. tibble or data.table) are a workhorse data structure that we will return to again and again in the next classes.\nFunctions in R\nWe have already used many functions e.g. seq, typeof, matrix, as.data.frame. Functions have rules for how arguments are specified.\nround(x, digits = 0)\nround: function namex: required argumentdigits: optional argument (Defaults to 0)\n\n\nnums <- c(1.5, 1.4, -1.6, 0.0099)\nround(nums)\n\n[1] 2 1 -2 0\n\nround(nums, digits = 1)\n\n[1] 1.5 1.4 -1.6 0.0\n\nThe positional order of the arguments specifies that nums will be assigned to x. Alternatively you can explicitly provide the argument x = nums.\n\n\nround(x = nums, digits = 1)\n\n[1] 1.5 1.4 -1.6 0.0\n\nround(nums, 1)\n\n[1] 1.5 1.4 -1.6 0.0\n\nround(digits = 1, x = nums)\n\n[1] 1.5 1.4 -1.6 0.0\n\nYou can write your own functions as well. Functions reduce copying and pasting code, which reduces errors and simplifies code by reducing objects in the global environment.\nWe’ll learn more about functions later in the course.\n\n\nadd_stuff <- function(x, y, z = 10) {\n x + y + z\n}\nadd_stuff(2, 2)\n\n[1] 14\n\nChaining operations with the pipe operator |>\nAs we’ve seen it is common to combine multiple functions into a single expression, which can be hard to read.\n\n\n# calculate total area of 6 smallest states\nsum(head(sort(state.area)))\n\n[1] 30823\n\nInstead we can use the pipe operator (|>) to pipe data from 1 function to another. The operator takes output from the left hand side and pipes it into the right hand side expression.\n\n\nstate.area |> sort() |> head() |> sum()\n\n[1] 30823\n\n# equivalently\nsort(state.area) |> head() |> sum()\n\n[1] 30823\n\n# equivalently\nsum(head(sort(state.area)))\n\n[1] 30823\n\nImplicitly, the data coming from the pipe is passed as the first argument to the right hand side expression.\nf(x, y) == x |> f(y)\nThe pipe allows complex operations to be conducted without having many intermediate variables or many unreadable nested parathenses.\nIf we need to pass the data to another argument or refer to the data we can use the _ placeholder. When used in a function the _ placeholder must be supplied with the argument name.\n\n\nstate.area |> sort(x = _) |> head(x = _) |> sum(x = _)\n\n# emulate head with selecting the fix 6 obs. \nstate.area |> sort() |> _[1:6] |> sum()\n\n\nWe still need to assign the result to a variable in order to store it.\n\n\ntotal_area <- state.area |> sort() |> head() |> sum()\n\n\n\n\n# this also works, but is discouraged...\nstate.area |> sort() |> head() |> sum() -> total_area\n\n\nLastly, it is common to break up each function call into a separate line for readability\n\n\ntotal_area <- state.area |> \n sort() |> \n head() |> \n sum()\n\n\nThe magrittr package first introduced the pipe operator, but it is different %>%. The two are similar, however the magrittr pipe uses . as a placeholder. You may see the %>% pipe in help and documentation.\nErrors, warnings, and messages\nR expression can fail due to invalid syntax or other problems. If an expression fails, it generally will not return the expected value and an “error” will be issued.\nErrors stop execution, and will cause your scripts to stop. If we include the below chunk in a R script or Rmarkdown it will fail.\n\n\nw <- \"0\" / 1\nw # w does not exist\n\n\nIn contrast, a R command may return a message or warning, both of which will not terminate the execution, but are providing some information about the command being run. Warnings generally should not be ignored as they often are pointing to issues you need to address.\n\n\nww <- c(1, 2, 3) + c(1, 2)\nww\n\n[1] 2 4 4\n\nMessages usually indicate something about the command being run, but are not indicative of an issue. For example, reporting to the user the number of lines processed by a function.\n\n\nmessage(\"we have processed X number of lines\")\n\n\nOften in your analysis code it is useful to throw an error if something strange or unexpected happens. stopifnot() is a useful command to do this.\n\n\nstopifnot(1 + 1 == 2)\nstopifnot(2 + 2 == 5)\n\n\nWorkspaces\nObjects that we assign to variables get stored in an environment known as the Global Environment. You can see the objects in the global environment using the ls() function, or by clicking on the environment tab in Rstudio.\n\n\nls()\n\n [1] \"add_stuff\" \"animals\" \"df\" \"is_in_regions\"\n [5] \"lst\" \"m\" \"n_cyl\" \"nums\" \n [9] \"res\" \"state.name\" \"top_hp_car\" \"total_area\" \n[13] \"ww\" \"x\" \n\nObjects can be removed from the environment, which can be helpful if you have a large memory object that is no longer needed.\n\n\nbig_matrix <- matrix(1:1e6, nrow = 1e5, ncol = 100)\n# show # of rows and columns\ndim(big_matrix)\n#' [1] 100000 100\n\n# remove matrix from environment\nrm(big_matrix)\nbig_matrix\n# 'Error: object 'big_matrix' not found\n\n\n\nWhen you close Rstudio, by default your global R environment is saved to a hidden file called .Rdata in the project directory. When you relaunch rstudio, R objects from your previous environment will be reloaded. This behavior can lead to many problems and we recommend disabling this option \nTo disable this option, go to Rstudio preferences and uncheck the “Restore .RData into workspace at startup” option and select the “Never” option for the “Save workspace to .RData on exit”.\nWe will discuss in later classes how you can save and reload specific R objects and discuss methods to import/export specific data types.\n\nOrganizing analyses\nA little bit of time spent upfront organizing your projects will make analyses easier to manage and reproduce.\nUse Rstudio projects. For the course I recommend making a new project for each class.\nUse multiple directories to separate raw data files from the analysis of the data. Organize the analyses with directories names with chronological dates\nHere’s an example organization strategy.\n.\n├── data\n│   ├── 2022-09-flow\n│   ├── 2022-09-rnaseq-1\n│   └── 2022-09-rnaseq-2\n├── docs\n│   └── project-goals.txt\n├── results\n│   ├── 2022-09-01-rnaseq-expt1\n│   │   └── gene-expression-analysis.Rmd\n│   ├── 2022-09-28-rnaseq-expt2\n│   │   └── splicing-analysis.Rmd\n│   └── 2022-10-01-flow-expt1\n│   └── flow-plots.R\n└── src\n └── rnaseq_pipeline.sh\nSome very good ideas and examples are discussed here:\n\nNoble WS. A quick guide to organizing computational biology projects. PLoS Comput Biol. 2009 Jul;5(7):e1000424. doi: 10.1371/journal.pcbi.1000424.\n\nProvide meaningful names for your files. Consider including ordinal values (e.g. 01, 02, 03) if analyses depend on previous results to indicate ordering of execution.\n# bad\nmodels.R\nanalysis.R\nexplore.R\nanalysis-redo-final-v2.R\n# good\nclean-data.R\nfit-model.R\nplot-data.R\n# better\n01_clean-data.R\n02_fit-model.R\n03_plot-data.R\nOrganizing your code\n\n“Good coding style is like correct punctuation: you can manage without it, butitsuremakesthingseasiertoread.”\n— Hadley Wickham\n\nCode is used to communicate with your computer, but it also is used to communicate with your future self and your colleagues.\nDon’t just write code for yourself right now, instead write your code with the expectation that your future self will need to reread, understand, and modify it in 6 months.\nUse comments to remind yourself what the code does. The # character tells R to ignore a line of text.\n# convert x to zscores\nzs <- (x - mean(x)) / sd(x)\nUse comments to break up long scripts into logical blocks\n# Load data ---------------------------\ndat <- read_csv(\"awesome-data.csv)\ncolnames(dat) <- c(\"sample\", \"color\", \"score\", \"prediction\")\n...\n...\n# modify data -------------------------\ndat <- mutate(dat, result = score + prediction)\n...\n...\n# Plot data ---------------------------\nggplot(dat, aes(sample, score)) + \n geom_point()\nUse sensible names for variables. Keep them short, but meaningful. Separate words with snake_case (e.g plot_df) or camelCase (plotDf) approach.\n# good\na <- width * height\np <- 2 * width + 2 * height\nmeasurement_df <- data.frame(area = a, perimeter = p)\n# bad\ny <- x1 * x2\nyy <- 2*x1 + 2*x2\ntmp <- data.frame(a = y, b = yy)\nSpace is free in code, use it liberally. Add spaces around operators.\n# Good\naverage <- mean(feet / 12 + inches, na.rm = TRUE)\n\n# Bad\naverage<-mean(feet/12+inches,na.rm=TRUE)\nSplit up complicated operations or long function calls into multiple lines. In general you can add a newline after a comma or a pipe operation (%>%). Indenting the code can also help with readability.\n# good\ndata <- complicated_function(x,\n minimizer = 1.4, \n sigma = 100,\n scale_values = FALSE, \n verbose = TRUE, \n additional_args = list(x = 100,\n fun = rnorm))\n# bad\ndata <- complicated_function(x, minimizer = 1.4, sigma = 100, scale_values = FALSE, verbose = TRUE, additional_args = list(x = 100, fun = rnorm))\n#good\nplot_df <- read_csv(\"awesome_data.csv\") %>% \n select(sample, scores, condition) %>%\n mutate(norm_scores = scores / sum(scores))\n \n#bad\nplot_df <- read_csv(\"awesome_data.csv\") %>% select(sample, scores, condition) %>% mutate(norm_scores = scores / sum(scores)) \nRstudio has a shortcuts to help format code\nCode -> Reformat code\nCode -> Reindent lines\nAcknowledgements and additional references\nThe content of this lecture was inspired by and borrows concepts from the following excellent tutorials:\nhttps://github.com/sjaganna/molb7910-2019https://github.com/matloff/fasteRhttps://r4ds.had.co.nz/index.htmlhttps://bookdown.org/rdpeng/rprogdatascience/http://adv-r.had.co.nz/Style.html\n\n\n\n", "preview": {}, - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -98,7 +115,7 @@ "categories": [], "contents": "\n\nContents\nIntroduction to the tidyverse\nloading R packages\ntibble versus data.frame\nConverting a base R data.frame to a tibble\nData import\nData import/export for excel files\nData import/export of R objects\nExploring data\ndplyr, a grammar for data manipulation\nBase R versus dplyr\ndplyr function overview\nFilter rows\narrange rows\n\nColumn operations\nselect columns\n\nWhen to quote or not quote?\nAdding new columns with mutate\nSummarizing columns\nGrouped operations\nString manipulation\nAcknowledgements and additional references\n\nThe Rmarkdown for this class is on github\nIntroduction to the tidyverse\nThe tidyverse is a collection of packages that share similar design philosophy, syntax, and data structures. The packages are largely developed by the same team that builds Rstudio.\nSome key packages that we will touch on in this course:\nreadr: functions for data import and exportggplot2: plotting based on the “grammar of graphics”dplyr: functions to manipulate tabular datatidyr: functions to help reshape data into a tidy formatstringr: functions for working with stringstibble: a redesigned data.frame\nloading R packages\nTo use an R package in an analysis we need to load the package using the library() function. This needs to be done once in each R session and it is a good idea to do this at the beginning of your Rmarkdown. For teaching purposes I will however sometimes load a package when I introduce a function from a package.\n\n\nlibrary(readr)\nlibrary(dplyr)\nlibrary(tibble)\n\n\ntibble versus data.frame\nA tibble is a re-imagining of the base R data.frame. It has a few differences from the data.frame.The biggest differences are that it doesn’t have row.names and it has an enhanced print method. If interested in learning more, see the tibble vignette.\nCompare data_df to data_tbl.\n\n\ndata_df <- data.frame(a = 1:3, \n b = letters[1:3], \n c = c(TRUE, FALSE, TRUE), \n row.names = c(\"ob_1\", \"ob_2\", \"ob_3\"))\ndata_df\n\ndata_tbl <- as_tibble(data_df)\ndata_tbl\n\n\nWhen you work with tidyverse functions it is a good practice to convert data.frames to tibbles. In practice many functions will work interchangeably with either base data.frames or tibble, provided that they don’t use row names.\nConverting a base R data.frame to a tibble\nIf a data.frame has row names, you can preserve these by moving them into a column before converting to a tibble using the rownames_to_column() from tibble.\n\n\nhead(mtcars)\n\n mpg cyl disp hp drat wt qsec vs am gear carb\nMazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4\nMazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4\nDatsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1\nHornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1\nHornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2\nValiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1\n\n\n\nmtcars_tbl <- rownames_to_column(mtcars, \"vehicle\")\nmtcars_tbl <- as_tibble(mtcars_tbl)\nmtcars_tbl\n\n# A tibble: 32 × 12\n vehicle mpg cyl disp hp drat wt qsec vs am gear carb\n \n 1 Mazda RX4 21 6 160 110 3.9 2.62 16.5 0 1 4 4\n 2 Mazda RX4 … 21 6 160 110 3.9 2.88 17.0 0 1 4 4\n 3 Datsun 710 22.8 4 108 93 3.85 2.32 18.6 1 1 4 1\n 4 Hornet 4 D… 21.4 6 258 110 3.08 3.22 19.4 1 0 3 1\n 5 Hornet Spo… 18.7 8 360 175 3.15 3.44 17.0 0 0 3 2\n 6 Valiant 18.1 6 225 105 2.76 3.46 20.2 1 0 3 1\n 7 Duster 360 14.3 8 360 245 3.21 3.57 15.8 0 0 3 4\n 8 Merc 240D 24.4 4 147. 62 3.69 3.19 20 1 0 4 2\n 9 Merc 230 22.8 4 141. 95 3.92 3.15 22.9 1 0 4 2\n10 Merc 280 19.2 6 168. 123 3.92 3.44 18.3 1 0 4 4\n# ℹ 22 more rows\n\nIf you don’t need the rownames, then you can use the as_tibble() function directly.\n\n\nmtcars_tbl <- as_tibble(mtcars)\n\n\nData import\nSo far we have only worked with built in or hand generated datasets, now we will discuss how to read data files into R.\nThe readr package provides a series of functions for importing or writing data in common text formats.\nread_csv(): comma-separated values (CSV) filesread_tsv(): tab-separated values (TSV) filesread_delim(): delimited files (CSV and TSV are important special cases)read_fwf(): fixed-width filesread_table(): whitespace-separated files\nThese functions are quicker and have better defaults than the base R equivalents (e.g. read.table or read.csv). These functions also directly output tibbles rather than base R data.drames\nThe readr checksheet provides a concise overview of the functionality in the package.\nTo illustrate how to use readr we will load a .csv file containing information about airline flights from 2014.\nFirst we will download the data files. You can download this data manually from github. However we will use R to download the dataset using the download.file() base R function.\n\n\n# test if file exists, if it doesn't then download the file.\nif(!file.exists(\"flights14.csv\")) {\n file_url <- \"https://raw.githubusercontent.com/Rdatatable/data.table/master/vignettes/flights14.csv\" \n download.file(file_url, \"flights14.csv\")\n} \n\n\nYou should now have a file called “flights14.csv” in your working directory (the same directory as the Rmarkdown). To read this data into R, we can use the read_csv() function. The defaults for this function often work for many datasets.\n\n\nflights <- read_csv(\"flights14.csv\")\nflights\n\n# A tibble: 253,316 × 11\n year month day dep_delay arr_delay carrier origin dest air_time distance\n \n 1 2014 1 1 14 13 AA JFK LAX 359 2475\n 2 2014 1 1 -3 13 AA JFK LAX 363 2475\n 3 2014 1 1 2 9 AA JFK LAX 351 2475\n 4 2014 1 1 -8 -26 AA LGA PBI 157 1035\n 5 2014 1 1 2 1 AA JFK LAX 350 2475\n 6 2014 1 1 4 0 AA EWR LAX 339 2454\n 7 2014 1 1 -2 -18 AA JFK LAX 338 2475\n 8 2014 1 1 -3 -14 AA JFK LAX 356 2475\n 9 2014 1 1 -1 -17 AA JFK MIA 161 1089\n10 2014 1 1 -2 -14 AA JFK SEA 349 2422\n# ℹ 253,306 more rows\n# ℹ 1 more variable: hour \n\nThere are a few commonly used arguments:\ncol_names: if the data doesn’t have column names, you can provide them (or skip them).\ncol_types: set this if the data type of a column is incorrectly inferred by readr\ncomment: if there are comment lines in the file, such as a header line prefixed with #, you want to skip, set this to #.\nskip: # of lines to skip before reading in the data.\nn_max: maximum number of lines to read, useful for testing reading in large datasets.\nThe readr functions will also automatically uncompress gzipped or zipped datasets, and additionally can read data directly from a URL.\nread_csv(\"https://raw.githubusercontent.com/Rdatatable/data.table/master/vignettes/flights14.csv\")\nThere are equivalent functions for writing data.frames from R to files:\nwrite_csv, write_tsv, write_delim.\nData import/export for excel files\nThe readxl package can read data from excel files and is included in the tidyverse. The read_excel() function is the main function for reading data.\nThe openxlsx package, which is not part of tidyverse but is on CRAN, can write excel files. The write.xlsx() function is the main function for writing data to excel spreadsheets.\nData import/export of R objects\nOften it is useful to store R objects as files on disk so that the R objects can be reloaded into R. These could be large processed datasets, intermediate results, or complex data structures that are not easily stored in rectangular text formats such as csv files.\nR provides the saveRDS() and readRDS() functions for storing and retrieving data in binary formats.\n\n\nsaveRDS(flights, \"flights.rds\") # save single object into a file\ndf <- readRDS(\"flights.rds\") # read object back into R\ndf\n\n# A tibble: 253,316 × 11\n year month day dep_delay arr_delay carrier origin dest air_time distance\n \n 1 2014 1 1 14 13 AA JFK LAX 359 2475\n 2 2014 1 1 -3 13 AA JFK LAX 363 2475\n 3 2014 1 1 2 9 AA JFK LAX 351 2475\n 4 2014 1 1 -8 -26 AA LGA PBI 157 1035\n 5 2014 1 1 2 1 AA JFK LAX 350 2475\n 6 2014 1 1 4 0 AA EWR LAX 339 2454\n 7 2014 1 1 -2 -18 AA JFK LAX 338 2475\n 8 2014 1 1 -3 -14 AA JFK LAX 356 2475\n 9 2014 1 1 -1 -17 AA JFK MIA 161 1089\n10 2014 1 1 -2 -14 AA JFK SEA 349 2422\n# ℹ 253,306 more rows\n# ℹ 1 more variable: hour \n\nIf you want to save/load multiple objects you can use save() and load().\n\n\nsave(flights, df, file = \"robjs.rda\") # save flight_df and df\n\n\nload() will load the data into the environment with the same objects names used when saving the objects.\n\n\nrm(flights, df)\nload(\"robjs.rda\")\n\n\nExploring data\nView() can be used to open an excel like view of a data.frame. This is a good way to quickly look at the data. glimpse() or str() give an additional view of the data.\nView(flights)\nstr(flights)\nglimpse(flights)\nAdditional R functions to help with exploring data.frames (and tibbles):\n\n\ndim(flights) # of rows and columns\nnrow(flights)\nncol(flights)\n\nhead(flights) # first 6 lines\ntail(flights) # last 6 lines\n\ncolnames(flights) # column names\nrownames(flights) # row names (not present in tibble)\n\n\nUseful base R functions for exploring values\n\n\nsummary(flights$distance) # get summary stats on column\n\nunique(flights$carrier) # find unique values in column cyl\n\ntable(flights$carrier) # get frequency of each value in column cyl\ntable(flights$origin, flights$dest) # get frequency of each combination of values\n\n\ndplyr, a grammar for data manipulation\nBase R versus dplyr\nIn the first two lectures we introduced how to subset vectors, data.frames, and matrices\nusing base R functions. These approaches are flexible, succinct, and stable, meaning that\nthese approaches will be supported and work in R in the future.\nSome criticisms of using base R are that the syntax is hard to read, it tends to be verbose, and it is difficult to learn. dplyr, and other tidyverse packages, offer alternative approaches which many find easier to use.\nSome key differences between base R and the approaches in dplyr (and tidyverse)\nUse of the tibble version of data.frame\ndplyr functions operate on data.frame/tibbles rather than individual vectors\ndplyr allows you to specify column names without quotes\ndplyr uses different functions (verbs) to accomplish the various tasks performed by the bracket [ base R syntax\ndplyr and related functions recognized “grouped” operations on data.frames, enabling operations on different groups of rows in a data.frame\ndplyr function overview\ndplyr provides a suite of functions for manipulating data\nin tibbles.\nOperations on Rows:\n- filter() chooses rows based on column values\n- arrange() changes the order of the rows\n- distinct() selects distinct/unique rows\n- slice() chooses rows based on location\nOperations on Columns:\n- select() changes whether or not a column is included\n- rename() changes the name of columns\n- mutate() changes the values of columns and creates new columns\nOperations on groups of rows:\n- summarise() collapses a group into a single row\nFilter rows\nReturning to our flights data. Let’s use filter() to select certain rows.\nfilter(tibble, , ...)\n\n\nfilter(flights, dest == \"LAX\") # select rows where the `dest` column is equal to `LAX\n\n# A tibble: 14,434 × 11\n year month day dep_delay arr_delay carrier origin dest air_time distance\n \n 1 2014 1 1 14 13 AA JFK LAX 359 2475\n 2 2014 1 1 -3 13 AA JFK LAX 363 2475\n 3 2014 1 1 2 9 AA JFK LAX 351 2475\n 4 2014 1 1 2 1 AA JFK LAX 350 2475\n 5 2014 1 1 4 0 AA EWR LAX 339 2454\n 6 2014 1 1 -2 -18 AA JFK LAX 338 2475\n 7 2014 1 1 -3 -14 AA JFK LAX 356 2475\n 8 2014 1 1 142 133 AA JFK LAX 345 2475\n 9 2014 1 1 -4 11 B6 JFK LAX 349 2475\n10 2014 1 1 3 -10 B6 JFK LAX 349 2475\n# ℹ 14,424 more rows\n# ℹ 1 more variable: hour \n\n\n\nfilter(flights, arr_delay > 200) # flights with arr_delay > 200\nfilter(flights, distance < 100) # flights less than 100 miles\nfilter(flights, year != 2014) # if no rows satisfy condition, then an empty tibble\n\n\nMultiple conditions can be used to select rows. For example we can select rows where the dest column is equal to LAX and the origin is equal to EWR. You can either use the & operator, or supply multiple arguments.\n\n\nfilter(flights, dest == \"LAX\", origin == \"EWR\")\nfilter(flights, dest == \"LAX\" & origin == \"EWR\")\n\n\nWe can select rows where the dest column is equal to LAX or the origin is equal to EWR using the | operator.\n\n\nfilter(flights, dest == \"LAX\" | origin == \"EWR\")\n\n\nThe %in% operator is useful for identifying rows with entries matching those in a vector of possibilities.\n\n\nfilter(flights, dest %in% c(\"LAX\", \"SLC\", \"SFO\"))\nfilter(flights, !dest %in% c(\"LAX\", \"SLC\", \"SFO\")) # ! will negate\n\n\nTry it out:\nUse filter to find flights to DEN with a delayed departure (dep_delay).\n\n\n...\n\n\narrange rows\narrange() can be used to sort the data based on values in a single column or multiple columns\narrange(tibble, )\nFor example, let’s find the flight with the shortest amount of air time by arranging the table based on the air_time (flight time in minutes).\n\n\n\n\n\narrange(flights, air_time, distance) # sort first on air_time, then on distance\n\n # to sort in decreasing order, wrap the column name in `desc()`.\narrange(flights, desc(air_time), distance)\n\n\nTry it out:\nUse arrange to determine which flight has the shortest distance?\n\n\n\nColumn operations\nselect columns\nselect() is a simple function that subsets the tibble to keep certain columns.\nselect(tibble, )\n\n\nselect(flights, origin, dest)\n\n# A tibble: 253,316 × 2\n origin dest \n \n 1 JFK LAX \n 2 JFK LAX \n 3 JFK LAX \n 4 LGA PBI \n 5 JFK LAX \n 6 EWR LAX \n 7 JFK LAX \n 8 JFK LAX \n 9 JFK MIA \n10 JFK SEA \n# ℹ 253,306 more rows\n\nthe : operator can select a range of columns, such as the columns from air_time to hour. The ! operator selects columns not listed.\n\n\nselect(flights, air_time:hour)\nselect(flights, !(air_time:hour))\n\n\nThere is a suite of utilities in the tidyverse to help with select columns with names that: matches(), starts_with(), ends_with(), contains(), any_of(), and all_of(). everything() is also useful as a placeholder for all columns not explicitly listed. See help ?select\n\n\n# keep columns that have \"delay\" in the name\nselect(flights, contains(\"delay\"))\n\n# select all columns except carrier\nselect(flights, -carrier)\n\n# reorder columns so that distance and hour are first columns\nselect(flights, starts_with(\"di\"), ends_with(\"ay\"))\n\n\nWhen to quote or not quote?\nIn general, when working with the tidyverse, you don’t need to quote the names of columns. In the example above, we needed quotes because “delay” is not a column name in the flights tibble.\nAdding new columns with mutate\nmutate() allows you to add new columns to the tibble.\nmutate(tibble, new_column_name = expression, ...)\n\n\nmutate(flights, total_delay = dep_delay + arr_delay)\n\n# A tibble: 253,316 × 12\n year month day dep_delay arr_delay carrier origin dest air_time distance\n \n 1 2014 1 1 14 13 AA JFK LAX 359 2475\n 2 2014 1 1 -3 13 AA JFK LAX 363 2475\n 3 2014 1 1 2 9 AA JFK LAX 351 2475\n 4 2014 1 1 -8 -26 AA LGA PBI 157 1035\n 5 2014 1 1 2 1 AA JFK LAX 350 2475\n 6 2014 1 1 4 0 AA EWR LAX 339 2454\n 7 2014 1 1 -2 -18 AA JFK LAX 338 2475\n 8 2014 1 1 -3 -14 AA JFK LAX 356 2475\n 9 2014 1 1 -1 -17 AA JFK MIA 161 1089\n10 2014 1 1 -2 -14 AA JFK SEA 349 2422\n# ℹ 253,306 more rows\n# ℹ 2 more variables: hour , total_delay \n\nWe can’t see the new column, so we add a select command to examine the columns of interest.\n\n\nmutate(flights, total_delay = dep_delay + arr_delay) |> \n select(dep_delay, arr_delay, total_delay)\n\n# A tibble: 253,316 × 3\n dep_delay arr_delay total_delay\n \n 1 14 13 27\n 2 -3 13 10\n 3 2 9 11\n 4 -8 -26 -34\n 5 2 1 3\n 6 4 0 4\n 7 -2 -18 -20\n 8 -3 -14 -17\n 9 -1 -17 -18\n10 -2 -14 -16\n# ℹ 253,306 more rows\n\nMultiple new columns can be made, and you can refer to columns made in preceding statements.\n\n\nmutate(flights, \n delay = dep_delay + arr_delay,\n delay_in_hours = delay / 60) |> \n select(delay, delay_in_hours)\n\n\nTry it out:\nCalculate the flight time (air_time) in hours rather than in minutes, add as a new column.\n\n\nmutate(flights, flight_time = air_time / 60)\n\n# A tibble: 253,316 × 12\n year month day dep_delay arr_delay carrier origin dest air_time distance\n \n 1 2014 1 1 14 13 AA JFK LAX 359 2475\n 2 2014 1 1 -3 13 AA JFK LAX 363 2475\n 3 2014 1 1 2 9 AA JFK LAX 351 2475\n 4 2014 1 1 -8 -26 AA LGA PBI 157 1035\n 5 2014 1 1 2 1 AA JFK LAX 350 2475\n 6 2014 1 1 4 0 AA EWR LAX 339 2454\n 7 2014 1 1 -2 -18 AA JFK LAX 338 2475\n 8 2014 1 1 -3 -14 AA JFK LAX 356 2475\n 9 2014 1 1 -1 -17 AA JFK MIA 161 1089\n10 2014 1 1 -2 -14 AA JFK SEA 349 2422\n# ℹ 253,306 more rows\n# ℹ 2 more variables: hour , flight_time \n\nSummarizing columns\nsummarize() is a function that will collapse the data from a column into a summary value based on a function that takes a vector and returns a single value (e.g. mean(), sum(), median()). It is not very useful yet, but will be very powerful when we discuss grouped operations.\n\n\nsummarize(flights, \n avg_arr_delay = mean(arr_delay),\n med_air_time = median(air_time))\n\n# A tibble: 1 × 2\n avg_arr_delay med_air_time\n \n1 8.15 134\n\nGrouped operations\nAll of the functionality described above can be easily expressed in base R syntax (see examples here). However, where dplyr really shines is the ability to apply the functions above to groups of data within each data frame.\nWe can establish groups within the data using group_by(). The functions mutate(), summarize(), and optionally arrange() will instead operate on each group independently rather than all of the rows.\nCommon approaches:\ngroup_by -> summarize: calculate summaries per group\ngroup_by -> mutate: calculate summaries per group and add as new column to original tibble\ngroup_by(tibble, )\n\n\ngroup_by(flights, carrier) # notice the new \"Groups:\" metadata. \n\n# calculate average dep_delay per carrier\ngroup_by(flights, carrier) |> \n summarize(avg_dep_delay = mean(dep_delay)) \n\n# calculate average arr_delay per carrier at each airport\ngroup_by(flights, carrier, origin) |> \n summarize(avg_dep_delay = mean(dep_delay)) \n\n# calculate # of flights between each origin and destination city, per carrier, and average air time.\n # n() is a special function that returns the # of rows per group\ngroup_by(flights, carrier, origin, dest) |>\n summarize(n_flights = n(),\n mean_air_time = mean(air_time)) \n\n\nHere are some questions that we can answer using grouped operations in a few lines of dplyr code.\nWhat is the average flight air_time between each origin airport and destination airport?\n\n\ngroup_by(flights, origin, dest) |> \n summarize(avg_air_time = mean(air_time))\n\n# A tibble: 221 × 3\n# Groups: origin [3]\n origin dest avg_air_time\n \n 1 EWR ALB 31.4\n 2 EWR ANC 424. \n 3 EWR ATL 111. \n 4 EWR AUS 210. \n 5 EWR AVL 89.7\n 6 EWR AVP 25 \n 7 EWR BDL 25.4\n 8 EWR BNA 115. \n 9 EWR BOS 40.1\n10 EWR BQN 197. \n# ℹ 211 more rows\n\nWhich cites take the longest (air_time) to fly between between on average? the shortest?\n\n\ngroup_by(flights, origin, dest) |> \n summarize(avg_air_time = mean(air_time)) |> \n arrange(desc(avg_air_time)) |> \n head(1)\n\n# A tibble: 1 × 3\n# Groups: origin [1]\n origin dest avg_air_time\n \n1 JFK HNL 625.\n\ngroup_by(flights, origin, dest) |> \n summarize(avg_air_time = mean(air_time)) |> \n arrange(avg_air_time) |> \n head(1)\n\n# A tibble: 1 × 3\n# Groups: origin [1]\n origin dest avg_air_time\n \n1 EWR AVP 25\n\nTry it out:\nWhich carrier has the fastest flight (air_time) on average from JFK to LAX?\n\n\n\nWhich month has the longest departure delays on average when flying from JFK to HNL?\n\n\n\nString manipulation\nstringr is a package for working with strings (i.e. character vectors). It provides a consistent syntax for string manipulation and can perform many routine tasks:\nstr_c: concatenate strings (similar to paste() in base R)str_count: count occurrence of a substring in a stringstr_subset: keep strings with a substringstr_replace: replace a string with another stringstr_split: split a string into multiple pieces based on a string\n\n\nlibrary(stringr)\nsome_words <- c(\"a sentence\", \"with a \", \"needle in a\", \"haystack\")\nstr_detect(some_words, \"needle\") # use with dplyr::filter\nstr_subset(some_words, \"needle\")\n\nstr_replace(some_words, \"needle\", \"pumpkin\")\nstr_replace_all(some_words, \"a\", \"A\")\n\nstr_c(some_words, collapse = \" \")\n\nstr_c(some_words, \" words words words\", \" anisfhlsdihg\")\n\nstr_count(some_words, \"a\")\nstr_split(some_words, \" \")\n\n\nstringr uses regular expressions to pattern match strings. This means that you can perform complex matching to the strings of interest. Additionally this means that there are special characters with behaviors that may be surprising if you are unaware of regular expressions.\nA useful resource when using regular expressions is https://regex101.com\n\n\ncomplex_strings <- c(\"10101-howdy\", \"34-world\", \"howdy-1010\", \"world-.\")\n# keep words with a series of #s followed by a dash, + indicates one or more occurrences.\nstr_subset(complex_strings, \"[0-9]+-\") \n\n# keep words with a dash followed by a series of #s\nstr_subset(complex_strings, \"-[0-9]+\") \n\nstr_subset(complex_strings, \"^howdy\") # keep words starting with howdy\nstr_subset(complex_strings, \"howdy$\") # keep words ending with howdy\nstr_subset(complex_strings, \".\") # . signifies any character\nstr_subset(complex_strings, \"\\\\.\") # need to use backticks to match literal special character\n\n\nLet’s use dplyr and stringr together.\nWhich destinations contain an “LL” in their 3 letter code?\n\n\nlibrary(stringr)\nfilter(flights, str_detect(dest, \"LL\")) |> \n select(dest) |> \n unique()\n\n# A tibble: 1 × 1\n dest \n \n1 FLL \n\nWhich 3-letter destination codes start with H?\n\n\nfilter(flights, str_detect(dest, \"^H\")) |> \n select(dest) |> \n unique()\n\n# A tibble: 4 × 1\n dest \n \n1 HOU \n2 HNL \n3 HDN \n4 HYA \n\nLet’s make a new column that combines the origin and dest columns.\n\n\nmutate(flights, new_col = str_c(origin, \":\", dest)) |> \n select(new_col, everything())\n\n# A tibble: 253,316 × 12\n new_col year month day dep_delay arr_delay carrier origin dest air_time\n \n 1 JFK:LAX 2014 1 1 14 13 AA JFK LAX 359\n 2 JFK:LAX 2014 1 1 -3 13 AA JFK LAX 363\n 3 JFK:LAX 2014 1 1 2 9 AA JFK LAX 351\n 4 LGA:PBI 2014 1 1 -8 -26 AA LGA PBI 157\n 5 JFK:LAX 2014 1 1 2 1 AA JFK LAX 350\n 6 EWR:LAX 2014 1 1 4 0 AA EWR LAX 339\n 7 JFK:LAX 2014 1 1 -2 -18 AA JFK LAX 338\n 8 JFK:LAX 2014 1 1 -3 -14 AA JFK LAX 356\n 9 JFK:MIA 2014 1 1 -1 -17 AA JFK MIA 161\n10 JFK:SEA 2014 1 1 -2 -14 AA JFK SEA 349\n# ℹ 253,306 more rows\n# ℹ 2 more variables: distance , hour \n\n\nShow session info\n\n\nsessionInfo()\n\nR version 4.3.1 (2023-06-16)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.2.1\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.3-arm64/Resources/lib/libRblas.0.dylib \nLAPACK: /Library/Frameworks/R.framework/Versions/4.3-arm64/Resources/lib/libRlapack.dylib; LAPACK version 3.11.0\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\ntime zone: America/Denver\ntzcode source: internal\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods base \n\nother attached packages:\n[1] stringr_1.5.1 tibble_3.2.1 dplyr_1.1.3 readr_2.1.4 \n\nloaded via a namespace (and not attached):\n [1] bit_4.0.5 jsonlite_1.8.7 compiler_4.3.1 crayon_1.5.2 \n [5] tidyselect_1.2.0 parallel_4.3.1 jquerylib_0.1.4 yaml_2.3.7 \n [9] fastmap_1.1.1 R6_2.5.1 generics_0.1.3 knitr_1.45 \n[13] distill_1.6 bslib_0.5.1 pillar_1.9.0 tzdb_0.4.0 \n[17] rlang_1.1.2 utf8_1.2.4 cachem_1.0.8 stringi_1.8.1 \n[21] xfun_0.41 sass_0.4.7 bit64_4.0.5 memoise_2.0.1 \n[25] cli_3.6.1 withr_2.5.2 magrittr_2.0.3 digest_0.6.33 \n[29] vroom_1.6.4 rstudioapi_0.15.0 hms_1.1.3 lifecycle_1.0.4 \n[33] vctrs_0.6.4 downlit_0.4.3 evaluate_0.23 glue_1.6.2 \n[37] fansi_1.0.5 rmarkdown_2.25 tools_4.3.1 pkgconfig_2.0.3 \n[41] htmltools_0.5.7 \n\nAcknowledgements and additional references\nThe content of this class borrows heavily from previous tutorials:\nR code style guide:\nhttp://adv-r.had.co.nz/Style.html\nTutorial organization:\nhttps://github.com/sjaganna/molb7910-2019\nOther R tutorials:\nhttps://github.com/matloff/fasteR\nhttps://r4ds.had.co.nz/index.html\nhttps://bookdown.org/rdpeng/rprogdatascience/\n\n\n\n", "preview": {}, - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -115,7 +132,7 @@ "categories": [], "contents": "\n\nContents\nOutline\nWhat is R\nWhy is R a popular language?\nThe R ecosystem\nGetting help\nBuilt-in documentation\nVignettes\nRstudio Cheatsheets\n\nThe more you try the more you will learn.\nUsing R interactively with the Console\nExample datasets in R\nAssigning values to variables\nVectors and atomic types in R\nVector types\nNA, Inf, and NaN values\nmaking vectors from scratch\nSubsetting vectors in R\nExercise:\nUsing vectors and subsetting to perform more complex operations\nExercise:\nReplacing or adding values at position\n\nR operations are vectorized\nReview\nAcknowledgements and additional references\n\n\nThe Rmarkdown for this class is on github\nOutline\nR language history and ecosystem\nFinding help and reading R documentation\nR fundamentals\nUsing the R console\nVariables\nVectors\nVector types\nOperators\nVectorization\n\nWhat is R\nFrom the R core developers:\n\nR is an integrated suite of software facilities for data manipulation, calculation and graphical display. It includes\nan effective data handling and storage facility,\na suite of operators for calculations on arrays, in particular matrices,\na large, coherent, integrated collection of intermediate tools for data analysis,\ngraphical facilities for data analysis and display either on-screen or on hardcopy, and\na well-developed, simple and effective programming language which includes conditionals, loops, user-defined recursive functions and input and output facilities.\n\n\nR, like S, is designed around a true computer language, and it allows users to add additional functionality by defining new functions. Much of the system is itself written in the R dialect of S, which makes it easy for users to follow the algorithmic choices made. For computationally-intensive tasks, C, C++ and Fortran code can be linked and called at run time. Advanced users can write C code to manipulate R objects directly.\n\n\nMany users think of R as a statistics system. We prefer to think of it as an environment within which statistical techniques are implemented. R can be extended (easily) via packages. There are about eight packages supplied with the R distribution and many more are available through the CRAN family of Internet sites covering a very wide range of modern statistics.\n\nWhy is R a popular language?\n\n\n\nFigure 1: R facilitates the data analysis process. From https://r4ds.had.co.nz/explore-intro.html.\n\n\n\nR is a programming language built by statisticians to facilitate interactive exploratory data analysis.\nR comes with (almost) everything you need built in to rapidly conduct data analysis and visualization.\nR has a large following, which makes it easy to find help and examples of analyses.\n- Rstudio/Posit Community\n- Bioconductor Support\n- R stackoverflow\nR works out of the box on major operating systems.\nR has a robust package system of packages from CRAN and bioinformatics focused packages from Bioconductor\nPublication quality plots can be produced with ease using functionality in the base R installation or provided by additional packages.\nR has a built in documentation system to make it easy to find help and examples of how to use R functionality.\nIt’s free, open-source, and has been around in it’s first public release since 1993.\nThe R ecosystem\nWhen you download R from CRAN, there are a number of packages included in the base installation (e.g. base, stats, and datasets). You can do effective data analysis with only the base installation (e.g. see fasteR tutorial). However a key strength of R is the 10,000+ user-developed packages which extend base R functionality.\n\n\n\nFigure 2: Major R package repositories and functions used to install packages.\n\n\n\nCRAN is the official R package repository and source for R. The tidyverse (which we will use in subsequent classes) is a set of packages with consistent design principles meant to extend functionality in base R.\nBioconductor hosts and maintains bioinformatics focused packages, built around a set of core data structures and functionality focused on genomics and bioinformatics.\nGithub hosts software for any software project. It is often used to host R packages in development stages and the actively developed source code for R packages.\nGetting help\nBuilt-in documentation\nThe ? operator can be used to pull up documentation about a function. The ?? operator uses a fuzzy search which can pull up help if you don’t remember the exact function name.\n?install.packages\n??install.package\nAlternatively you can click on the help pane and search for help in Rstudio.\nVignettes\nEvery R package includes a vignette to describe the functionality of the package which can be a great resource to learn about new packages.\nThese can be accessed via the vignette() function, or via the help menu in Rstudio.\nvignette(\"dplyr\")\nRstudio Cheatsheets\nSee Help > Cheatsheets for very helpful graphical references.The base R, dplyr, and ggplot2 cheatsheets are especially useful.\nThe more you try the more you will learn.\nLearning a foreign language requires continual practice speaking and writing the language. To learn you need to try new phrases and expressions. To learn you have to make mistakes. The more you try and experiment the quicker you will learn.\nLearning a programming language is very similar. We communicate by organizing a series of steps in the right order to instruct the computer to accomplish a task.\nType and execute commands, rather than copy and pasting, you will learn faster. Fiddle around with the code, see what works and what doesn’t.\nProbably everything we do in the class can be done by a LLM such as ChatGPT. These tools can help you, but you will be more effective at using them if you understand the fundamentals. You will also be more productive in the long term if you understand the basics.\nUsing R interactively with the Console\nR commands can be executed in the “Console”, which is an interactive shell that waits for you to run commands.\nThe > character is a prompt that indicates the beginning of a line. The prompt goes away when a command is being executed, and returns upon completion, or an error.\nYou can interrupt a command with Esc, Ctrl + c, clicking the STOP sign in the upper right corner, or Session -> Interrupt R or Terminate R.\nBefore running another command ensure that the > prompt is present.\nR can be used as a simple calculator:\n\n\n1 + 1\n\n[1] 2\n\n3-7 # value of 7 subtracted from 3\n\n[1] -4\n\n3/2 # Division\n\n[1] 1.5\n\n5^2 # 5 raised to the second power\n\n[1] 25\n\n2 + 3 * 5 # R respects the order of math operations.\n\n[1] 17\n\nExample datasets in R\nR and R packages include small datasets to demonstrate how to use a package or functionality. data() will show you many of the datasets included with a base R installation. We will use the state datasets, which contain data on the 50 US states.\n\n\nstate.abb\n\n [1] \"AL\" \"AK\" \"AZ\" \"AR\" \"CA\" \"CO\" \"CT\" \"DE\" \"FL\" \"GA\" \"HI\" \"ID\" \"IL\"\n[14] \"IN\" \"IA\" \"KS\" \"KY\" \"LA\" \"ME\" \"MD\" \"MA\" \"MI\" \"MN\" \"MS\" \"MO\" \"MT\"\n[27] \"NE\" \"NV\" \"NH\" \"NJ\" \"NM\" \"NY\" \"NC\" \"ND\" \"OH\" \"OK\" \"OR\" \"PA\" \"RI\"\n[40] \"SC\" \"SD\" \"TN\" \"TX\" \"UT\" \"VT\" \"VA\" \"WA\" \"WV\" \"WI\" \"WY\"\n\nstate.area \n\n [1] 51609 589757 113909 53104 158693 104247 5009 2057 58560\n[10] 58876 6450 83557 56400 36291 56290 82264 40395 48523\n[19] 33215 10577 8257 58216 84068 47716 69686 147138 77227\n[28] 110540 9304 7836 121666 49576 52586 70665 41222 69919\n[37] 96981 45333 1214 31055 77047 42244 267339 84916 9609\n[46] 40815 68192 24181 56154 97914\n\nThese are R objects, specifically vectors. A vector is collection of values of all the same data type. Note that each position in the vector has a number, called an index. We will talk more about the vectors shortly.\nLet’s start using some simple R functions to characterize the size of the US states. Type the following in the console and hit return to execute or call the mean function.\n\n\nmean(state.area)\n\n[1] 72367.98\n\nmean is a function. A function takes an input (as an argument) and returns a value.\n\n\n# simple summary functions\nsum(state.area)\n\n[1] 3618399\n\nmin(state.area)\n\n[1] 1214\n\nmax(state.area)\n\n[1] 589757\n\nmedian(state.area)\n\n[1] 56222\n\nlength(state.area)\n\n[1] 50\n\n# sort the values in ascending order\nsort(state.area)\n\n [1] 1214 2057 5009 6450 7836 8257 9304 9609 10577\n[10] 24181 31055 33215 36291 40395 40815 41222 42244 45333\n[19] 47716 48523 49576 51609 52586 53104 56154 56290 56400\n[28] 58216 58560 58876 68192 69686 69919 70665 77047 77227\n[37] 82264 83557 84068 84916 96981 97914 104247 110540 113909\n[46] 121666 147138 158693 267339 589757\n\nHere each of these functions returned a value, which was printed upon completion with the print() function, and is equivalent to e.g. print(mean(state.area)).\nAssigning values to variables\nIn R you can use either the <- or the = operators to assign objects to variables. The <- is the preferred style. If we don’t assign an operation to a variable, then it will be printed only and disappear from our environment.\n\n\nx <- length(state.area)\nx # x now stores the length of the state.area vector, which is 50\n\n[1] 50\n\n\n\nx <- x + 10 # overwrites x with new value\n\nx + 20 \n\n[1] 80\n\nNow, what is the value of x?\n\n\nx = ... ?\n...\n\n#[1] 60\n\n\nVectors and atomic types in R\nThere are fundamental data types in R which represent integer, `characters, numeric, and logical values, as well as a few other specialized types.\nEach of these types are represented in vectors, which are a collection of values of the same type. In R there are no scalar types, for example there is no integer type, rather single integer values are stored in an integer vector with length of 1. This is why you see the [1] next to for example 42 when you print it. The [1] indicates the position in the vector.\n\n\n42\n\n[1] 42\n\nVector types\nR has character, integer, double(aka numeric) and logical vector types, as well as more specialized factor, raw, and complex types. We can determine the vector type using the typeof function.\n\n\ntypeof(1.0)\n\n[1] \"double\"\n\ntypeof(\"1.0\")\n\n[1] \"character\"\n\ntypeof(1)\n\n[1] \"double\"\n\ntypeof(1L)\n\n[1] \"integer\"\n\ntypeof(TRUE)\n\n[1] \"logical\"\n\ntypeof(FALSE)\n\n[1] \"logical\"\n\ntypeof(\"hello world\")\n\n[1] \"character\"\n\nYou can change the type of a vector, provided that there is a method to convert between types.\n\n\nas.numeric(\"1.0\")\n\n[1] 1\n\nas.numeric(\"hello world\")\n\n[1] NA\n\nas.character(1.5)\n\n[1] \"1.5\"\n\nas.integer(1.5)\n\n[1] 1\n\nas.integer(TRUE)\n\n[1] 1\n\nas.character(state.area)\n\n [1] \"51609\" \"589757\" \"113909\" \"53104\" \"158693\" \"104247\" \"5009\" \n [8] \"2057\" \"58560\" \"58876\" \"6450\" \"83557\" \"56400\" \"36291\" \n[15] \"56290\" \"82264\" \"40395\" \"48523\" \"33215\" \"10577\" \"8257\" \n[22] \"58216\" \"84068\" \"47716\" \"69686\" \"147138\" \"77227\" \"110540\"\n[29] \"9304\" \"7836\" \"121666\" \"49576\" \"52586\" \"70665\" \"41222\" \n[36] \"69919\" \"96981\" \"45333\" \"1214\" \"31055\" \"77047\" \"42244\" \n[43] \"267339\" \"84916\" \"9609\" \"40815\" \"68192\" \"24181\" \"56154\" \n[50] \"97914\" \n\nNA, Inf, and NaN values\nOften you will find data that contains missing, non-number, or infinite values. There are represented in R as NA, NaN or Inf values.\n\n\n1 / 0 \n\n[1] Inf\n\n-( 1 / 0)\n\n[1] -Inf\n\n0 / 0\n\n[1] NaN\n\nNA\n\n[1] NA\n\nAnd these can be detected in a vector using various is.* functions.\n\n\nis.na()\nis.nan()\nis.infinite()\n\n\nmaking vectors from scratch\nThe c function concatenates values into a vector.\n\n\nc(2, 5, 4)\n\n[1] 2 5 4\n\nc(TRUE, FALSE, TRUE)\n\n[1] TRUE FALSE TRUE\n\nc(\"dog\", \"cat\", \"bird\")\n\n[1] \"dog\" \"cat\" \"bird\"\n\nVectors can only have 1 type, so if you supply multiple types c will silently coerce the result to a single type.\n\n\nc(TRUE, 1.9)\n\n[1] 1.0 1.9\n\nc(FALSE, \"TRUE\")\n\n[1] \"FALSE\" \"TRUE\" \n\nc(1L, 2.0, TRUE, \"Hello\")\n\n[1] \"1\" \"2\" \"TRUE\" \"Hello\"\n\nNumeric ranges can be generated using : or seq\n\n\n1:10\n\n [1] 1 2 3 4 5 6 7 8 9 10\n\nseq(0, 1, by = 0.1)\n\n [1] 0.0 0.1 0.2 0.3 0.4 0.5 0.6 0.7 0.8 0.9 1.0\n\nThere are also functions for sampling from various distributions or vectors.\ne.g.\n\n\n# get 5 values from a normal distribution with mean of 0 and sd of 1\nrnorm(5)\n\n[1] 0.46625168 -1.08376829 -0.39536586 -0.08183084 0.33499270\n\n# get 5 values from uniform distribution from 0 to 1\nrunif(5)\n\n[1] 0.5951937 0.3089459 0.8601477 0.1417087 0.3356661\n\n# sample 5 area values \nsample(state.area, 5)\n\n[1] 121666 9304 2057 589757 24181\n\nSubsetting vectors in R\nR uses 1-based indexing to select values from a vector. The first element of a vector is at index 1. The [ operator can be used to extract (or assign) elements in a vector. Integer vectors or logical vectors can be used to extract values.\n\n\n# extract the second value from the state area and name vectors\nstate.area[2]\n\n[1] 589757\n\nstate.name[2]\n\n[1] \"Alaska\"\n\n\n\n# extract the 1st, 3rd, and 5th name\nstate.name[c(1, 3, 5)]\n\n[1] \"Alabama\" \"Arizona\" \"California\"\n\n# extract a range of names from 2 -> 7\nstate.name[2:7]\n\n[1] \"Alaska\" \"Arizona\" \"Arkansas\" \"California\" \n[5] \"Colorado\" \"Connecticut\"\n\nExtracting a value that does not (yet) exist will yield an NA\n\n\nstate.name[51]\n\n[1] NA\n\nExercise:\nWhat is the total area occupied by the 10 smallest states? What is the total area occupied by the 10 largest states?\n\n\n# hint use the `sort()` function \nsum(sort(state.area)[1:10])\n\n[1] 84494\n\nsum(sort(state.area)[41:50])\n\n[1] 1808184\n\nsum(sort(state.area, decreasing = TRUE)[1:10])\n\n[1] 1808184\n\nUsing vectors and subsetting to perform more complex operations\nWhat if we wanted to know which states have an area greater than 100,000 (square miles)?\nWe can do this in a few steps, which will showcase how simple vector operations, when combined become powerful.\nFirst we can use relational operators to compare values:\n\n\n# are the values of x equal to 10?\nx <- 6:10\nx == 10\n\n[1] FALSE FALSE FALSE FALSE TRUE\n\nx > 10 : are the values of x greater than 10\nx >= 10: are the values of x greater than or equal to 10\nx < 10 : are the values of x less than 10\nx <= 10: are the values of x less than or equal 10\nThese operators fundamentally compare two vectors.\n\n\n# which values of x are equal to the values in y\ny <- c(6, 6, 7, 7, 10)\nx == y\n\n[1] TRUE FALSE FALSE FALSE TRUE\n\nHere, when we ask x < 10 R internally recycles 10 to a vector the same length as x, then evaluates if each element of x is less than 10.\nSee ?Comparison or ?>`` for help menu on relational operators.\nWith this we can now ask, are the state.area values greater than 100000?\n\n\nstate.area > 100000\n\n [1] FALSE TRUE TRUE FALSE TRUE TRUE FALSE FALSE FALSE FALSE FALSE\n[12] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE\n[23] FALSE FALSE FALSE TRUE FALSE TRUE FALSE FALSE TRUE FALSE FALSE\n[34] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE TRUE FALSE\n[45] FALSE FALSE FALSE FALSE FALSE FALSE\n\nThe returned logical vector can be used to subset a vector of the same length. The positions that are TRUE will be retained, whereas the FALSE positions will be dropped.\n\n\n# return the area values > 100000\nstate.area[state.area > 100000]\n\n[1] 589757 113909 158693 104247 147138 110540 121666 267339\n\n# alternatively find the position of the TRUE values using which()\nwhich(state.area > 100000)\n\n[1] 2 3 5 6 26 28 31 43\n\nBut how do we find the state names with areas over 100,000?\nFor this dataset the names of the states are in the same order as the state areas.\nTherefore:\n\n\nstate.name[state.area > 100000]\n\n[1] \"Alaska\" \"Arizona\" \"California\" \"Colorado\" \"Montana\" \n[6] \"Nevada\" \"New Mexico\" \"Texas\" \n\nExercise:\nLet’s answer a related question, how many states are larger than 100,000 square miles?\n\n\n# multiple approaches will work\nlength(which(state.area > 100000))\n\n[1] 8\n\nsum(state.area > 100000)\n\n[1] 8\n\nUsing the sum() function works because TRUE is stored as 1 and FALSE is stored as 0.\n\n\nas.integer(c(TRUE, FALSE, TRUE))\n\n[1] 1 0 1\n\nsum(c(TRUE, FALSE, TRUE))\n\n[1] 2\n\nReplacing or adding values at position\nValues in a vector can be also replaced or added by assignment at specific indexes. In this case the bracket [ notation is left of the assignment operator <-. You can read this as assign value on right to positions in the object on the left.\n\n\n# What if Colorado was named to Colorodo?\nstate.name[6] <- \"Colorodo\"\n\n# what if there were more states added to the US?\nstate.name[c(51, 52)] <- c(\"Puerto Rico\", \"Guam\")\n\n\nThis is a very useful syntax to modify a subset of a vector:\n\n\nx <- c(1, NA, 2, 3)\n\n# replace all values > 1 with 100\nx[x > 1] <- 100 \nx\n\n[1] 1 NA 100 100\n\nis.na() returns TRUE if a value is NA, FALSE otherwise:\n\n\n# replace NA values with -100\nx[is.na(x)] <- -100\nx\n\n[1] 1 -100 100 100\n\nR operations are vectorized\nAs you’ve seen, operations in R tend to execute on all element in a vector. This is called vectorization, and is a key benefit of working in R.\nFor example, say we wanted to take the natural log of some numbers. For this we use the log function.\n\n\nx <- 1:5\nlog(x)\n\n[1] 0.0000000 0.6931472 1.0986123 1.3862944 1.6094379\n\nIf you are used to programming in other languages (e.g C or python) you might have written a for loop to do the same, something like this.\nfor (i in x) { \n log(i)\n}\nIn R this is generally not necessary. The built in vectorization saves typing and makes for very compact and efficient code in R. You can write for loops in R (more on this later in the course) however using the built in vectorization is generally a faster and easier to read solution.\nReview\nTo review today’s material, do the following:\nFor each section with code, try out your own commands. You will learn faster if you type the code yourself and experiment with different commands. If you get errors, try to find help, or ask questions in the class slack channel.\nAcknowledgements and additional references\nThe content of this lecture was inspired by and borrows concepts from the following excellent tutorials:\nhttps://github.com/sjaganna/molb7910-2019\nhttps://github.com/matloff/fasteR\nhttps://r4ds.had.co.nz/index.html\nhttps://bookdown.org/rdpeng/rprogdatascience/\nhttp://adv-r.had.co.nz/Style.html\n\n\n\n", "preview": {}, - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -132,7 +149,7 @@ "categories": [], "contents": "\n\nContents\nInstalling R\nWindows\nMacOS\nLinux\n\nInstall Rstudio\nInstall compiler tools\nWindows\nMacOS\nLinux\n\nInstalling the tidyverse and Rmarkdown packages\nIntroduction to using Rstudio and Rmarkdown\n(Appendix) Installing packages from other sources\n\nThis article will explain how to install R, Rstudio, and R packages.\nPlease watch the video that gives an overview of using Rstudio, installing packages, and the Rmarkdown format.\nInstalling R\nDownload R from CRAN. Go to the CRAN homepage https://cran.r-project.org/. Select the link to download R for your operating system.\nIf you already have R installed, we recommend upgrading to the latest version of R by following the directions below\nWindows\nGo to CRAN, and click Download R for Windows. Next click the base link and select Download R-4.3.2 for Windows to download the .exe file. Open this file to install R.\nMacOS\nGo to CRAN, and click Download R for macOS. Under Latest Release there are two options available depending on which CPU is used in your laptop. Mac uses either Intel (x86) or arm64 (i.e. M1 or M2) processors. You can determine which type you have by clicking on the Apple menu and selecting “About this Mac”. Next to Chip or Processor, it will say either M1 or M2, if you have an arm64 CPU, or it will say Intel Core or similar, indicating you have an Intel x86 CPU.\nDownload the R-4.3.2-arm64.pkg for arm64 or R-4.3.2-x86_64.pkg for Intel x86. Open and follow the prompts to install.\nLinux\nIf you are on linux, then follow the documentation for your linux OS.\nInstall Rstudio\nGo to the Rstudio website and download Rstudio Desktop for your operating system\nOnce downloaded and installed, open up Rstudio to complete the rest of the tutorial.\nInstall compiler tools\nSome R package installations may require a compiler, which is usually not available by default on Windows or macOS.\nWindows\nYou need to install Rtools from CRAN. Follow this link to download RTools 4.3 using the Rtools43 installer https://cran.r-project.org/bin/windows/Rtools/ .\nMacOS\nTo install the necessary compilers, we will follow the recommend steps outlined by CRAN: https://mac.r-project.org/tools/\nXcode\nFirst you will need to install the Xcode command line tools. To do so open Rstudio and click on the “Terminal” tab, which is to the right of the “Console” Tab. Alternatively you can open the Terminal app directly from /Applications/Utilities/ or use the spotlight search tool, search for “terminal”, and open the Terminal App.\nType the following into the terminal and hit Enter:\nsudo xcode-select --install\nWhen prompted for Password:, type in your macOS user password (you wont see any characters printed as you type), and press enter. Click “Install” in the pop up and agree to the license agreement (if you agree of course). This download will require ~9Gb of space so it will take some time to download and install. Verify the installation by typing into terminal:\ngcc --version\nWhich should print something similar to either this:\n#' gcc (GCC) 4.8.5\n#' Copyright (C) 2015 Free Software Foundation, Inc.\n#' This is free software; see the source for copying conditions. There is NO\n#' warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\nor this:\n#' Configured with: --prefix=/Library/Developer/CommandLineTools/usr --with-gxx-include-dir=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/include/c++/4.2.1\n#' Apple clang version 13.0.0 (clang-1300.0.29.30)\n#' Target: arm64-apple-darwin21.3.0\n#' Thread model: posix\n#' InstalledDir: /Library/Developer/CommandLineTools/usr/bin\nHere’s a youtube video explainer that also shows the process.\ngfortran\nNext you need to install gfortran. If you’ve installed the most recent version of R (or at least 4.3.0), then you can install using the gfortran-12.2-universal.pkg.\nOnce you’ve run the gfortran installer the last step is to make sure that this program is in your PATH. This step will make the gfortran program visible to R, and other programs.\nFirst determine which type of shell you have (typically bash or zsh). Execute the following in a terminal (click either on the terminal pane in Rstudio, or open the terminal app in macOS).\necho $SHELL\nIf you see /bin/zsh then make a plain text file called .zshrc in your home directory (e.g. /Users/Your-macOS-username/.zshrc), if it doesn’t already exist. If instead you see /bin/bash then make a file called .bashrc in your home directory, if it doesn’t already exist. You can use Rstudio to make a new plain-text file (File->New file->Text) or by opening up the Textedit app, then click Format->Make Plain Text.\nAdd the following line of text to the file (and keep any other text if already present).\nexport PATH=$PATH:/opt/gfortran/bin\nSave the text file to your home directory. You may need to rename the file after saving to ensure that it doesn’t end with .txt. (e.g. rename .zshrc.txt -> .zshrc). This file will be a hidden file. Hidden files can be seen in the Finder app by pressing Command + Shift + . (period) to toggle on/off visualizing hidden files.\nClose and reopen Rstudio.\nLinux\nYou should have a compiler available already.\nInstalling the tidyverse and Rmarkdown packages\nNow that you have R and Rstudio set up we will install packages.\nPackages are extensions to the base R installation that provide additionally functionality to the language. In this course we will use packages from the tidyverse, which is a collection of packages commonly used for data science and interactive data analysis. Installing the tidyverse package will install an entire collection of tidyverse packages.\nCRAN is the official R package repository. CRAN has 18,000+ packages, including the tidyverse packages. Packages from CRAN are installed using the install.packages() R function. A successful install of a package will only need to be done once, until you update R to a new version.\nOpen Rstudio to launch R. Then in the console pane, execute the following command to install the tidyverse:\ninstall.packages(\"tidyverse\")\n\n\n\nThis command will take a few minutes to run while all of the packages are installed. Package installation will be completed once the > prompt reappears. Once complete, test package installation by loading the package(s)\nlibrary(tidyverse)\nIf successful you will see something like this:\n\n\n\nAn error will look like this (note misspelled package name for demonstration purposes):\n\n\n\nIf loading tidyverse completes without errors then the packages have been installed. You’ll also now see additional packages (ggplot2, dplyr, tidyr) listed under the “Packages” pane.\nIf there is an error installing tidyverse, you’ll likely see the following at the end of the command:\n#' Warning in install.packages :\n#' installation of package ‘tidyverse’ had non-zero exit status\nIf this happens, contact the course instructors to help troubleshoot the installation issue.\nAnother package that we will use in the course is rmarkdown, to install run:\ninstall.packages(\"rmarkdown\")\nand verify installation by running library(rmarkdown)\nIntroduction to using Rstudio and Rmarkdown\nNow that you have installed R and Rstudio, please watch this video (~20 minutes) that provides an overview of how to use Rstudio IDE and an introduction to the Rmarkdown format.\nintro-to-rstudio.mp4\nintro-to-rstudio.mov\n(Appendix) Installing packages from other sources\nThere are 2 additional commonly used repositories for R packages. These are not needed to complete the prerequisite but are useful resources that you will use as you perform more coding in R.\nBioconductor is a repository that hosts 2,000+ bioinformatics related packages.\nTo install bioconductor packages you should use the CRAN package BiocManager. BiocManager has a function called install() to install bioconductor packages. For example to install ComplexHeatmap\ninstall.packages(\"BiocManager\")\nlibrary(BiocManager)\ninstall(\"ComplexHeatmap\")\n# or equivalently you could run BiocManager::install(\"ComplexHeatmap\")\nGithub hosts open-source code from millions of software projects. R packages hosted on github can be installed using the remotes package. Packages on github are generally the development version of a package, or a package that has not been contributed to either CRAN or Bioconductor. To install you’ll need to find the organization name and the repository name on github to install.\nFor example to install the LaCroixColorR package:\ninstall.packages(\"remotes\")\nremotes::install_github('johannesbjork/LaCroixColoR')\n\n# or equivalently you could use BiocManager, which uses remotes internally\nBiocManager::install(`johannesbjork/LaCroixColoR`)\n\n\n\n", "preview": "posts/2023-11-06-install-r/img/install-packages.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -149,7 +166,7 @@ "categories": [], "contents": "\nThe Rmarkdown for this document is\nhttps://github.com/rnabioco/bmsc-7810-pbda/blob/main/_posts/2022-11-23-class-6-intro-to-ggplot2-part3/class-6-intro-to-ggplot2-part3.Rmd\nGoals for today\nNew dataset diamonds\nFaceting plots\nStoring plots as variables\nColor palettes\nApplying themes\nCombining plots with patchwork\nDataset: Diamonds\n\n\n\nA dataset containing the prices and other attributes of almost 54,000\ndiamonds.\n\n\nhead(diamonds)\n\n# A tibble: 6 × 10\n carat cut color clarity depth table price x y z\n \n1 0.23 Ideal E SI2 61.5 55 326 3.95 3.98 2.43\n2 0.21 Premium E SI1 59.8 61 326 3.89 3.84 2.31\n3 0.23 Good E VS1 56.9 65 327 4.05 4.07 2.31\n4 0.29 Premium I VS2 62.4 58 334 4.2 4.23 2.63\n5 0.31 Good J SI2 63.3 58 335 4.34 4.35 2.75\n6 0.24 Very Good J VVS2 62.8 57 336 3.94 3.96 2.48\n\nA data frame with 53940 rows and 10 variables:\nprice = price in US dollars ($326–$18,823)\ncarat = weight of the diamond (0.2–5.01)\ncut = quality of the cut (Fair, Good, Very Good, Premium, Ideal)\ncolor = diamond color, from D (best) to J (worst)\nclarity = a measurement of how clear the diamond is (I1 (worst), SI2,\nSI1, VS2, VS1, VVS2, VVS1, IF (best))\nx = length in mm (0–10.74)\ny = width in mm (0–58.9)\nz = depth in mm (0–31.8)\ndepth = total depth percentage = z / mean(x, y) = 2 * z / (x + y)\n(43–79)\ntable = width of top of diamond relative to widest point (43–95)\n\n\nggplot(diamonds, aes(x=carat, y=price)) + \n geom_point()\n\n\n\nExercise: Review the last class. Make a histogram showing the\ndistribution of diamond prices. Color by the cut of the diamond. What\nstatements can you make about the relationships shown.\n\n\n\nExercise: More review. Create a freqpoly plot showing the frequency\ncount of the carat and the color as the cut of diamond. Does this help\nexplain the ideal cut price?\n\n\n\nThere are so many data points in this dataset as seen by our original\nscatterplot. Before moving on we can subset this dataset by using sample\nto grab a random selection of 1000 rows for downstream analysis.\n\n\nset.seed(1337) # set the random seed so that we get the same random rows everytime\n\nsubset_diamonds <- diamonds[sample(nrow(diamonds), 1000), ]\n\nggplot(subset_diamonds, aes(x=carat, y=price)) + \n geom_point()\n\n\n\nIntroducing the Facet\nOne way that we can take an attribute from your data and expand it to\nplot it into multiple plots, one for each level, letting you view them\nseparately. Just as a cut diamond has different flat edges called\nfacets, in ggplot this type of breaking out the levels of the data into\nmultiple plots is called “faceting”. One of the easiest ways to do this\nis by using the facet_wrap() function.\n\n\nggplot(subset_diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point() + \n facet_wrap(~cut, nrow = 1)\n\n\n\nThe second type of facet function is the facet_grid()\n\n\nggplot(subset_diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point() + \n facet_grid(clarity ~ cut)\n\n\n\nThis is a good time to introduce a way to modify the size of the figure\nbeing displayed in RMarkdown. We can edit the curly braces to give\nspecial instructions for the cell. Kent has previous showed this to you\nas well. Here we can add fig.width=20 to increase the width of the\nfigure. You can also try fig.height. There are numerous ways you can\ninfluence the plot using this format and most of them start with the\nfig. prefix.\n\n\nggplot(diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point() + \n facet_grid(clarity ~ cut)\n\n\n\nExercise: Use the dataset from last class iris. Make a scatterplot of\nSepal Width and Sepal Length and color by the Species. Use a\nfacet_wrap to break out the Species.\n\n\n\nSaving Plot Objects\nOne concept that can be useful is that you can assign ggplot plots to a\nvariable just like any other object in R. This can allow you to reuse\nthe plot over and over again simply by calling the variable name you\nsaved the plot. You can also continue to add layers to these plots and\ncan we a quick way to test and compare different versions of a plot.\n\n\np1 <- ggplot(subset_diamonds, aes(x=carat, y=price, color=cut)) +\n geom_point()\n\n\nNotice that nothing was plotting when you run this code. Instead the\nplot is saved to the p1 variable. We can visualize this plot anytime\nsimply by calling the variable.\n\n\np1\n\n\n\nWe can add any additional layers just as we would when building the\nplot. Let’s look at a facet_wrap of the clarity.\n\n\np1 + facet_wrap(~clarity)\n\n\n\nWe changed our mind and now we want to compare this to the same base\nplot but use a facet_grid breaking out the diamond color.\n\n\np1 + facet_grid(clarity~color)\n\n\n\nColor Palettes\nYou can easily change the types and ranges of colors being used in your\nplots. Here is the default color palette:\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point()\n\n\n\nWe can use the scale_color_brewer() to set a different type of\npalette. There are many default options to choose from and maybe more\ncustom ones you can install.\nhttps://r-graph-gallery.com/38-rcolorbrewers-palettes.html\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point() +\n scale_color_brewer(palette = \"RdYlBu\")\n\n\n\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point() +\n scale_color_brewer(palette = \"Accent\")\n\n\n\n\n\nggplot(subset_diamonds, aes(carat, price, color = clarity)) +\n geom_point() +\n scale_color_manual(values = c(\"red\", \"blue\", \"green\", \"yellow\", \"purple\", \"white\", \"black\", \"gray\"))\n\n\n\nThemes\nOne of the most fun aspects of ggplot is the ability to quickly change\nthe entire look of your plots with themes.\n\n\nptest <- ggplot(iris, aes(x=Sepal.Width, y=Sepal.Length, color = Species)) +\n geom_point() +\n facet_wrap(~ Species)\n\nptest\n\n\n\n\n\nptest + theme_dark()\n\n\n\n\n\nptest + theme_minimal()\n\n\n\n\n\nptest + theme_bw()\n\n\n\n\n\nptest + theme_classic()\n\n\n\n\n\nptest + theme_void()\n\n\n\nYou can install custom themes….\nhttps://ryo-n7.github.io/2019-05-16-introducing-tvthemes-package/\nhttps://github.com/Mikata-Project/ggthemr\nhttp://xkcd.r-forge.r-project.org/\nCombining multiple plots\nOne useful technique when assembling figures is to be able to stitch\nmultiple plots together into a single image. There is a special add on\npackage that allows us to do just that with simple syntax. This package\nis called patchwork and will need to be installed as it is not\nincluded in the tidyverse. It can be installed with\ninstall.packages(\"patchwork\"). More info at\nhttps://patchwork.data-imaginist.com/\n\n\nlibrary(patchwork)\n\n\nSave the plots as object variables.\n\n\np1 <- ggplot(mtcars) + \n geom_point(aes(mpg, disp))\n\np2 <- ggplot(mtcars) + \n geom_boxplot(aes(gear, disp, group = gear))\n\n\nTo use patchwork simply place the plus operator to “add” two plots\ntogether:\n\n\np1 + p2\n\n\n\nWhy stop at just two plots? We can keep adding more.\n\n\np3 <- ggplot(mtcars) + \n geom_smooth(aes(disp, qsec))\n\np4 <- ggplot(mtcars) + \n geom_bar(aes(carb))\n\n\nAnd use more complex ways of displaying them.\n\n\n(p1 + p2 + p3) / p4\n\n\n\nTo annotate the whole group we need to use a special plot_annotation()\nfunction:\n\n\n(p1 | p2 | p3) / p4 + \n plot_annotation(\n title = 'The surprising truth about mtcars',\n subtitle = 'These 3 plots will reveal yet-untold secrets about our beloved data-set',\n caption = 'Disclaimer: None of these plots are insightful')\n\n\n\nYou can even automatically add the subplot letter annotations. Publish\ntime!\n\n\n(p1 | p2 | p3) / p4 + \n plot_annotation(tag_levels = 'A')\n\n\n\n\n\n(p1 | p2 | p3) / p4 + \n plot_annotation(title = \"Figure 1: Motor Trend 1974 Car Stats\", tag_levels = 'A')\n\n\n\nExercise: Change the order of the plots combined with patchwork so that\np4 is in the middle of the top row and p2 is now on the bottom row. See\nhow the plot adapts.\n\n\n\nThanks for listening. Keep on plotting and exploring the world of\nggplot2!\n\n\n\n", "preview": "posts/2022-11-23-class-6-intro-to-ggplot2-part3/class-6-intro-to-ggplot2-part3_files/figure-html5/unnamed-chunk-3-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -166,7 +183,7 @@ "categories": [], "contents": "\nThe Rmarkdown for this document is: https://github.com/rnabioco/bmsc-7810-pbda/blob/main/_posts/2022-11-23-class-5-intro-to-ggplot2-part2/class-5-intro-to-ggplot2-part2.Rmd\nGoals for today\nNew dataset: Iris\nPlotting the categorical data from iris measurements\nBox plots and violin plots\nFrequency and density plots\nUsing stat layers\nAdding additional annotations\nAxis, scales, and coordinate Systems\nThe Iris Dataset\nFor this class we are going to use a new built in dataset that involves\nthe measurements of Iris flowers. In particular the measurements involve\nthe width and length of two structures of the flower: the petal and the\nsepal. Here is an overview of flower structure.\n\n\n\nThe Iris dataset is classically used in machine learning and\nclassification projects. Three species of iris were included in this\nstudy: iris setosa, iris versicolor, and iris virginica. Measurements\nwere taken in 1936 by famous statistician RA Fisher known for the\nStudent’s t-test and F-distribution.\nhttp://archive.ics.uci.edu/ml/datasets/Iris\n\n\n\nLet’s look at the this new dataset with head. You can see that it is\nin tidy format with each observation being a new row.\n\n\nhead(iris)\n\n Sepal.Length Sepal.Width Petal.Length Petal.Width Species\n1 5.1 3.5 1.4 0.2 setosa\n2 4.9 3.0 1.4 0.2 setosa\n3 4.7 3.2 1.3 0.2 setosa\n4 4.6 3.1 1.5 0.2 setosa\n5 5.0 3.6 1.4 0.2 setosa\n6 5.4 3.9 1.7 0.4 setosa\n\nTo get a list of the species in this study we can look at all the\nunique() entries in the Species column.\n\n\nunique(iris$Species)\n\n[1] setosa versicolor virginica \nLevels: setosa versicolor virginica\n\nEach one of the species is represented and now we have the exact names\nas written by each measurement. To get the number of measurements for\neach species we can use the summary() function.\n\n\nsummary(iris$Species)\n\n setosa versicolor virginica \n 50 50 50 \n\nWe can begin by looking at the relationships between some of the\nmeasurements by looking at a scatter plot. Here we have Sepal.Length on\nthe x-axis and Sepal.Width on the y-axis.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point()\n\n\n\nExercise: Despite this showing all the data points. How is this not very\ninformative? As a review of last class, add to this plot to make it more\ninformative?\n\n\n\nExercise: Remake this scatterplot but this time for Petal.Width and\nPetal.Length and plot ONLY the iris virginica species data points.\n\n\n\nPlotting the Categorical Data\nSpecies data points with geom_point\nTypically we can look at the distribution of a particular measurement\nvalue based on the category of the measurement, in this case the\nspecies. In this way we can make comparisons between the species. As\nbefore we can use a geom_point_() to plot the values for each species.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_point()\n\n\n\nWhile this does show a basic distribution of Sepal.Width for each\nSpecies, many of the points that have the same value are actually\nhidden! One way we can improve on this is by adding a bit of jitter or\nrandom horizontal position to each point.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter()\n\n\n\nNotice that if you rerun the plot the points are in different locations.\nThe space added by the jitter is randomly generated everytime. Don’t\nexpect them to look the same everytime!\nSide note: You can also use geom_point() geometry function with the\nposition = position_jitter() setting and it will generate the same\nplot as with geom_jitter()\nYou can also tighten the range of the jitter by specifying a width.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter(width=0.1)\n\n\n\nThe Boxplot\nA frequently used plot that is used to better descriptively show this\ntype of data is a boxplot. We can generate a box plot of this data\nsimply by adding a second geom layer called geom_boxplot(). This way\nwe keep the point layer but also have the boxplot.\n\n\n\nHere we can add a geom_boxplot layer to our existing jittered\nscatterplot.\n\n\nggplot(iris, (aes(x = Species, y = Sepal.Width))) +\n geom_jitter() +\n geom_boxplot()\n\n\n\nExercise: Many of the points are hidden behind the boxplot. Try changing\nthe order of the layers to see if it matters. What is another way you\ncould fix this?\n\n\n\nViolin Plot\nAnother type of frequently used plot is the violin plot. This plot shows\na continuous density distribution.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_violin() +\n geom_jitter()\n\n\n\nStats Layers\nStats or statistics layers allows us to calculate certain metrics about\nour data and potentially visualize them. First we will look at some of the geom that use stats in their plots.\nFrequency and Density Plots\nFor instance here is a new type of plot that calculates frequency of counts across all measurements of\nSepal.Width. It uses a stat to count the number of measurements at specific values. We could also show the color aes to visualize all the species.\n\n\nggplot(iris) +\n geom_freqpoly(aes(x = Sepal.Width))\n\n\n\ngeom_dotplot() is another way to visualize representative counts. Note that settings stackgroups = TRUE allows you to see all of the dots by stacking them vertically on top of one another without overlap. It uses a stat to count the number of measurements at specific values and represents them as a dot.\n\n\nggplot(iris) +\n geom_dotplot(aes(x = Sepal.Width, fill = Species), stackgroups = TRUE)\n\n\n\nDensity plots can overlap to show a comparison between groups and visualize distribution. It uses a stat to calculate a density metric.\n\n\nggplot(iris) +\n geom_density(aes(x = Sepal.Width, color = Species))\n\n\n\nFinally we have a traditional histogram representing the counts of specific measurement values as above but plotted as a bar plot. It also uses a stat to count the number of measurements at these specific values.\n\n\nggplot(iris) +\n geom_histogram(aes(x = Sepal.Width))\n\n\n\nUnderneath the hood the geom_histogram function is using a stat\nfunction called bin this essentially taking each measurement and\nplacing it in a specific sized category and calculating the frequency of\nthis occurrence. We can modify either the binwidth or the number of\nbins arguments to modify this behavior. For instance if there are 50\nmeasurements from say 1 to 4.5. This range would be divided by the\nnumber of bins. Each measurement value would fall into one of these bins\nand a count would be added for that bin.\n\n\nggplot(iris) +\n geom_histogram(aes(x = Sepal.Width), stat = \"bin\", bins = 10)\n\n\n\nStat Functions\nStats layers are additional information that we calculate and add to the\nplot. Essentially every geom_ function that we have been seen utilizes\ncalculations to produce the plots. Each of these geom_ functions has\nan equivalent stat_ function. It is beyond the scope of this class to\nget into the details of all of these stat functions. Here we will look\nat a particular function called stat_summary that we can use to plot\nsome summary statistics.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(fun = \"mean\",\n geom = \"point\",\n color = \"red\")\n\n\n\nSome of the other options for stat_summary:\ngeoms: point, errorbar, pointrange, linerange, crossbar\nfuns: mean, median, max, min\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(fun = \"mean\",\n geom = \"crossbar\",\n width = 0.5,\n color = \"red\")\n\n\n\nWe can combine multiple stat_summary layers to add additional\ninformation.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(fun = \"mean\",\n geom = \"crossbar\",\n width = 0.5,\n color = \"red\") +\n stat_summary(fun = \"median\",\n geom = \"crossbar\",\n width = 0.5,\n color = \"blue\")\n\n\n\nPlotting the standard error and the confidence intervals\nPlotting the standard error.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(geom = \"errorbar\",\n fun.data = mean_se)\n\n\n\nTo calculate the standard deviation and produce the confidence intervals\nyou can pass mean_cl_normal to the fun.data argument. Note you may\nneed to install the Hmisc package to get this working.\ninstall.packages(\"Hmisc\")\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_jitter() +\n stat_summary(geom = \"errorbar\",\n fun.data = mean_cl_normal)\n\n\n\nAnnotations\nAnnotations are easy ways to add extra emphasis to your plots. It can be\nmuch more efficient to have them placed on your plots programatically\nrather than trying to add them later with Photoshop or Illustrator.\nUsing geom_text()\ngeom_text() is an easy way to play text on a plot to annotate. We can even use its aes() function to add column information to the plot like so.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point() +\n geom_text(aes(label=Species))\n\n\n\nNot very practical. Let’s look at the documentation to get some better ideas.\n\n\n?geom_text\n\n\nThere are several options we can add to make things a little neater.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point() +\n geom_text(aes(label=Species), nudge_y = .1, check_overlap = T, size = 3)\n\n\n\nWe can also manually place text anywhere we would like in the plot. This could be a way to annotate whole groups or parts of the visualization.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_text(aes(label=\"setosa\"), x=5, y=4, size = 5) +\n geom_text(aes(label=\"versicolor\"), x=5.5, y=2.25, size = 5) + \n geom_text(aes(label=\"virginica\"), x=7.5, y=3.5, size = 5)\n\n\n\nThe annotate function\nThe annotate function can be used to pass specific types of geometries\nthat you can manually draw on your plot.\n\n\n?annotate\n\n\nHere is an example of drawing a rectangle.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n annotate(\"rect\", xmin=5.5, xmax=6.5, ymin=2.5 , ymax=3.2, alpha=0.2, color=\"blue\")\n\n\n\nUsing a segment geom to produce an arrow. Notice how we need to add the\narrow function.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n annotate(\"segment\", x = 7, xend = 7, y = 4.5, yend = 3.25, color = \"pink\", size=3, alpha=0.6, arrow=arrow())\n\n\n\nDrawing intercept lines with geom_lines\nYou can add horizontal or vertical lines to show cut offs.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_hline(yintercept=4, color = \"orange\", size = 1)\n\n\n\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_vline(xintercept=7, color = \"orange\", size = 1)\n\n\n\nCan add a slope line.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color= Species)) +\n geom_abline(slope = .5, intercept = 1)\n\n\n\nFiltering data as annotation\nYou can also filter your data during the annotation process and use that\nas a way to clearly highlight features of interest.\nHere by limiting the color to specific measurements.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point() + \n geom_point(data = filter(iris, Sepal.Width > 3.25), aes(color = Species))\n\n\n\nAnd here by limiting the text annotation to specific measurements.\n\n\nggplot(iris, aes(x = Sepal.Length, y = Sepal.Width)) +\n geom_point(aes(color = Species)) + \n geom_text(data = filter(iris, Sepal.Width > 4), aes(label = Species), vjust = 1)\n\n\n\nExercise: Plot a scatter plot of the Petal.Length and Petal.Width and color by the species of iris. Place a rectangle around the group of points representing the data from the setosa species. Place text above the rectangle that displays “smallest flower”.\n\n\n\nAxis, Scales, and Coordinate Systems\nScales are ways of modifying how the data and the coordinates are shown. When you run this code below there are actually several default hidden scales functions being added.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point()\n\n\n\nNotice how there are three scale function layers added. These are actually being run above but are hidden by default. If you run this version you will get the same plot as above.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n scale_x_continuous() + \n scale_y_continuous() + \n scale_colour_discrete()\n\n\n\nBasically scale_x_ and scale_y_ functions can be used to modify the respective axis appearance and type. For instance we can change the x axis to be on a log scale by using scale_x_log10(). Great way to visualize without having to transform the actual data.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n scale_x_log10()\n\n\n\nYou can also reverse an axis.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n scale_x_reverse()\n\n\n\nYou can manually set the x and y axis range by using the xlim() and ylim() functions.\n\n\nggplot(iris, aes(x = Petal.Length, y = Petal.Width)) +\n geom_point() +\n xlim(0,10) +\n ylim(0,5)\n\n\n\nThe third default scale in the plot was scale_colour_discrete(). This type of scale modifies how the color can be mapped across the data.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width, color= Sepal.Length)) + \n geom_jitter() + \n scale_color_gradient(low = \"blue\", high = \"red\")\n\n\n\n\n\n#use autocomplete to all the scales options\n#scale_\n\n\nLast class I showed that you could quickly change the axis to swap the\ncoordinates. Here is another way to do that by interacting with the\ncoordinate layer using the coord_flip() function.\n\n\nggplot(iris, aes(x = Species, y = Sepal.Width)) +\n geom_violin() +\n geom_jitter() +\n coord_flip()\n\n\n\nSessionInfo\n\n\nsessionInfo()\n\nR version 4.2.2 (2022-10-31)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.6\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRblas.0.dylib\nLAPACK: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRlapack.dylib\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods \n[7] base \n\nother attached packages:\n[1] forcats_0.5.2 stringr_1.4.1 dplyr_1.0.10 purrr_0.3.5 \n[5] readr_2.1.3 tidyr_1.2.1 tibble_3.1.8 ggplot2_3.4.0 \n[9] tidyverse_1.3.2\n\nloaded via a namespace (and not attached):\n [1] fs_1.5.2 lubridate_1.8.0 RColorBrewer_1.1-3 \n [4] httr_1.4.4 tools_4.2.2 backports_1.4.1 \n [7] bslib_0.4.1 utf8_1.2.2 R6_2.5.1 \n[10] rpart_4.1.19 Hmisc_4.7-2 DBI_1.1.3 \n[13] colorspace_2.0-3 nnet_7.3-18 withr_2.5.0 \n[16] tidyselect_1.2.0 gridExtra_2.3 downlit_0.4.2 \n[19] compiler_4.2.2 cli_3.4.1 rvest_1.0.3 \n[22] htmlTable_2.4.1 xml2_1.3.3 labeling_0.4.2 \n[25] sass_0.4.2 scales_1.2.1 checkmate_2.1.0 \n[28] digest_0.6.30 foreign_0.8-83 rmarkdown_2.17 \n[31] base64enc_0.1-3 jpeg_0.1-10 pkgconfig_2.0.3 \n[34] htmltools_0.5.3 dbplyr_2.2.1 fastmap_1.1.0 \n[37] highr_0.9 htmlwidgets_1.5.4 rlang_1.0.6 \n[40] readxl_1.4.1 rstudioapi_0.14 jquerylib_0.1.4 \n[43] farver_2.1.1 generics_0.1.3 jsonlite_1.8.3 \n[46] distill_1.5 googlesheets4_1.0.1 magrittr_2.0.3 \n[49] Formula_1.2-4 interp_1.1-3 Matrix_1.5-1 \n[52] Rcpp_1.0.9 munsell_0.5.0 fansi_1.0.3 \n[55] lifecycle_1.0.3 stringi_1.7.8 yaml_2.3.6 \n[58] grid_4.2.2 crayon_1.5.2 deldir_1.0-6 \n[61] lattice_0.20-45 haven_2.5.1 splines_4.2.2 \n[64] hms_1.1.2 knitr_1.40 pillar_1.8.1 \n[67] reprex_2.0.2 glue_1.6.2 evaluate_0.17 \n[70] latticeExtra_0.6-30 data.table_1.14.4 modelr_0.1.9 \n[73] png_0.1-7 vctrs_0.5.0 tzdb_0.3.0 \n[76] cellranger_1.1.0 gtable_0.3.1 assertthat_0.2.1 \n[79] cachem_1.0.6 xfun_0.34 broom_1.0.1 \n[82] survival_3.4-0 googledrive_2.0.0 gargle_1.2.1 \n[85] memoise_2.0.1 cluster_2.1.4 ellipsis_0.3.2 \n\n\n\n\n", "preview": "posts/2022-11-23-class-5-intro-to-ggplot2-part2/class-5-intro-to-ggplot2-part2_files/figure-html5/unnamed-chunk-6-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -183,7 +200,7 @@ "categories": [], "contents": "\n\nContents\nRmarkdown\nCaching\nBetter, save and load .rds files\n\nhere, folder structure management\nstyler, clean up code readability\nReproducibility\nPrint environment information with sessionInfo()\nUse renv to manage package dependencies\n\nBenchmarking, with microbenchmark and profvis\nDebugging R code\nLook at the call stack with traceback()\n\nBuilding your own R package\nshiny, interactive web app for data exploration\nParsing specific types of formats:\nJSON\nGenomics data (fasta, fastq, vcf, bam, bed, bigwig)\n\nUsing R on the command-line\nGit and Github\nPut your code on GitHub\nExample repos (RBI)\nAsking for help with other packages on GitHub\n\nFinding useful packages\nBioconductor\nFinding help online\nCheat sheets\nOffline help\nSometimes code is just broken\nAdditional Resources\nGeneral/Data science\nGenomics\nA meta-list of R resources\nWriting high-performance R functions with R + C++\n\n\nRmarkdown\nRead the Guide to RMarkdown for an exhaustive description of the various formats and options for using RMarkdown documents. Note that HTML for this class were all made from Rmd, using the distill blog format\nThe Rmarkdown for this class is on github\nCaching\nYou can speed up knitting of your Rmds by using caching to store the results from each chunk, instead of rerunning them each time. Note that if you modify the code chunk, previous caching is ignored.\nFor each chunk, set {r, cache = TRUE}\nOr the option can be set globally at top of the document. Like this:\nknitr::opts_chunk$set(cache = TRUE)\nBetter, save and load .rds files\nRun once, save, and load instead of rerunning resource intensive parts.\nIf you have non-deterministic functions, such as kmeans, remember to set.seed, or save and load result objects.\n\nif(!file.exists(\"long-step.rds\")){\n ...\n code or a script\n source(\"path/to/script.R\")\n ...\n saveRDS(obj, \"long-step.rds\")\n} else {\n obj <- readRDS(\"long-step.rds\")\n}\n\nhere, folder structure management\nhttps://github.com/jennybc/here_here\n\n\nhere::here() # always points to top-level of current project\n\nhere::here(\"_posts\", \"2022-10-03-install-r\", \"install-r.Rmd\") # never confused about folder structure\n\n\nstyler, clean up code readability\nRefer to this style guide often, so you don’t have to go back to make the code readable/publishable later.\n\n\nstyler::style_text(\"\nmy_fun <- function(x, \ny, \nz) {\n x+ z\n}\n \")\n\n#> \n#> my_fun <- function(x,\n#> y,\n#> z) {\n#> x + z\n#> }\n\n\n# styler::style_file # for an entire file\n# styler::style_dir # directory \n\n\nReproducibility\nPrint environment information with sessionInfo()\nIt’s very helpful to have a record of which packages were used in an analysis. One approach is to print the sessionInfo().\n\nShow session info\n\n\nsessionInfo()\n\n#> R version 4.2.0 (2022-04-22)\n#> Platform: x86_64-apple-darwin17.0 (64-bit)\n#> Running under: macOS Big Sur/Monterey 10.16\n#> \n#> Matrix products: default\n#> BLAS: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRblas.0.dylib\n#> LAPACK: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRlapack.dylib\n#> \n#> locale:\n#> [1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n#> \n#> attached base packages:\n#> [1] stats graphics grDevices utils datasets methods \n#> [7] base \n#> \n#> other attached packages:\n#> [1] here_1.0.1 forcats_0.5.1 stringr_1.4.1 dplyr_1.0.10 \n#> [5] purrr_0.3.5 readr_2.1.2 tidyr_1.2.0 tibble_3.1.8 \n#> [9] ggplot2_3.3.6 tidyverse_1.3.1\n#> \n#> loaded via a namespace (and not attached):\n#> [1] lubridate_1.8.0 assertthat_0.2.1 rprojroot_2.0.3 \n#> [4] digest_0.6.30 utf8_1.2.2 prettycode_1.1.0 \n#> [7] R6_2.5.1 cellranger_1.1.0 backports_1.4.1 \n#> [10] reprex_2.0.1 evaluate_0.16 httr_1.4.4 \n#> [13] pillar_1.8.1 rlang_1.0.6 readxl_1.4.0 \n#> [16] rstudioapi_0.13 jquerylib_0.1.4 R.utils_2.12.0 \n#> [19] R.oo_1.25.0 rmarkdown_2.14 styler_1.7.0 \n#> [22] munsell_0.5.0 broom_0.8.0 compiler_4.2.0 \n#> [25] modelr_0.1.8 xfun_0.32 pkgconfig_2.0.3 \n#> [28] htmltools_0.5.2 downlit_0.4.2 tidyselect_1.2.0 \n#> [31] fansi_1.0.3 crayon_1.5.2 tzdb_0.3.0 \n#> [34] dbplyr_2.2.1 withr_2.5.0 R.methodsS3_1.8.2\n#> [37] grid_4.2.0 jsonlite_1.8.3 gtable_0.3.0 \n#> [40] lifecycle_1.0.3 DBI_1.1.3 magrittr_2.0.3 \n#> [43] scales_1.2.0 cli_3.4.1 stringi_1.7.8 \n#> [46] cachem_1.0.6 fs_1.5.2 xml2_1.3.3 \n#> [49] bslib_0.3.1 ellipsis_0.3.2 generics_0.1.3 \n#> [52] vctrs_0.4.1 distill_1.5 tools_4.2.0 \n#> [55] R.cache_0.15.0 glue_1.6.2 hms_1.1.2 \n#> [58] fastmap_1.1.0 yaml_2.3.6 colorspace_2.0-3 \n#> [61] rvest_1.0.2 memoise_2.0.1 knitr_1.39 \n#> [64] haven_2.5.0 sass_0.4.1\n\nSee also the sessioninfo package, which provide more details:\n\n\nsessioninfo::session_info()\n\n#> ─ Session info ─────────────────────────────────────────────────────\n#> setting value\n#> version R version 4.2.0 (2022-04-22)\n#> os macOS Big Sur/Monterey 10.16\n#> system x86_64, darwin17.0\n#> ui X11\n#> language (EN)\n#> collate en_US.UTF-8\n#> ctype en_US.UTF-8\n#> tz America/Denver\n#> date 2022-12-16\n#> pandoc 2.19.2 @ /Applications/RStudio.app/Contents/MacOS/quarto/bin/tools/ (via rmarkdown)\n#> \n#> ─ Packages ─────────────────────────────────────────────────────────\n#> package * version date (UTC) lib source\n#> assertthat 0.2.1 2019-03-21 [1] CRAN (R 4.2.0)\n#> backports 1.4.1 2021-12-13 [1] CRAN (R 4.2.0)\n#> broom 0.8.0 2022-04-13 [1] CRAN (R 4.2.0)\n#> bslib 0.3.1 2021-10-06 [1] CRAN (R 4.2.0)\n#> cachem 1.0.6 2021-08-19 [1] CRAN (R 4.2.0)\n#> cellranger 1.1.0 2016-07-27 [1] CRAN (R 4.2.0)\n#> cli 3.4.1 2022-09-23 [1] CRAN (R 4.2.0)\n#> colorspace 2.0-3 2022-02-21 [1] CRAN (R 4.2.0)\n#> crayon 1.5.2 2022-09-29 [1] CRAN (R 4.2.0)\n#> DBI 1.1.3 2022-06-18 [1] CRAN (R 4.2.0)\n#> dbplyr 2.2.1 2022-06-27 [1] CRAN (R 4.2.0)\n#> digest 0.6.30 2022-10-18 [1] CRAN (R 4.2.0)\n#> distill 1.5 2022-09-07 [1] CRAN (R 4.2.0)\n#> downlit 0.4.2 2022-07-05 [1] CRAN (R 4.2.0)\n#> dplyr * 1.0.10 2022-09-01 [1] CRAN (R 4.2.0)\n#> ellipsis 0.3.2 2021-04-29 [1] CRAN (R 4.2.0)\n#> evaluate 0.16 2022-08-09 [1] CRAN (R 4.2.0)\n#> fansi 1.0.3 2022-03-24 [1] CRAN (R 4.2.0)\n#> fastmap 1.1.0 2021-01-25 [1] CRAN (R 4.2.0)\n#> forcats * 0.5.1 2021-01-27 [1] CRAN (R 4.2.0)\n#> fs 1.5.2 2021-12-08 [1] CRAN (R 4.2.0)\n#> generics 0.1.3 2022-07-05 [1] CRAN (R 4.2.0)\n#> ggplot2 * 3.3.6 2022-05-03 [1] CRAN (R 4.2.0)\n#> glue 1.6.2 2022-02-24 [1] CRAN (R 4.2.0)\n#> gtable 0.3.0 2019-03-25 [1] CRAN (R 4.2.0)\n#> haven 2.5.0 2022-04-15 [1] CRAN (R 4.2.0)\n#> here * 1.0.1 2020-12-13 [1] CRAN (R 4.2.0)\n#> hms 1.1.2 2022-08-19 [1] CRAN (R 4.2.0)\n#> htmltools 0.5.2 2021-08-25 [1] CRAN (R 4.2.0)\n#> httr 1.4.4 2022-08-17 [1] CRAN (R 4.2.0)\n#> jquerylib 0.1.4 2021-04-26 [1] CRAN (R 4.2.0)\n#> jsonlite 1.8.3 2022-10-21 [1] CRAN (R 4.2.0)\n#> knitr 1.39 2022-04-26 [1] CRAN (R 4.2.0)\n#> lifecycle 1.0.3 2022-10-07 [1] CRAN (R 4.2.0)\n#> lubridate 1.8.0 2021-10-07 [1] CRAN (R 4.2.0)\n#> magrittr 2.0.3 2022-03-30 [1] CRAN (R 4.2.0)\n#> memoise 2.0.1 2021-11-26 [1] CRAN (R 4.2.0)\n#> modelr 0.1.8 2020-05-19 [1] CRAN (R 4.2.0)\n#> munsell 0.5.0 2018-06-12 [1] CRAN (R 4.2.0)\n#> pillar 1.8.1 2022-08-19 [1] CRAN (R 4.2.0)\n#> pkgconfig 2.0.3 2019-09-22 [1] CRAN (R 4.2.0)\n#> prettycode 1.1.0 2019-12-16 [1] CRAN (R 4.2.0)\n#> purrr * 0.3.5 2022-10-06 [1] CRAN (R 4.2.0)\n#> R.cache 0.15.0 2021-04-30 [1] CRAN (R 4.2.0)\n#> R.methodsS3 1.8.2 2022-06-13 [1] CRAN (R 4.2.0)\n#> R.oo 1.25.0 2022-06-12 [1] CRAN (R 4.2.0)\n#> R.utils 2.12.0 2022-06-28 [1] CRAN (R 4.2.0)\n#> R6 2.5.1 2021-08-19 [1] CRAN (R 4.2.0)\n#> readr * 2.1.2 2022-01-30 [1] CRAN (R 4.2.0)\n#> readxl 1.4.0 2022-03-28 [1] CRAN (R 4.2.0)\n#> reprex 2.0.1 2021-08-05 [1] CRAN (R 4.2.0)\n#> rlang 1.0.6 2022-09-24 [1] CRAN (R 4.2.0)\n#> rmarkdown 2.14 2022-04-25 [1] CRAN (R 4.2.0)\n#> rprojroot 2.0.3 2022-04-02 [1] CRAN (R 4.2.0)\n#> rstudioapi 0.13 2020-11-12 [1] CRAN (R 4.2.0)\n#> rvest 1.0.2 2021-10-16 [1] CRAN (R 4.2.0)\n#> sass 0.4.1 2022-03-23 [1] CRAN (R 4.2.0)\n#> scales 1.2.0 2022-04-13 [1] CRAN (R 4.2.0)\n#> sessioninfo 1.2.2 2021-12-06 [1] CRAN (R 4.2.0)\n#> stringi 1.7.8 2022-07-11 [1] CRAN (R 4.2.0)\n#> stringr * 1.4.1 2022-08-20 [1] CRAN (R 4.2.0)\n#> styler 1.7.0 2022-03-13 [1] CRAN (R 4.2.0)\n#> tibble * 3.1.8 2022-07-22 [1] CRAN (R 4.2.0)\n#> tidyr * 1.2.0 2022-02-01 [1] CRAN (R 4.2.0)\n#> tidyselect 1.2.0 2022-10-10 [1] CRAN (R 4.2.0)\n#> tidyverse * 1.3.1 2021-04-15 [1] CRAN (R 4.2.0)\n#> tzdb 0.3.0 2022-03-28 [1] CRAN (R 4.2.0)\n#> utf8 1.2.2 2021-07-24 [1] CRAN (R 4.2.0)\n#> vctrs 0.4.1 2022-04-13 [1] CRAN (R 4.2.0)\n#> withr 2.5.0 2022-03-03 [1] CRAN (R 4.2.0)\n#> xfun 0.32 2022-08-10 [1] CRAN (R 4.2.0)\n#> xml2 1.3.3 2021-11-30 [1] CRAN (R 4.2.0)\n#> yaml 2.3.6 2022-10-18 [1] CRAN (R 4.2.0)\n#> \n#> [1] /Library/Frameworks/R.framework/Versions/4.2/Resources/library\n#> \n#> ────────────────────────────────────────────────────────────────────\n\nUse renv to manage package dependencies\nThe renv package allows you to have a separate set of R packages for each project. It also can record and restore the set of R pacakges used in a project. This is very helpful when you need to return to a project months (or years) later and want to have the same set of packages. It also makes it easier to share your packages with collaborators.\nSee also:\nconda for managing various command line programs (python, R, c, etc.)\ndocker for generating a fully reproducible operating system environment.\nBenchmarking, with microbenchmark and profvis\n\n\n# example, compare base and readr csv reading functions\npath_to_file <- here(\"data/class3/dmel_peptides_lifecycle.csv.gz\")\n\nres <- microbenchmark::microbenchmark(\n base = read.csv(path_to_file),\n readr = readr::read_csv(path_to_file),\n times = 5\n)\nprint(res, signif = 2)\n\n#> Unit: milliseconds\n#> expr min lq mean median uq max neval\n#> base 3300 3500 3500 3600 3600 3700 5\n#> readr 280 290 370 310 410 560 5\n\n\n\n# example, looking at each step of a script\nlibrary(tidyverse)\np <- profvis::profvis({\n\n df <- readr::read_csv(path_to_file)\n df <- df %>%\n filter(e02_4 < 50) %>%\n mutate(sequence = str_to_lower(Sequence))\n})\np\n\n\n\nDebugging R code\nR has a debugger built in. You can debug a function:\ne.g.:\n\n\ndebug(read_csv) # set a function to debug\nread_csv(path_to_file) # will enter debug mode\nundebug(read_csv)\n\n\nRstudio has great support for debugging functions in Rscripts or in packages:\nhttps://support.posit.co/hc/en-us/articles/200713843-Debugging-R-code-with-the-RStudio-IDE\nLook at the call stack with traceback()\n\n\ncool_function <- function(x) {\n \n internal_function <- function(y) {\n \n hard_to_find <- function(z) {\n \"doesn't work here\" + 10\n }\n \n hard_to_find()\n }\n internal_function()\n}\n\ncool_function(1)\ntraceback()\n\n\n\nBuilding your own R package\nIt’s surprisingly easy, particularly with Rstudio, to write your own R package to store your code. Putting your code in a package makes it much easier to debug, document, add tests, and distribute your code.\nhttps://r-pkgs.org/\nhttps://hilaryparker.com/2014/04/29/writing-an-r-package-from-scratch/\nshiny, interactive web app for data exploration\nMaking an interactive interface to data and plotting is easy in R. Examples and corresponding code can be found at https://shiny.rstudio.com/gallery/.\nParsing specific types of formats:\nThere are generally packages built to read and write from most formats. Some examples:\nJSON\nCheck out jsonlite\n\n\nlibrary(jsonlite)\njson_file <- \"http://api.worldbank.org/country?per_page=10®ion=OED&lendingtype=LNX&format=json\"\nworldbank_data <- fromJSON(json_file, flatten=TRUE)\nworldbank_data\n\n#> [[1]]\n#> [[1]]$page\n#> [1] 1\n#> \n#> [[1]]$pages\n#> [1] 4\n#> \n#> [[1]]$per_page\n#> [1] \"10\"\n#> \n#> [[1]]$total\n#> [1] 32\n#> \n#> \n#> [[2]]\n#> id iso2Code name capitalCity longitude latitude region.id\n#> 1 AUS AU Australia Canberra 149.129 -35.282 EAS\n#> 2 AUT AT Austria Vienna 16.3798 48.2201 ECS\n#> 3 BEL BE Belgium Brussels 4.36761 50.8371 ECS\n#> 4 CAN CA Canada Ottawa -75.6919 45.4215 NAC\n#> 5 CHE CH Switzerland Bern 7.44821 46.948 ECS\n#> 6 CZE CZ Czechia Prague 14.4205 50.0878 ECS\n#> 7 DEU DE Germany Berlin 13.4115 52.5235 ECS\n#> 8 DNK DK Denmark Copenhagen 12.5681 55.6763 ECS\n#> 9 ESP ES Spain Madrid -3.70327 40.4167 ECS\n#> 10 EST EE Estonia Tallinn 24.7586 59.4392 ECS\n#> region.iso2code region.value adminregion.id\n#> 1 Z4 East Asia & Pacific \n#> 2 Z7 Europe & Central Asia \n#> 3 Z7 Europe & Central Asia \n#> 4 XU North America \n#> 5 Z7 Europe & Central Asia \n#> 6 Z7 Europe & Central Asia \n#> 7 Z7 Europe & Central Asia \n#> 8 Z7 Europe & Central Asia \n#> 9 Z7 Europe & Central Asia \n#> 10 Z7 Europe & Central Asia \n#> adminregion.iso2code adminregion.value incomeLevel.id\n#> 1 HIC\n#> 2 HIC\n#> 3 HIC\n#> 4 HIC\n#> 5 HIC\n#> 6 HIC\n#> 7 HIC\n#> 8 HIC\n#> 9 HIC\n#> 10 HIC\n#> incomeLevel.iso2code incomeLevel.value lendingType.id\n#> 1 XD High income LNX\n#> 2 XD High income LNX\n#> 3 XD High income LNX\n#> 4 XD High income LNX\n#> 5 XD High income LNX\n#> 6 XD High income LNX\n#> 7 XD High income LNX\n#> 8 XD High income LNX\n#> 9 XD High income LNX\n#> 10 XD High income LNX\n#> lendingType.iso2code lendingType.value\n#> 1 XX Not classified\n#> 2 XX Not classified\n#> 3 XX Not classified\n#> 4 XX Not classified\n#> 5 XX Not classified\n#> 6 XX Not classified\n#> 7 XX Not classified\n#> 8 XX Not classified\n#> 9 XX Not classified\n#> 10 XX Not classified\n\nGenomics data (fasta, fastq, vcf, bam, bed, bigwig)\nCheck out rtracklayer and Rsamtools:\ne.g. read a FASTA file into R:\n\n\nlibrary(Rsamtools)\n\n# get path to test file included in a package\nfasta_file <- system.file('extdata', 'ce2dict1.fa', package = 'Rsamtools')\nscanFa(fasta_file)\n\n#> DNAStringSet object of length 5:\n#> width seq names \n#> [1] 18 GCGAAACTAGGAGAGGCT pattern01\n#> [2] 25 CTGTTAGCTAATTTTAAAAATAAAT pattern02\n#> [3] 24 ACTACCACCCAAATTTAGATATTC pattern03\n#> [4] 24 AAATTTTTTTTGTTGCAAATTTGA pattern04\n#> [5] 25 TCTTCTTGGCTTTGGTGGTACTTTT pattern05\n\nUsing R on the command-line\nThe command line can be accessed via the Terminal app on macOS, or using the windows subsystem for linux (WSL).\nThere command line is a place where you can run executable programs (a C, python, R, or whatever). It’s what using a computer looked like before the existence of a Graphical User Interface. It is impossible to conduct data analysis without gaining some experience with working on the command line.\nR is an executable, and we can pull up an R console using:\nR\nIn Rmarkdown you can also include other languages including bash (which is a common language of the command line).You need to change the r to bash in the code chunk (or to python or other languages).\nYou can run simple commands by using Rscript or R with the -e option.\n\nR -e \"print('hello')\"\n#> \n#> R version 4.2.0 (2022-04-22) -- \"Vigorous Calisthenics\"\n#> Copyright (C) 2022 The R Foundation for Statistical Computing\n#> Platform: x86_64-apple-darwin17.0 (64-bit)\n#> \n#> R is free software and comes with ABSOLUTELY NO WARRANTY.\n#> You are welcome to redistribute it under certain conditions.\n#> Type 'license()' or 'licence()' for distribution details.\n#> \n#> Natural language support but running in an English locale\n#> \n#> R is a collaborative project with many contributors.\n#> Type 'contributors()' for more information and\n#> 'citation()' on how to cite R or R packages in publications.\n#> \n#> Type 'demo()' for some demos, 'help()' for on-line help, or\n#> 'help.start()' for an HTML browser interface to help.\n#> Type 'q()' to quit R.\n#> \n#> > print('hello')\n#> [1] \"hello\"\n#> > \n#> >\n\n\nRscript -e \"print('hello')\"\n#> [1] \"hello\"\n\nAlternatively you can write a R script, which can be then called from Rscript. For example if we wrote an R script called cool_function.R.\n#!/usr/bin/env Rscript # allows calling with ./cool_function.R if executable\n\nargs = commandArgs(trailingOnly=TRUE) # collect command line arguments \nprint(args) # args is a list e.g. argument1 argument2...\nWe could call on the command line:\nRscript path/to/cool_function.R argument1 argument2 ...\n#or\npath/to/cool_function.R argument1 argument2 ...\nGit and Github\nFrom https://jmcglone.com/guides/github-pages/Git is a command line tool for version control, which allows us to:\nrolling back code to a previous state if needed\nbranched development, tackling individual issues/tasks\ncollaboration\nFrom https://blog.programster.org/git-workflowsGit was first created by Linus Torvalds for coordinating development of Linux. Read this guide for Getting started\n, checkout this interactive guide and check out this Tutorial written from an R data analyst perspective.\n\n# for bioinformatics, get comfortable with command line too\n\nls\ngit status # list changes to tracked files\ngit blame resources.Rmd # see who contributed\ngit commit -m \"added something cool\" # save state\ngit push # push git to a git repository (e.g. github)\ngit pull # pull changes from git repository\n\nThis can be handled by Rstudio as well (new tab next to Connections and Build)\nPut your code on GitHub\nAs you write more code, especially as functions and script pipelines, hosting and documenting them on GitHub is great way to make them portable and searchable. Even the free tier of GitHub accounts now has private repositories (repo).\nIf you have any interest in a career in data science/informatics, GitHub is also a common showcase of what (and how well/often) you can code. After some accumulation of code, definitely put your GitHub link on your CV/resume.\nCheck out the quickstart from github:\nhttps://docs.github.com/en/get-started/quickstart/hello-world\nExample repos (RBI)\nthis class\nvalr\nr-source # mirror of R source code\nreadr\nAsking for help with other packages on GitHub\nEvery package should include README, installation instructions, and a maintained issues page where questions and bugs can be reported and addressed. Example: readr GitHub page Don’t be afraid to file new issues, but very often your problems are already answered in the closed section.\nFinding useful packages\nIn most cases, what you need is already made into well-documented packages, and you don’t have to reinvent the wheel (but sometimes you should?). Depending on where the package is curated, installation is different. Some examples below:\nGviz - visualize gene model\neuler - making custom euler/venn diagrams\nemo - inserting emojis into Rmd\n\n\n# BiocManager::install(\"Gviz\") # from bioconductor\nvignette(\"Gviz\")\n\n# install.packages(\"eulerr\") # from CRAN\nplot(eulerr::euler(list(set1 = c(\"geneA\", \"geneB\", \"geneC\"), \n set2 = c(\"geneC\", \"geneD\"))))\n\n\n\n\n# devtools::install_github(\"hadley/emo\") # from github\nemo::ji(\"smile\")\n\n#> 😄\n\nBioconductor\n\n2,000+ R packages dedicated to bioinformatics. Included a coherent framework of data structures (e.g. SummarizedExperiment) built by dedicated Core members. Also includs many annotation and experimental datasets built into R packages and objects (See AnnotationHub and ExperimentHub)\nhttps://bioconductor.org/\nUse BiocManager::install() to install these packages\nRNA-seq workflow\nOrchestrating single cell analysis\nFinding help online\n\nThe R studio community forums are a great resource for asking questions about tidyverse related packages.\nStackOverflow provides user-contributed questions and answers on a variety of topics.\nFor help with bioconductor packages, visit the Bioc support page\nFind out if others are having similar issues by searching the issue on the package GitHub page.\nCheat sheets\nRstudio links to common ones here: Help -> Cheatsheets. More are hosted online, such as for regular expressions.\nUseful to keep your own stash too.\nOffline help\nThe RBI fellows hold standing office hours on Thursdays over zoom. We are happy to help out with coding and RNA/DNA-related informatics questions. Send us an email to schedule a time (rbi.fellows@cuanschutz.edu).\nSometimes code is just broken\nNo one writes perfect code. Developers often expect that there will be bugs in their code. If you suspect bugs or mishandled edge cases, go to the package GitHub and search the issues section to see if the problem has been reported or fixed. If not, submit an issue that describes the problem.\nThe reprex package makes it easy to\nproduce well-formatted reproducible examples that demonstrate the problem. Often developers will be thankful for your help with making their software better.\nAdditional Resources\nGeneral/Data science\nIntroduction to Data Science\nStats in R\nR programming for data science\nR for data science\nAdvanced R\nfasteR base R tutorial\nGenomics\nPH525x series - Biomedical Data Science\nBioinformatics Data Skills\nBiostar Handbook\nGenomics Workshop\nA meta-list of R resources\nhttps://github.com/iamericfletcher/awesome-r-learning-resources\nWriting high-performance R functions with R + C++\nhttp://adv-r.had.co.nz/Rcpp.html\nhttps://dirk.eddelbuettel.com/code/rcpp.html\n\n\n\n", "preview": {}, - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -200,7 +217,7 @@ "categories": [], "contents": "\nThe Rmarkdown for this class is on github\n\n\n# conditionally download all of the files used in rmarkdown from github \nsource(\"https://github.com/rnabioco/bmsc-7810-pbda/raw/main/_posts/2022-12-05-class-10-programming-in-r-part-1/download-files.R\")\n\n\n\nWhat is a function?\nAs an analyst you will eventually find yourself in the position of wanting to reuse a block of code. There are two general ways to do this:\ncopy-and-paste\nwrite a function\nA function is essentially a block of code that you’ve given a name and saved for later. Functions have several advantages:\nThey make your code easier to read\nThey reduce the chance of mistakes from repeated copying and pasting\nThey make it easier to adapt your code for different requirements\nFurther reading\nR for Data Science by Garrett Grolemund and Hadley Wickham\nAdvanced R by Hadley Wickham\n\n\nlibrary(tidyverse)\n\n\n\n\n# An example: you want to rescale a numeric vector so all values are between 0 and 1\na <- rnorm(n = 10)\na\n\n#> [1] -1.1292023 0.3885605 -1.6844503 0.7907017 -0.4599947 1.3780085\n#> [7] 0.2672932 -0.9226258 -1.9090162 0.7799124\n\nrng <- range(a)\n(a - rng[1]) / (rng[2] - rng[1])\n\n#> [1] 0.23724005 0.69898370 0.06831889 0.82132570 0.44083073 1.00000000\n#> [7] 0.66209099 0.30008609 0.00000000 0.81804331\n\n\n\n# What if we want to repeat this on other vectors?\n# One way is to copy and paste\nb <- rnorm(n = 10)\nc <- rnorm(n = 10)\nrng <- range(b)\nnew_b <- (b - rng[1]) / (rng[2] - rng[1])\n\nrng <- range(c)\nnew_c <- (c - rng[1]) / (rng[2] - rng[1])\n# A better way is to write a function...\n\n\n\n\nHow to write a function\nThere are three general steps for writing functions:\nPick a name\nIdentify the inputs\nAdd code to the body\n\n\n# Lets write a function to rescale a numeric vector\nrescale_vec <- function(x) {\n \n rng <- range(x)\n (x - rng[1]) / (rng[2] - rng[1])\n}\nrescale_vec(b)\nrescale_vec(c)\n\n\n\nWrite functions for the following bits of code\n\n\n# function 1\nx / sum(x)\n\n# function 2\n(x + y) / z\n\n# function 3\nsqrt(sum((x - mean(x))^2) / (length(x) - 1))\n\n\n\nShow answer\n\n\ncalc_sd <- function(x) {\n sqrt(sum((x - mean(x))^2) / (length(x) - 1))\n}\n\ncalc_sd <- function(x) {\n l <- length(x) - 1\n m <- mean(x)\n v <- sum((x - m)^2) / l\n sqrt(v)\n}\n\n\n\nThe function execution environment\nWhen running a function an execution environment is created, which is separate from the global environment\nThe execution environment contains objects created within the function\nWhen R searches for an object referenced by a function, the execution environment takes precedence\nIf an object is not found in the function environment, R will search in the global environment\n\nCan objects present in the global environment be referenced from within a function?\n\n\n# Earlier we saved a numeric vector \"a\"\na\n\n#> [1] -1.1292023 0.3885605 -1.6844503 0.7907017 -0.4599947 1.3780085\n#> [7] 0.2672932 -0.9226258 -1.9090162 0.7799124\n\nsum_nums <- function(x) {\n x + a\n}\n# Yes!\nsum_nums(10)\n\n#> [1] 8.870798 10.388560 8.315550 10.790702 9.540005 11.378009\n#> [7] 10.267293 9.077374 8.090984 10.779912\n\n\nCan code executed within a function modify an object present in the global environment?\n\n\nsum_nums <- function(x) {\n a <- x + a\n}\n# When we run sum_nums(), will this overwrite our original vector?\nsum_nums(10)\n# No! (not when using the '<-' assignment operator)\na\n\n#> [1] -1.1292023 0.3885605 -1.6844503 0.7907017 -0.4599947 1.3780085\n#> [7] 0.2672932 -0.9226258 -1.9090162 0.7799124\n\n\n\nA more relevant example\nThe brauer_gene_exp data contains a data set from a manuscript describing how gene expression changes in yeast under several nutrient limitation conditions. We’ll use this data to illustrate the utility and the power of functions.\nUsing the Brauer data lets create a scatter plot comparing growth rate vs expression for the gene YDL104C. Use facet_wrap() to create a separate plot for each nutrient.\n\n\nbrauer_gene_exp <- read_csv(\"data/brauer_gene_exp.csv.gz\")\n\n\n\n\n\n\nWhat if you want to create this plot for other genes? Write a function the takes a data.frame and systematic_name as inputs and creates scatter plots for each nutrient\n\n# Fill in the function body\n# You can include default values for your arguments\nplot_expr <- function(input, sys_name = \"YNL049C\") {\n \n ????\n \n}\n\n\nShow answer\n\n\nplot_expr <- function(input, sys_name = \"YNL049C\") {\n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = 2) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n\n\n\n\np <- plot_expr(\n input = brauer_gene_exp, \n sys_name = \"YDL104C\"\n)\n# You can also use the %>% pipe with your custom functions\np <- brauer_gene_exp %>%\n plot_expr(sys_name = \"YDL104C\")\np\n\n\n\n\nModify our plotting function to add the gene name as the plot title and the molecular function (MF) as a subtitle\n\nShow answer\n\n\nplot_expr <- function(input, sys_name) {\n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n plot_title <- gg_data$name[1]\n plot_sub <- gg_data$MF[1]\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = 2) +\n labs(title = plot_title, subtitle = plot_sub) +\n ggtitle(plot_title) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n\n\n\n\nbrauer_gene_exp %>%\n plot_expr(\"YDL104C\")\n\n\n\n\n\nCopy-on-modify semantics\nAs you’ve seen objects that are passed to a function are not modified within the function by default. Intuitively you can think of each object being copied within the function environment to avoid modification of the original. However this would be memory inefficient and slow approach, as copying multiple large objects takes time and space.\nInstead R adopts a “copy-on-modify” approach with objects. Objects are only copied when it is necessary. The same is true of objects outside of functions.\n\n\nchange_to_char <- function(large_object) {\n # large object is not a copy, but a reference\n large_object\n \n # now a new copy of large_object is made\n large_object <- as.character(large_object)\n large_object\n}\n\nmat <- matrix(1:100, nrow = 10)\n# not copied\na <- mat\n\n# mat not copied yet\nmat[1:5, 1:5]\n\n# now a copy is made\nmat2 <- as.character(mat)\nmat2 <- as.data.frame(mat)\n\n\nConditional statements\nif statements allow you to execute code depending on defined conditions.\n\nif (condition) {\n code executed when condition is TRUE\n \n} else {\n code executed when condition is FALSE\n}\n\nR has a set of operators that can be used to write conditional statements\nOperator\nDescription\n<\nless than\n<=\nless or equal\n>\ngreater than\n>=\ngreater or equal\n==\nequal\n!=\nnot equal\n!x\nnot x\nx | y\nx or y (returns a vector of logicals)\nx || y\nx or y (returns single TRUE or FALSE)\nx & y\nx and y (returns a vector of logicals)\nx && y\nx and y (returns single TRUE or FALSE)\nx %in% y\nx is present in y\n\nAdd an if statement to our plotting function to account for a missing gene name\n\n\nplot_expr <- function(input, sys_name) {\n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n plot_title <- gg_data$name[1]\n plot_sub <- gg_data$MF[1]\n \n ????\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = 2) +\n labs(title = plot_title, subtitle = plot_sub) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n\n\n\nShow answer\n\n\nplot_expr <- function(input, sys_name) {\n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n plot_title <- gg_data$name[1]\n plot_sub <- gg_data$MF[1]\n \n if (is.na(plot_title)) {\n plot_title <- sys_name\n }\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = 2) +\n labs(title = plot_title, subtitle = plot_sub) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n\n\n\n\nbrauer_gene_exp %>%\n plot_expr(\"YNL095C\")\n\n\n\n\nConditional statements can be linked together\n\n# Using 'else if'\nif (condition_1) {\n executed when condition_1 is TRUE\n \n} else if (condition_2) {\n executed when condition_1 is FALSE and condition_2 is TRUE\n \n} else {\n executed when condition_1 and condition_2 are FALSE\n}\n# The 'and' operator\nif (condition_1 && condition_2) {\n executed when condition_1 and condition_2 are TRUE\n \n} else {\n executed when condition_1 or condition_2 are FALSE\n}\n# The 'or' operator\nif (condition_1 || condition_2) {\n executed when condition_1 or condition_2 are TRUE\n \n} else {\n executed when condition_1 and condition_2 are FALSE\n}\n\n\n\nMessages, warnings, and errors\nstop() warning(), message(), and stopifnot() are commonly used functions in R for reporting information and/or stopping execution based on a condition.\n\n\nstop(\"information about error to user, stops execution\")\nwarning(\"information about warning to user, does not stop execution\")\nmessage(\"information that is not an error or warning, does not stop execution\")\nstopifnot(2 + 2 != 4) # shortcut for if(condition is FALSE) stop()\n\n\nSee also tryCatch() for “catching” errors and performing alternative actions.\nChecking inputs\nWhen writing functions it can be useful to check input values to make sure they are valid. Lets modify our plotting function to check that sys_name is a string.\nis.character()\nis.numeric()\nis.logical()\nis.factor()\n\n\nplot_expr <- function(input, sys_name) {\n \n if (!is.character(sys_name)) {\n stop(\"sys_name must be a string!\")\n }\n \n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n plot_title <- gg_data$name[1]\n plot_sub <- gg_data$MF[1]\n \n if (is.na(plot_title)) {\n plot_title <- sys_name\n }\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = 2) +\n labs(title = plot_title, subtitle = plot_sub) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\nbrauer_gene_exp %>%\n plot_expr(\"YDL104C\")\n\n\n\n\nModify our plotting function to check that sys_name is present in the input. Hint: try the %in% operator\n\nplot_expr <- function(input, sys_name) {\n \n if (!is.character(sys_name)) {\n stop(\"sys_name must be a string!\")\n }\n \n if ( ???? ) {\n stop( ???? )\n }\n \n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n plot_title <- gg_data$name[1]\n plot_sub <- gg_data$MF[1]\n \n if (is.na(plot_title) ){\n plot_title <- sys_name\n }\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = 2) +\n labs(title = plot_title, subtitle = plot_sub) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n\n\nShow answer\n\n\n\n\n\nPassing arguments with the ellipsis (…)\nThe ellipsis allows a function to take an arbitrary number of arguments, which can then be passed to an inner function. This is nice when you have an inner function that has a lot of useful arguments. Lets first try this with our simple rescale_vec() function.\n\n\nrescale_vec <- function(x, ...) {\n rng <- range(x, ...)\n (x - rng[1]) / (rng[2] - rng[1])\n}\nrescale_vec(a)\n\n#> [1] 0.23724005 0.69898370 0.06831889 0.82132570 0.44083073 1.00000000\n#> [7] 0.66209099 0.30008609 0.00000000 0.81804331\n\na[1] <- NA\nrescale_vec(a, na.rm = T)\n\n#> [1] NA 0.69898370 0.06831889 0.82132570 0.44083073 1.00000000\n#> [7] 0.66209099 0.30008609 0.00000000 0.81804331\n\n\nModify our plotting function so the user can change the point size, shape, and alpha\n\n\n# A cumbersome way\nplot_expr <- function(input, sys_name, pt_size = 2, pt_shape = 1, pt_alpha = 1) {\n input %>%\n filter(systematic_name == sys_name) %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(size = pt_size, shape = pt_shape, alpha = pt_alpha) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n# With the ellipsis\nplot_expr <- function(input, sys_name, ...) {\n input %>%\n filter(systematic_name == sys_name) %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(...) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n# Now we can easily change the point size and shape\nplot_expr(\n input = brauer_gene_exp,\n sys_name = \"YDL104C\",\n size = 5,\n shape = 2,\n alpha = 0.75\n)\n\n\n\n\n\nSaving your functions for later\nA good way to save commonly used functions is to keep them in a separate R script. You can load your functions using the source() command.\n\n\nsource(\"path/to/my_functions.R\")\n\n\n\nShow session info\n\n\nsessionInfo()\n\n#> R version 4.2.0 (2022-04-22)\n#> Platform: x86_64-apple-darwin17.0 (64-bit)\n#> Running under: macOS Big Sur/Monterey 10.16\n#> \n#> Matrix products: default\n#> BLAS: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRblas.0.dylib\n#> LAPACK: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRlapack.dylib\n#> \n#> locale:\n#> [1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n#> \n#> attached base packages:\n#> [1] stats graphics grDevices utils datasets methods \n#> [7] base \n#> \n#> other attached packages:\n#> [1] forcats_0.5.1 stringr_1.4.1 dplyr_1.0.10 purrr_0.3.5 \n#> [5] readr_2.1.2 tidyr_1.2.0 tibble_3.1.8 ggplot2_3.3.6 \n#> [9] tidyverse_1.3.1\n#> \n#> loaded via a namespace (and not attached):\n#> [1] lubridate_1.8.0 assertthat_0.2.1 digest_0.6.30 \n#> [4] utf8_1.2.2 R6_2.5.1 cellranger_1.1.0\n#> [7] backports_1.4.1 reprex_2.0.1 evaluate_0.16 \n#> [10] highr_0.9 httr_1.4.4 pillar_1.8.1 \n#> [13] rlang_1.0.6 readxl_1.4.0 rstudioapi_0.13 \n#> [16] jquerylib_0.1.4 rmarkdown_2.14 labeling_0.4.2 \n#> [19] bit_4.0.4 munsell_0.5.0 broom_0.8.0 \n#> [22] compiler_4.2.0 modelr_0.1.8 xfun_0.32 \n#> [25] pkgconfig_2.0.3 htmltools_0.5.2 downlit_0.4.2 \n#> [28] tidyselect_1.2.0 fansi_1.0.3 crayon_1.5.2 \n#> [31] tzdb_0.3.0 dbplyr_2.2.1 withr_2.5.0 \n#> [34] grid_4.2.0 jsonlite_1.8.3 gtable_0.3.0 \n#> [37] lifecycle_1.0.3 DBI_1.1.3 magrittr_2.0.3 \n#> [40] scales_1.2.0 cli_3.4.1 stringi_1.7.8 \n#> [43] vroom_1.5.7 cachem_1.0.6 farver_2.1.0 \n#> [46] fs_1.5.2 xml2_1.3.3 bslib_0.3.1 \n#> [49] ellipsis_0.3.2 generics_0.1.3 vctrs_0.4.1 \n#> [52] distill_1.5 tools_4.2.0 bit64_4.0.5 \n#> [55] glue_1.6.2 hms_1.1.2 parallel_4.2.0 \n#> [58] fastmap_1.1.0 yaml_2.3.6 colorspace_2.0-3\n#> [61] rvest_1.0.2 memoise_2.0.1 knitr_1.39 \n#> [64] haven_2.5.0 sass_0.4.1\n\n\n\n\n", "preview": "posts/2022-12-05-class-10-programming-in-r-part-1/programming-in-r-pt1_files/figure-html5/unnamed-chunk-10-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -217,7 +234,7 @@ "categories": [], "contents": "\n\nContents\nIntroduction\nfor loops\nUsing the Brauer data\nA note on vectorization\nIntroduction to the apply family of functions.\niterating over matrices\nIterating over multiple vectors\nUsing the map() family of functions from the purrr package\nAdditional resources related to Programming in R\n\nThe Rmarkdown for this class is on github\n\n\n# conditionally download all of the files used in rmarkdown from github \nsource(\"https://github.com/rnabioco/bmsc-7810-pbda/raw/main/_posts/2022-12-05-class-10-programming-in-r-part-1/download-files.R\")\n\n\n\nIntroduction\nAs discussed in the previous class, you should try to limit duplication in your code. One way to do this is by writing functions, another way is through iteration. Reducing code duplication has several benefits:\nYour code easier to read\nYou reduce the chance of mistakes from repeated copying and pasting\nIt is easier to adapt your code for different requirements\n\n\nlibrary(tidyverse)\nlibrary(patchwork)\n\n\n\n\nfor loops\nfor loops allow you to run the same code block repeatedly without copying and pasting.\n\n\nfor(i in 1:4){\n message(\"printing \", i, \" is fun\")\n}\n\nvalues <- c(\"A\", \"B\", \"C\", \"D\")\nfor(val in values){\n message(val)\n}\n\n\nWhen iterating over a vector, usually it is most useful to iterate over the index of each element (aka the position in the vector), rather than the values themselves.\n\n\nfor(i in 1:length(values)){\n val <- values[i]\n message(\"index = \", i, \" value = \", val)\n}\n\n\nWe will generally want to store the output generated in the for loop. A common paradigm is\nto preallocate a place to store the output. This is a faster approach than growing the output at each iteration (for more detail see this R-bloggers post).\nWe can generate vectors (and lists) of a given type and length using the vector() function.\n\n\nn <- length(values)\n\n# make an empty vector of same length as values (4)\noutput <- vector(mode = \"character\", length = n)\noutput\n\n#> [1] \"\" \"\" \"\" \"\"\n\n\nfor(i in 1:n){ \n # get value at position i\n val <- values[i]\n # assign value to output character vector at position i\n output[i] <- tolower(val)\n}\noutput\n\n#> [1] \"a\" \"b\" \"c\" \"d\"\n\n\n\n# It's helpful to think about what happens during each cycle of the loop\noutput[1] <- tolower(\"A\") # i = 1\noutput[2] <- tolower(\"B\") # i = 2\noutput[3] <- tolower(\"C\") # i = 3\noutput[4] <- tolower(\"D\") # i = 4\n\n\n\n\nLets use rnorm() to create a list of 5 vectors with different values for ‘mean’\n\n\n# One way to do this is by copying and pasting\nvec_in <- c(1, 50, 20, 5, 70) # input\n\nout <- vector(\"list\", length(vec_in)) # output\n\nout[[1]] <- rnorm(n = 10, mean = vec_in[1]) \nout[[2]] <- rnorm(n = 10, mean = vec_in[2])\nout[[3]] <- rnorm(n = 10, mean = vec_in[3])\nout[[4]] <- rnorm(n = 10, mean = vec_in[4])\nout[[5]] <- rnorm(n = 10, mean = vec_in[5])\nout\n\n#> [[1]]\n#> [1] -0.78152016 2.10160745 -0.11415327 1.62145591 1.47734308\n#> [6] 0.04282456 0.43233384 -0.36207649 1.34259857 1.33220909\n#> \n#> [[2]]\n#> [1] 49.79548 50.69003 48.40643 49.79333 50.25014 50.27296 48.15528\n#> [8] 49.82848 50.95952 48.85871\n#> \n#> [[3]]\n#> [1] 19.50403 20.24307 22.10350 19.79465 20.07914 17.35373 21.81705\n#> [8] 20.52651 20.08495 19.96808\n#> \n#> [[4]]\n#> [1] 4.731921 5.349496 4.359502 6.216023 4.028793 5.085959 5.026359\n#> [8] 6.584741 4.507993 6.139226\n#> \n#> [[5]]\n#> [1] 70.35925 69.22108 69.32543 69.93716 69.45157 68.92867 70.90439\n#> [8] 71.56308 71.55584 67.76194\n\n\n\n# Use a for loop to reduce code duplication\nvec_in <- c(1, 50, 20, 5, 70) # input\n\nout <- vector(\"list\", length(vec_in)) # output\n\nfor (i in 1:length(vec_in)) { # sequence\n \n out[[i]] <- rnorm(n = 10, mean = vec_in[i]) # body\n \n}\n\n\n\nWrite a for loop that uses rnorm() to create 3 vectors of different lengths. Store the vectors in a list. Use mean = 0 and sd = 1 (the default). \n\nShow answer\n\n\nvec_in <- c(5, 10, 2) # input\nn <- length(vec_in)\nout <- vector(\"list\", n) # output\n\nfor (i in 1:length(vec_in)) { # sequence\n out[[i]] <- rnorm(n = vec_in[i])\n}\n\nout\n\n#> [[1]]\n#> [1] -0.70441511 -1.29016176 -0.73045302 -0.38346352 0.02470951\n#> \n#> [[2]]\n#> [1] -0.2454521 -1.6694252 0.3136690 -0.1473203 1.6922765 0.4013761\n#> [7] 1.5953518 -0.3535064 1.0189067 1.8831290\n#> \n#> [[3]]\n#> [1] 0.8518765 -0.6266713\n\n\nSo far we have used 1:length(x) to specify the sequence to iterate over. A better alternative is using seq_along(x) instead of 1:length(x) . This guards against errors when an empty vector is passed to 1:length(x).\n\n\n# seq_along() mimics 1:length() for non-empty vectors\nvec_in <- c(5, 10, 2)\n\n1:length(vec_in)\n\n#> [1] 1 2 3\n\n\nseq_along(vec_in)\n\n#> [1] 1 2 3\n\n\n# seq_along() correctly handles empty vectors\nemp_vec <- vector(\"numeric\", 0)\n\n1:length(emp_vec)\n\n#> [1] 1 0\n\n\nseq_along(emp_vec)\n\n#> integer(0)\n\n\n\nUsing the Brauer data\nUsing the Brauer gene expression data lets create a figure showing the growth rate vs expression for four genes\n\n\nbrauer_gene_exp <- read_csv(\"data/brauer_gene_exp.csv.gz\")\n\n\n\n\n# This is the function we wrote in class-10\nplot_expr <- function(input, sys_name, ...) {\n \n gg_data <- input %>%\n filter(systematic_name == sys_name)\n \n plot_title <- gg_data$name[1]\n plot_sub <- gg_data$MF[1]\n \n gg_data %>%\n ggplot(aes(rate, expression, color = nutrient)) +\n geom_point(...) +\n labs(title = plot_title, subtitle = plot_sub) +\n facet_wrap(~ nutrient) +\n theme(legend.position = \"none\")\n}\n\n\nLets try this with the copy-and-paste method, storing the plots in a list.\n\n\nvec_in <- c(\"YDL104C\", \"YLR115W\", \"YMR183C\", \"YML017W\") # input\n\nout <- vector(\"list\", length(vec_in)) # output \n\nout[[1]] <- plot_expr(brauer_gene_exp, sys_name = vec_in[1]) \nout[[2]] <- plot_expr(brauer_gene_exp, sys_name = vec_in[2])\nout[[3]] <- plot_expr(brauer_gene_exp, sys_name = vec_in[3])\nout[[4]] <- plot_expr(brauer_gene_exp, sys_name = vec_in[4])\n\nwrap_plots(out)\n\n\n\n\nRe-write the code from above using a for loop to generate our figure\n\nShow answer\n\n\nvec_in <- c(\"YDL104C\", \"YLR115W\", \"YMR183C\", \"YML017W\") # input\n\nout <- vector(\"list\", length(vec_in)) # output\n\nfor (i in seq_along(vec_in)) { \n out[[i]] <- plot_expr(brauer_gene_exp, sys_name = vec_in[i]) \n}\nout\nwrap_plots(out)\n\n\n\n\nA note on vectorization\nIn general you should try to use a vectorized function/approach before using iteration. Vectorized approaches will be faster and require less code to run. If you are working with a vector or matrix, then there is likely a vectorized operation that can be used.\nThere are however a few common places that iteration is used:\n- To process multiple datasets/data.frames. (e.g. apply a function to a list of data.frames)\n- plotting (e.g. make many plots with a varying input parameter or aesthetic)\n- perform a custom operation that does not have a vectorized approach\nIntroduction to the apply family of functions.\nfor loops are a powerful tool to reduce code duplication, however your code can be simplified using the lapply function and related apply functions in base R. These functions essentially run for (i in seq_along(x)) behind the scenes so you don’t have to explicitly type this.\nThere is a function for each type of output:\nlapply() iterate over a vector, applying a function, returning a list\nsapply() iterate over a vector, applying a function, coercing the output to a vector\napply() iterate over a row, column, or all elements of a matrix\nmapply() iterate over each element of multiple supplied vectors\nThe lapply function requires two inputs: lapply(X, FUN, ...)\nX is a list or atomic vector\nFUN is a function\n… additional arguments to FUN\n\n\n# We previously used a for loop to create vectors with different values for mean\nvals <- c(1, 50, 20, 5, 70) # input\n\nout <- vector(\"list\", length(vals)) # output\n\nfor (i in seq_along(vals)) { # sequence\n \n out[[i]] <- rnorm(n = 10, mean = vals[i]) # body\n \n}\n\n# Using lapply() we can further simplify this code\n# x indicates where each element of the vector should be inserted\n# this is an example of an anonymous function\nout <- lapply(vals, function(x) rnorm(n = 10, mean = x))\n\n# we can also define the function first\nrnorm_custom <- function(x){\n rnorm(n = 10, mean = x)\n}\n\nout <- lapply(vals, rnorm_custom)\n\n# You can use brackets to include a multi-line code block\nout <- lapply(vals, function(x) {\n \n rnorm(n = 10, mean = x)\n \n})\n\n# Each element of the vector is passed to the first available argument\nout <- lapply(vals, rnorm, n = 10)\n#out <- lapply(vals, rnorm)\n\n\n\nUse rnorm() and lapply() to create 3 vectors of different lengths\n\nShow answer\n\n\n\nout <- lapply(c(10, 1, 4), rnorm)\nout <- lapply(c(10, 1, 4), function(x) rnorm(x))\nrnorm2 <- function(x){\n rnorm(n = x)\n}\nout <- lapply(c(10, 1, 4), function(x) rnorm2(x))\nout <- lapply(c(10, 1, 4), rnorm2)\nout\n\n#> [[1]]\n#> [1] 0.09307276 -0.95825854 0.25879688 0.42386975 -1.03153289\n#> [6] -0.42439767 -1.09057098 2.72994732 -0.97469569 -0.35996991\n#> \n#> [[2]]\n#> [1] 1.160781\n#> \n#> [[3]]\n#> [1] 1.90688834 -0.28012929 -0.01025581 0.28326510\n\n\nRe-write the code from above using lapply() to generate our growth rate figure\n\nShow answer\n\n\ngenes <- c(\"YDL104C\", \"YOR069W\", \"YLR115W\", \"YPR036W\")\n\nplot_expr(brauer_gene_exp, sys_name = vec_in[i]) \n\nexpr_plots <- lapply(genes, plot_expr, input = brauer_gene_exp)\n\nexpr_plots <- lapply(genes, function(gene){\n plot_expr(brauer_gene_exp, sys_name = gene)\n})\n\nexpr_plots\nwrap_plots(expr_plots)\n\n\n\n\nNote that we can iterate over lists in addition to vectors. A common operation might be to read in multiple files and perform some operation\n\n\n# get paths to files in \"data\" directory (dir() is an alias for list.files())\nfile_names <- dir(\"data\", full.names = TRUE)\n\n# read each file into R and store in a list\nlst_of_dfs <- lapply(file_names, read_csv)\n\n# get nrow of each file\nlapply(lst_of_dfs, nrow)\n\n# select 5 random rows\nlapply(lst_of_dfs, slice_sample, n = 5)\n\n# check if any NAs are present\nlapply(lst_of_dfs, function(x){\n sum(is.na(x)) > 0\n})\n\n\nNote that a data.frame is a list in R, such that each column is one element of a list (e.g. see output of typeof(mtcars)). So if we use lapply() on a data.frame it will iterate over each column.\n\n\nlapply(mtcars, mean)\n\n#> $mpg\n#> [1] 20.09062\n#> \n#> $cyl\n#> [1] 6.1875\n#> \n#> $disp\n#> [1] 230.7219\n#> \n#> $hp\n#> [1] 146.6875\n#> \n#> $drat\n#> [1] 3.596563\n#> \n#> $wt\n#> [1] 3.21725\n#> \n#> $qsec\n#> [1] 17.84875\n#> \n#> $vs\n#> [1] 0.4375\n#> \n#> $am\n#> [1] 0.40625\n#> \n#> $gear\n#> [1] 3.6875\n#> \n#> $carb\n#> [1] 2.8125\n\nlapply(mtcars, class)\n\n#> $mpg\n#> [1] \"numeric\"\n#> \n#> $cyl\n#> [1] \"numeric\"\n#> \n#> $disp\n#> [1] \"numeric\"\n#> \n#> $hp\n#> [1] \"numeric\"\n#> \n#> $drat\n#> [1] \"numeric\"\n#> \n#> $wt\n#> [1] \"numeric\"\n#> \n#> $qsec\n#> [1] \"numeric\"\n#> \n#> $vs\n#> [1] \"numeric\"\n#> \n#> $am\n#> [1] \"numeric\"\n#> \n#> $gear\n#> [1] \"numeric\"\n#> \n#> $carb\n#> [1] \"numeric\"\n\nsapply() will coerce the output to be a vector rather than a list.\n\n\nsapply(mtcars, mean)\n\n#> mpg cyl disp hp drat wt \n#> 20.090625 6.187500 230.721875 146.687500 3.596563 3.217250 \n#> qsec vs am gear carb \n#> 17.848750 0.437500 0.406250 3.687500 2.812500\n\nlapply(mtcars, mean) %>% unlist()\n\n#> mpg cyl disp hp drat wt \n#> 20.090625 6.187500 230.721875 146.687500 3.596563 3.217250 \n#> qsec vs am gear carb \n#> 17.848750 0.437500 0.406250 3.687500 2.812500\n\niterating over matrices\nThe apply() function allows you to iterates over a row, column, or all elements of a matrix.\n\n\nmat <- matrix(rnorm(100),\n nrow = 10,\n dimnames = list(paste0(\"row_\", letters[1:10]),\n paste0(\"column_\", letters[11:20])))\nmat\n\n#> column_k column_l column_m column_n column_o\n#> row_a -0.203086633 0.07864718 -1.3457612 0.8138788 -0.6534113\n#> row_b 0.113780860 -0.53261140 0.8611586 1.1080451 0.6905031\n#> row_c 0.083889011 1.35808026 -1.3323075 -1.0897243 0.6264521\n#> row_d 0.388760687 -1.96261669 1.6803024 -0.2616634 -1.8375725\n#> row_e -0.706199080 -1.69440925 0.4724637 1.3689316 1.2821530\n#> row_f 1.184512726 0.62947036 -1.1436721 0.4094435 0.9236262\n#> row_g -0.137533890 -0.23739221 -0.9420202 -0.7022158 -0.7195549\n#> row_h -0.954653848 -0.70387061 -0.7237769 0.3988952 0.6243948\n#> row_i -0.632167830 0.67052978 0.6324444 1.3874809 -0.4839324\n#> row_j -0.008142204 0.89070963 0.8311431 -0.9941451 1.0325004\n#> column_p column_q column_r column_s column_t\n#> row_a 1.0179468 0.4954533 1.1057192531 3.5326247 -0.8933596\n#> row_b -0.8065804 0.9631998 0.8824393747 1.7087607 0.1790658\n#> row_c 0.1663159 0.8422092 -0.0004235358 -1.6059234 0.4284422\n#> row_d 1.6646793 -0.7724212 -1.3779874423 -0.3355617 1.0587549\n#> row_e -0.8703873 2.6248080 -0.2949655063 -0.8288719 0.9975842\n#> row_f -0.2040175 0.3038377 -1.5465034162 -2.1691018 0.7270989\n#> row_g 2.2610814 -0.2749178 0.4854347770 -0.5420193 -0.2447763\n#> row_h -0.2587964 1.7100910 1.6116602055 -0.1071203 -0.1193766\n#> row_i 1.3675458 -0.1130078 -1.3900392612 0.1564137 1.2553816\n#> row_j -0.5836963 0.8724003 0.1998572485 -1.0915885 2.7918135\n\nThe arguments are:apply(X, MARGIN, FUN, ...)\nX = matrix\nMARGIN = specify how to iterate (1 = by row, 2 = by column, c(1,2) = all values)\nFUN = function to apply\n\n\n# get max value in each row\napply(mat, 1, max)\n\n# get max value in each column\napply(mat, 2, max)\n\n# multiply each value by 1e6 \napply(mat, c(1, 2), function(x) x * 1e6)\n\n\nAlternatively you can use for loops to iterate over matrices by using indexing to select/replace elements\n\n\nnr <- nrow(mat)\nnc <- ncol(mat)\n\n# iterate by column, seq_len makes range 1:nc\noutput <- vector(\"numeric\", nc)\nfor(i in seq_len(nc)){\n col <- mat[, i]\n output[i] <- sum(col + 5)\n}\n\n\n\n\n# iterate by row, seq_len makes range 1:nr \noutput <- vector(\"numeric\", nr)\nfor(i in seq_len(nr)){\n row <- mat[i, ]\n output[i] <- sum(row + 5)\n}\n\n\n\n\n# iterate by row, and column using nested for loops\n\n# make an output matrix, filled with 0s, with same names as mat\noutput <- matrix(0, nrow = nr, ncol = nc, dimnames = dimnames(mat))\n\nfor(i in seq_len(nr)){\n for(j in seq_len(nc)){\n val <- mat[i, j]\n output[i, j] <- val * 1e6\n }\n}\n\n\nNote that many operations on matrices are vectorized already and you often don’t need to use apply or for loops\n\n\nmat * 1e6\n\n\nUse the matrixStats package for common row-wise or column-wise operations before using apply.\n\n\n# in base R\nrowMeans(mat)\n\n#> row_a row_b row_c row_d row_e \n#> 0.39486512 0.51677615 -0.05229900 -0.17553255 0.23511075 \n#> row_f row_g row_h row_i row_j \n#> -0.08853054 -0.10539142 0.14774466 0.28506489 0.39408521\n\n\n# other functions in matrixStats\nlibrary(matrixStats)\nrowMaxs(mat)\n\n#> [1] 3.532625 1.708761 1.358080 1.680302 2.624808 1.184513 2.261081\n#> [8] 1.710091 1.387481 2.791813\n\nrowMaxs(mat, useNames = T)\n\n#> row_a row_b row_c row_d row_e row_f row_g \n#> 3.532625 1.708761 1.358080 1.680302 2.624808 1.184513 2.261081 \n#> row_h row_i row_j \n#> 1.710091 1.387481 2.791813\n\ncolMedians(mat)\n\n#> [1] -0.07283805 -0.07937251 -0.12565659 0.40416938 0.62542345\n#> [6] -0.01885079 0.66883127 0.09971686 -0.43879048 0.57777055\n\nIterating over multiple vectors\nIf you have two or more vectors containing values that you want to iterate over element-wise and process with a function this can be accomplished with mapply().\nmapply(FUN, ..., MoreArgs = NULL, SIMPLIFY = TRUE)\nFUN = function\n… = vectors to iterate over, supplied as named arguments\nMoreArgs = additional arguments with are not vectors supplied as a list\nSIMPLIFY = whether to simplify the result\n\n\ngenes <- c(\"YDL104C\", \"YOR069W\", \"YLR115W\", \"YPR036W\")\nshapes <- c(1, 2, 3, 4)\n\nexpr_plots <- mapply(plot_expr, # function\n sys_name = genes, # vector 1 \n shape = shapes, # vector 2\n MoreArgs = list(input = brauer_gene_exp), # fixed arguments\n SIMPLIFY = FALSE) # return a list, rather than coerce to simpler type\nwrap_plots(expr_plots)\n\n\n\n\nUse mapply() to create plots for 4 different genes, each with a different point size\n\nShow answer\n\n\ngenes <- c(\"YDL104C\", \"YOR069W\", \"YLR115W\", \"YPR036W\")\nsizes <- c(1, 2, 4, 6)\n\nexpr_plots <- mapply(plot_expr, # function\n sys_name = genes, \n size = sizes,\n MoreArgs = list(input = brauer_gene_exp),\n SIMPLIFY = FALSE)\n\nwrap_plots(expr_plots)\n\n\n\n\nAlternatively you may find it more readable to use a for loop and a list to store all of the arguments.\n\n\nn_plots <- 4\nplot_args <- list(\n sys_name = c(\"YDL104C\", \"YOR069W\", \"YLR115W\", \"YPR036W\"),\n size = c(2, 4, 6, 8),\n shape = c(1, 2, 3, 4)\n)\n\n# preallocate a list of length 4\nexpr_plots <- vector(mode = \"list\", length = n_plots)\n\nfor(i in seq_len(n_plots)){\n p <- plot_expr(input = brauer_gene_exp,\n sys_name = plot_args$sys_name[i],\n size = plot_args$size[i],\n shape = plot_args$shape[i])\n expr_plots[[i]] <- p\n}\n\nwrap_plots(expr_plots)\n\n\n\nUsing the map() family of functions from the purrr package\nThe purrr package from the tidyverse provides functions similar to lapply, which require even less code than lapply.\nThere is a function for each type of output:\nmap() makes a list (similar to lapply())\nmap2() iterate over two vector/lists by element (similar to mapply() with 2 vectors)\npmap() iterate over each element of a list element-wise (similar to mapply())\nmap_dfr() iterate over a vector/list, then combine output into a data.frame using bind_rows()\nmap_cfr() iterate over a vector/list, then combine output into a data.frame using bind_cols()\nmap_lgl() iterate over a vector/list, return a logical vector (similar to vapply)\nmap_int() iterate over a vector/list, return an integer vector\nmap_dbl() iterate over a vector/list, return a double vector\nmap_chr() iterate over a vector/list, return a character vector\nEach map() function requires two inputs: map(.x, .f, ...)\n.x is a list or atomic vector\n.f is a function or formula\n\n\n# We previously lapply to create vectors with different values for mean\nvals <- c(1, 50, 20, 5, 70) # input\nout <- lapply(vals, function(x) rnorm(n = 10, x))\n\n# Using map() we can further simplify this code\n# .x indicates where each element of the vector should be inserted\n# the ~ is shorthand for an anonymous function e.e. function(.x)\nout <- map(.x = vals, .f = ~ rnorm(n = 10, mean = .x))\nout <- map(vals, ~rnorm(n = 10, mean = .x))\n\n# You can use brackets to include a multi-line code block\nout <- map(vals, ~ {\n rnorm(n = 10, mean = .x)\n})\n\n# or use syntax similar to lapply\nout <- map(vals, function(input) mean(n = 10, input))\n\n# map() allows for very readable code\n# Each element of the vector is passed to the first available argument\nout <- map(vals, rnorm, n = 10)\nout <- lapply(vals, rnorm, n = 10)\n\n\nAdditional resources related to Programming in R\nControl Structures, from R Programming for Data Science\nProgramming Basics: Introduction to Data Science\nControl Flow: Advanced R\n\nShow session info\n\n\nsessionInfo()\n\n#> R version 4.2.0 (2022-04-22)\n#> Platform: x86_64-apple-darwin17.0 (64-bit)\n#> Running under: macOS Big Sur/Monterey 10.16\n#> \n#> Matrix products: default\n#> BLAS: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRblas.0.dylib\n#> LAPACK: /Library/Frameworks/R.framework/Versions/4.2/Resources/lib/libRlapack.dylib\n#> \n#> locale:\n#> [1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n#> \n#> attached base packages:\n#> [1] stats graphics grDevices utils datasets methods \n#> [7] base \n#> \n#> other attached packages:\n#> [1] matrixStats_0.62.0 patchwork_1.1.1 forcats_0.5.1 \n#> [4] stringr_1.4.1 dplyr_1.0.10 purrr_0.3.5 \n#> [7] readr_2.1.2 tidyr_1.2.0 tibble_3.1.8 \n#> [10] ggplot2_3.3.6 tidyverse_1.3.1 \n#> \n#> loaded via a namespace (and not attached):\n#> [1] lubridate_1.8.0 assertthat_0.2.1 digest_0.6.30 \n#> [4] utf8_1.2.2 R6_2.5.1 cellranger_1.1.0\n#> [7] backports_1.4.1 reprex_2.0.1 evaluate_0.16 \n#> [10] highr_0.9 httr_1.4.4 pillar_1.8.1 \n#> [13] rlang_1.0.6 readxl_1.4.0 rstudioapi_0.13 \n#> [16] jquerylib_0.1.4 rmarkdown_2.14 labeling_0.4.2 \n#> [19] bit_4.0.4 munsell_0.5.0 broom_0.8.0 \n#> [22] compiler_4.2.0 modelr_0.1.8 xfun_0.32 \n#> [25] pkgconfig_2.0.3 htmltools_0.5.2 downlit_0.4.2 \n#> [28] tidyselect_1.2.0 fansi_1.0.3 crayon_1.5.2 \n#> [31] tzdb_0.3.0 dbplyr_2.2.1 withr_2.5.0 \n#> [34] grid_4.2.0 jsonlite_1.8.3 gtable_0.3.0 \n#> [37] lifecycle_1.0.3 DBI_1.1.3 magrittr_2.0.3 \n#> [40] scales_1.2.0 cli_3.4.1 stringi_1.7.8 \n#> [43] vroom_1.5.7 cachem_1.0.6 farver_2.1.0 \n#> [46] fs_1.5.2 xml2_1.3.3 bslib_0.3.1 \n#> [49] ellipsis_0.3.2 generics_0.1.3 vctrs_0.4.1 \n#> [52] distill_1.5 tools_4.2.0 bit64_4.0.5 \n#> [55] glue_1.6.2 hms_1.1.2 parallel_4.2.0 \n#> [58] fastmap_1.1.0 yaml_2.3.6 colorspace_2.0-3\n#> [61] rvest_1.0.2 memoise_2.0.1 knitr_1.39 \n#> [64] haven_2.5.0 sass_0.4.1\n\n\n\n\n", "preview": "posts/2022-12-05-class-11-programming-in-r-pt2/programming-in-r-pt2_files/figure-html5/unnamed-chunk-13-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} }, { @@ -234,7 +251,7 @@ "categories": [], "contents": "\nThe Rmarkdown for this document is\nhttps://github.com/rnabioco/bmsc-7810-pbda/blob/main/_posts/2022-11-17-class-4-intro-to-ggplot2/class-4-intro-to-ggplot2.Rmd\nGoals for today\nIntroduction to plotting with the ggplot2 package\nThe grammar of graphics concept\nBasic plotting\nAdding additional information\nOther geometries\nMultiple geometries\nSaving plots\nAdditional Helpful Resources\nggplot2 package homepage :: https://ggplot2.tidyverse.org/\nggplot2 reference :: https://ggplot2.tidyverse.org/reference R for\nData Science :: https://r4ds.had.co.nz/\nggplot2 Book :: https://ggplot2-book.org/\nGallery of Plots and Examples :: https://r-graph-gallery.com/\nData Visualization with ggplot2 :: Cheat sheet ::\nhttps://github.com/rstudio/cheatsheets/blob/main/data-visualization.pdf\nThe ggplot2 Package\n\n\n\nThis package allows you to declaratively create graphics by giving a set\nof variables to map to aesthetics and then layer graphical directives to\nproduce a plot. It’s part of the tidyverse of R packages for data\nscience and analysis, sharing in their design philosophy. It’s an\nalternative to the built in R graphics and plotting functions.Written by Hadley Wickham\nGrammar of Graphics\n\n\n\nGrammar gives languages rules.\nGrammar has a technical meaning.\nGrammar makes language expressive.\n-Leland Wilkinson 1945-2021\nLayers of logical command flow and readability.\nLayers of ggplot2\n\n\n\nBasic Grammar\nPlot = data + aesthetics + geometry\ndata = the dataset, typically a dataframeaesthetics = map variables x and y to axisgeometry = type of graphic or plot to be rendered\nfacets = multiple plotsstatistics = add calculationstheme = make the plot pretty or follow a particular style\n\n\n# ggplot(, aes()) + ()\n\n?ggplot # bring up the ggplot function help\n\n\nConsider the Type of Data you want to plot\n\n\n\nData to Plot\nTo begin plotting we need to start with some data to visualize. Here we\ncan use a built-in dataset regarding Motor Trend Car Road Tests called\nmtcars. This dataset is a dataframe which is a key format for using\nwith ggplot. We can preview the data structure using the head()\nfunction.\n\n\n#some built in data.\n\nhead(mtcars)\n\n mpg cyl disp hp drat wt qsec vs am gear carb\nMazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4\nMazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4\nDatsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1\nHornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1\nHornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2\nValiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1\n\nThe data was extracted from the 1974 Motor Trend US magazine, and\ncomprises fuel consumption and 10 aspects of automobile design and\nperformance for 32 automobiles (1973–74 models).\nA data frame with 32 observations on 11 (numeric) variables.\n[, 1] mpg = Miles/(US) gallon\n[, 2] cyl = Number of cylinders\n[, 3] disp = Displacement (cu.in.)\n[, 4] hp = Gross horsepower\n[, 5] dra = Rear axle ratio\n[, 6] wt = Weight (1000 lbs)\n[, 7] qsec = 1/4 mile time\n[, 8] vs = Engine (0 = V-shaped, 1 = straight)\n[, 9] am = Transmission (0 = automatic, 1 = manual)\n[,10] gear = Number of forward gears\n[,11] carb = Number of carburetors-R Documentation\nBasic Plot\nUsing the basic ggplot grammar of graphics template we can produce a\nscatterplot from the dataframe.\n\n\n# ggplot(, aes()) + ()\n\n\nThe first part of the expression calls the ggplot function and takes\nthe dataframe and the aes function which are the aesthetics\nmappings. In this case we are mapping the x-axis to be the wt variable\nand the y-axis to be the mpg variable . If you only evaluate the first\npart this is what you get:\n\n\nggplot(mtcars, aes(x=wt, y=mpg))\n\n\n\nNext we have to add the geometry layer to be able to actually see the\ndata. Here we are adding the geom_point geometry which allows you to\nvisualize the data as points. You use a plus sign to add these\nadditional layers.\n\n\nggplot(mtcars, aes(x=wt, y=mpg)) + geom_point()\n\n\n\nWe can change the data being plotted by picking a different column from\nthe dataframe. For instance here we are plotting the horsepower(hp)\nversus miles per gallon(mpg). Also note that we can make the code more\nreadable by placing proceeding layers on a different line after the plus\nsign. A common error is misplacing the plus sign. It must be trailing on\nthe line before the next layer.\n\n\nggplot(mtcars, aes(x=hp, y=mpg)) + \n geom_point()\n\n\n\nExercise: Try building a scatterplot on your own. This time plot the\nvariables corresponding to the number of cylinders and the type of\ntransmission.\n\n\n\nExercise: Modify the scatterplot to plot horsepower instead of the type\nof transmission. Can you start to see a relationship with the data?\nAdding Additional Information to the Plot\nTitle\nWe can add a title to the plot simply by adding another layer and the\nggtitle() function.\n\n\nggplot(mtcars, aes(x=hp, y=mpg)) + \n geom_point() +\n ggtitle(\"1974 Cars: Horsepower vs Miles Per Gallon\")\n\n\n\nX and Y axis Labels\nWe can overwrite the default labels and add our own to the x and y axis\nby using the xlab() and ylab() functions respectively.\n\n\nggplot(mtcars, aes(x=hp, y=mpg)) + \n geom_point() +\n ggtitle(\"1974 Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\")\n\n\n\nSet title and axis labels in one layer\n\n\nggplot(mtcars, aes(x=hp, y=mpg, alpha = 0.5)) + \n geom_point() +\n labs(x = \"Horepower\", \n y = \"Miles Per Gallon\", \n title = \"Horsepower vs Miles Per Gallon Scatterplot\",\n subtitle = \"Motor Trend Car Road Tests - 1974\",\n caption = \"Smith et al. 1974\")\n\n\n\nNotice that we also added an alpha aesthetic which helps us visualize\noverlapping points. We can add a show.legend = FALSE argument to the\ngeom_point function to remove the alpha legend and clean up the plot\nfigure. Let’s try it. You can also specify a vector of aesthetics to\ndisplay.\nCheck the documentation ?geom_point.\nGetting Geometry Specific Help\nWe can easily add a third bit of information to the plot by using the\ncolor aesthetic. Each geometry has its own list of aesthetics that you\ncan add and modify. Consult the help page for each one.\n\n\n?geom_point() # bring up the help page for geom_point()\n\n\nAdding the Color Aesthetic\nHere we are adding the color aesthetic.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\")\n\n\n\nAnd we can relabel the legend title for the new color aesthetic to make\nit more readable.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nA Fourth Aesthetic\nYou can even continue to add even more information to the plot through\nadditional aesthetics. Though this might be a bit much.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl, size = wt)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\", size=\"weight (x1000lb)\")\n\n\n\nInstead we can use a specific value instead of the wt variable to\nadjust the size of the dots.\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl, size = 3)) + \n geom_point() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nOther Geometries\nThere are many other geometries that you can use in your plots.\nhttps://ggplot2.tidyverse.org/reference\nHere is a short list:\ngeom_point(): scatterplot\ngeom_line(): lines connecting points by increasing value of x\ngeom_path(): lines connecting points in sequence of appearance\ngeom_boxplot(): box and whiskers plot for categorical variables\ngeom_bar(): bar charts for categorical x axis\ngeom_col(): bar chart where heights of the bars represent values in the\ndata\ngeom_histogram(): histogram for continuous x axis\ngeom_violin(): distribution kernel of data dispersion\ngeom_smooth(): function line based on data\ngeom_bin2d(): heatmap of 2d bin counts\ngeom_contour(): 2d contours of a 3d surface\ngeom_count(): count overlapping points\ngeom_density(): smoothed density estimates\ngeom_dotplot(): dot plot\ngeom_hex(): hexagonal heatmap of 2d bin counts\ngeom_freqpoly(): histogram and frequency polygons\ngeom_jitter(): jittered point plot geom_polygon(): polygons\ngeom_line()\nBut utilizing the right plot to efficiently show your data is key. Here\nwe swapped the geom_point for geom_line to see what would happen. You\ncould also try something like geom_bin2d()\n\n\nggplot(mtcars, aes(x=hp, y=mpg, color=cyl)) + \n geom_line() +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nPlotting the Categories as a Bar Chart with geom_col()\nThe geom_col() geometry is a type of bar plot that uses the heights of\nthe bars to represent values in the data. Let’s look at plotting this\ntype of data for the cars in this dataset.\n\n\n?geom_col()\n\n\n\n\nhead(mtcars)\n\n mpg cyl disp hp drat wt qsec vs am gear carb\nMazda RX4 21.0 6 160 110 3.90 2.620 16.46 0 1 4 4\nMazda RX4 Wag 21.0 6 160 110 3.90 2.875 17.02 0 1 4 4\nDatsun 710 22.8 4 108 93 3.85 2.320 18.61 1 1 4 1\nHornet 4 Drive 21.4 6 258 110 3.08 3.215 19.44 1 0 3 1\nHornet Sportabout 18.7 8 360 175 3.15 3.440 17.02 0 0 3 2\nValiant 18.1 6 225 105 2.76 3.460 20.22 1 0 3 1\n\nLooking back at the data structure of mtcars, we see that the names of\nthe cars are stored as the row names of the data frame. We can access\nthis using the rownames()function and use it in subsequent plots.\nQ: What was another way to address this issue, discussed in the first\nblock?\n\n\nrownames(mtcars)\n\n [1] \"Mazda RX4\" \"Mazda RX4 Wag\" \"Datsun 710\" \n [4] \"Hornet 4 Drive\" \"Hornet Sportabout\" \"Valiant\" \n [7] \"Duster 360\" \"Merc 240D\" \"Merc 230\" \n[10] \"Merc 280\" \"Merc 280C\" \"Merc 450SE\" \n[13] \"Merc 450SL\" \"Merc 450SLC\" \"Cadillac Fleetwood\" \n[16] \"Lincoln Continental\" \"Chrysler Imperial\" \"Fiat 128\" \n[19] \"Honda Civic\" \"Toyota Corolla\" \"Toyota Corona\" \n[22] \"Dodge Challenger\" \"AMC Javelin\" \"Camaro Z28\" \n[25] \"Pontiac Firebird\" \"Fiat X1-9\" \"Porsche 914-2\" \n[28] \"Lotus Europa\" \"Ford Pantera L\" \"Ferrari Dino\" \n[31] \"Maserati Bora\" \"Volvo 142E\" \n\n\n\nggplot(mtcars, aes(x=rownames(mtcars), y=mpg)) + \n geom_col() +\n ggtitle(\"1974 Cars: Miles Per Gallon\")\n\n\n\nYou will learn other ways to make this more legible later. For a quick\nfix we can swap the x and y mappings.\n\n\nggplot(mtcars, aes(y=rownames(mtcars), x=mpg)) + \n geom_col() +\n ggtitle(\"1974 Cars: Miles Per Gallon\")\n\n\n\nWe can reorder the data to make it easier to visualize important\ninformation.\n\n\nggplot(mtcars, aes(y=reorder(rownames(mtcars), mpg), x=mpg)) + \n geom_col() +\n ggtitle(\"1974 Cars: Ranked by Miles Per Gallon\")\n\n\n\nExercise: Plot a bar chart using geom_col() with the mtcar dataset. Plot\nthe names of the cars ranked by the weight of each car. Try adding a\nthird aesthetic color for horsepower.\n\n\n\nMultiple Geometries\nYou can also add another layer of geometry to the same ggplot. Notice\nyou can have two separate aesthetic declarations and they have moved\nfrom the ggplot function to their respective geom_ functions.\n\n\n# ggplot(data = , mapping = aes()) + \n# () + \n# () \n\n# OR\n\n# ggplot(data = ) + \n# (mapping = aes()) + \n# (mapping = aes()) \n\nggplot(mtcars) +\n geom_point(aes(x=hp, y=mpg)) +\n geom_line(aes(x=hp, y=mpg, color=cyl)) +\n ggtitle(\"Modern Cars: Horsepower vs Miles Per Gallon\") +\n ylab(\"miles per gallon\") + \n xlab(\"horsepower\") +\n labs(color=\"#cylinders\")\n\n\n\nThis particular geometry addition isn’t very useful.\nExercise: Try adding geom_smooth() instead of geom_line().\nSaving Plots\nSaving these plots is easy! Simply call the ggsave() function to save\nthe last plot that you created. You can specify the file format by\nchanging the extension after the filename.\n\n\nggsave(\"plot.png\") # saves the last plot to a PNG file in the current working directory\n\n\nYou can also specify the dots per inch and the width of height of the\nimage to ensure publication quality figures upon saving.\n\n\nggsave(\"plot-highres.png\", dpi = 300, width = 8, height = 4) # you can specify the dots per inch (dpi) and the width and height parameters\n\n\nExercise: Try saving the last plot that we produced as a jpg. Can you\nnavigate to where it saved and open it on your computer?\nMore Examples\nLets take a look at gallery resource to preview different plot types and\nget ideas for our own plots.https://r-graph-gallery.com/\nSessionInfo\n\n\nsessionInfo()\n\nR version 4.2.2 (2022-10-31)\nPlatform: aarch64-apple-darwin20 (64-bit)\nRunning under: macOS Monterey 12.6\n\nMatrix products: default\nBLAS: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRblas.0.dylib\nLAPACK: /Library/Frameworks/R.framework/Versions/4.2-arm64/Resources/lib/libRlapack.dylib\n\nlocale:\n[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8\n\nattached base packages:\n[1] stats graphics grDevices utils datasets methods \n[7] base \n\nother attached packages:\n[1] forcats_0.5.2 stringr_1.4.1 dplyr_1.0.10 purrr_0.3.5 \n[5] readr_2.1.3 tidyr_1.2.1 tibble_3.1.8 ggplot2_3.4.0 \n[9] tidyverse_1.3.2\n\nloaded via a namespace (and not attached):\n [1] lubridate_1.8.0 assertthat_0.2.1 digest_0.6.30 \n [4] utf8_1.2.2 R6_2.5.1 cellranger_1.1.0 \n [7] backports_1.4.1 reprex_2.0.2 evaluate_0.17 \n[10] httr_1.4.4 highr_0.9 pillar_1.8.1 \n[13] rlang_1.0.6 googlesheets4_1.0.1 readxl_1.4.1 \n[16] rstudioapi_0.14 jquerylib_0.1.4 rmarkdown_2.17 \n[19] labeling_0.4.2 googledrive_2.0.0 munsell_0.5.0 \n[22] broom_1.0.1 compiler_4.2.2 modelr_0.1.9 \n[25] xfun_0.34 pkgconfig_2.0.3 htmltools_0.5.3 \n[28] downlit_0.4.2 tidyselect_1.2.0 fansi_1.0.3 \n[31] crayon_1.5.2 tzdb_0.3.0 dbplyr_2.2.1 \n[34] withr_2.5.0 grid_4.2.2 jsonlite_1.8.3 \n[37] gtable_0.3.1 lifecycle_1.0.3 DBI_1.1.3 \n[40] magrittr_2.0.3 scales_1.2.1 cli_3.4.1 \n[43] stringi_1.7.8 cachem_1.0.6 farver_2.1.1 \n[46] fs_1.5.2 xml2_1.3.3 bslib_0.4.1 \n[49] ellipsis_0.3.2 generics_0.1.3 vctrs_0.5.0 \n[52] distill_1.5 tools_4.2.2 glue_1.6.2 \n[55] hms_1.1.2 fastmap_1.1.0 yaml_2.3.6 \n[58] colorspace_2.0-3 gargle_1.2.1 rvest_1.0.3 \n[61] memoise_2.0.1 knitr_1.40 haven_2.5.1 \n[64] sass_0.4.2 \n\n\n\n\n", "preview": "posts/2022-11-17-class-4-intro-to-ggplot2/class-4-intro-to-ggplot2_files/figure-html5/unnamed-chunk-8-1.png", - "last_modified": "2023-12-07T16:39:32+00:00", + "last_modified": "2023-12-07T16:51:54+00:00", "input_file": {} } ] diff --git a/search.json b/search.json index bc358f6..71f2b7c 100644 --- a/search.json +++ b/search.json @@ -5,21 +5,21 @@ "title": "Practical Biological Data Analysis", "author": [], "contents": "\n\n\n\n", - "last_modified": "2023-12-07T16:40:49+00:00" + "last_modified": "2023-12-07T16:53:02+00:00" }, { "path": "index.html", "title": "Practical Biological Data Analysis", "author": [], "contents": "\nPractical Biological Data Analysis with R and RStudio\nParticipation in this course requires completion of an assignment prior to the course start date (see the Prerequisites section).\nOverview\nIn this short course you will learn to analyze and visualize complex data sets using the R statistical programming language and the RStudio IDE. We will focus on key analysis skills and foundational programming concepts necessary for the efficient and reproducible analysis of biological data sets.\nOrganization and Contacts\nKent Riemondy (Director, Instructor): kent.riemondy@cuanschutz.edu\nMichael Kaufman (Instructor): michael.kaufman@cuanschutz.edu\nRyan Sheridan (Instructor): ryan.sheridan@cuanschutz.edu\nKristen Wells-Wrasman (Instructor): kristen.wells-wrasman@cuanschutz.edu\nThe course (2 credit hours) consists of 13 two hour classes held Mon through Fri from Nov 29 through Dec 15 from 8:00 -10:00 am. The course will be held in Research 1 North (P18) in the P18-CTL-1309 Computer lab. All classes will be recorded and made available through Canvas.\nVirtual office hours will also be provided outside of course hours.\nGoals\nBuild competence in R so that students use R for data analysis rather than using interactive applications such as Excel or Prism.\nDevelop reproducible and efficient data analysis habits.\nIntroduce data visualization techniques for complex datasets.\nPrerequisites\nA personal computer with a common operating system ( macOS, Linux, or Windows), and internet access is necessary to participate in this class. Tablets or iPads will not be supported. Please reach out to us ASAP if you do not have access to a computer, or if you have concerns about the suitability of your device. There will be a required prerequisite assignment to be completed prior to the start of the course. The assignment will involve installing necessary software, and completion of material providing basic familiarity with R and interacting with the Rstudio IDE. Office hours will be provided prior to the course to assist with any issues that arise with completing the prerequisite assignment.\nAssignments\nStudent’s grades will be determined by completion of the prerequisite assignment (10%), class participation (10%), and 4 homework assignments (80%), to be completed by the end of the course.\nCourse assistance\nWe will provide a Slack workspace for discussion during and after classes, to collaborate, and to get help. In addition the course instructors will hold virtual office hours to provide assistance outside of the course. Please email the course instructors via (rbi.fellows@cuanschutz.edu) to schedule a time.\n\n\n\n", - "last_modified": "2023-12-07T16:40:49+00:00" + "last_modified": "2023-12-07T16:53:03+00:00" }, { "path": "schedule.html", "title": "Schedule", "author": [], "contents": "\nClass Schedule\nClass 1: Wednesday, Nov. 29\nIntroduction to the R statistical programming language\n\nClass 2: Thursday, Nov. 30\nR fundamentals\n\nClass 3: Friday, Dec. 1\nData wrangling with the tidyverse\n\nClass 4: Monday, Dec. 4\nReshaping data into a tidy format\n\nClass 5: Tuesday, Dec. 5\nIntroduction to ggplot2 (pt.1)\n\nClass 6: Wednesday, Dec. 6\nIntroduction to ggplot2 (pt.2)\n\nClass 7: Thursday, Dec. 7\nData analysis vignette\n\nClass 8: Friday, Dec. 8\nIntroduction to matrix operations\n\nClass 9: Monday, Dec. 11\nIntroduction to clustering techniques\n\nClass 10: Tuesday, Dec. 12\nIntroduction to visualizing data with heatmaps\n\nClass 11: Wednesday, Dec. 13\nProgramming fundamentals in R (pt. 1)\n\nClass 12: Thursday, Dec. 14\nProgramming fundamentals in R (pt. 2)\n\nClass 13: Friday, Dec. 15\nCourse wrap up\n\nProblem Sets\nProblem set 1: Due Friday Dec 1 (midnight)\nProblem set 2: Due Tuesday Dec 5 (midnight)\nProblem set 3: Due Friday Dec. 8 (midnight)\nProblem set 4: Due Wednesday Dec. 13 (midnight)\n\n\n\n", - "last_modified": "2023-12-07T16:40:50+00:00" + "last_modified": "2023-12-07T16:53:03+00:00" } ], "collections": ["posts/posts.json"]