摘要:而這一次的項目,原本以為開發(fā)挺順利的,但是開發(fā)完了,才發(fā)現(xiàn)自己犯了一個低級而嚴(yán)重的錯,這樣的一個失誤,我一直耿耿于懷。但是監(jiān)聽用戶退出頁面微信瀏覽器上面的那個返回或者關(guān)閉按鈕卻死活不行。也容易犯一些低級的錯誤。
1.前言
前端從事了超過兩年,修復(fù)了無數(shù)的bug,寫了無數(shù)的bug;挖了很多次坑,填了很多次坑;犯了很多次錯,彌補了很多次,學(xué)習(xí)了很多次。一般而言,對于bug、坑,都是修復(fù)完了或者填完了,并且記住為什么會產(chǎn)生bug,為什么有坑,為什么犯錯,怎么解決的,下次怎么避免,就行了,就學(xué)習(xí)到了。而這一次的項目,原本以為開發(fā)挺順利的,但是開發(fā)完了,才發(fā)現(xiàn)自己犯了一個低級而嚴(yán)重的錯,這樣的一個失誤,我一直耿耿于懷。
2.起因在3月9號的這一天,公司有個活動,希望用答題活動推廣自己的小程序。結(jié)果因為開發(fā)時間太緊,小程序在3月5號才提審。在3月8號早上,小程序還沒有審核,在不得已的情況下,只能把答題活動以網(wǎng)頁的形式進行,使用vue開發(fā)。由于在3月9號要用到這個答題活動,所以3月8號必須要完成開發(fā),測試,驗收。
開發(fā)的過程,都挺順利,只是把小程序的一些代碼,改成vue開發(fā)移動端網(wǎng)站的方式,把標(biāo)簽換了,樣式稍微重寫一下,項目就跑起來了,至于一些交互邏輯,由于不能使用小程序的API,只能另找良方代替,但問題基本不大。
麻煩的一個需求就是:當(dāng)用戶沒答完題中途退出的時候,要記錄用戶的答題狀態(tài)。比如答了哪些題目,哪些題目錯了,哪些題目正確了,拿了多少分等數(shù)據(jù)。在小程序里面,很輕松可以利用生命周期函數(shù) unload() 進行監(jiān)聽。當(dāng)用戶沒答完題退出頁面的時候,把用戶當(dāng)前的答題數(shù)據(jù),傳給后臺,讓后臺進行保存。在用戶下次進入頁面的時候,我可以根據(jù)后臺返回的用戶答題狀態(tài),進行信息的展示。如果用戶沒答過題目,就重新開始答題,如果用戶上次退出的時候,沒答完題目,就按照退出時的進度,讓用戶重新答題,如果答完了題目,直接顯示答題結(jié)果頁面。
這個需求不難實現(xiàn),小程序有 onload() 和 unload() 兩個生命周期函數(shù),只是在這兩個函數(shù)里面,調(diào)兩次接口而已。
但在網(wǎng)頁里面,監(jiān)聽用戶進入頁面簡單。但是監(jiān)聽用戶退出頁面(微信瀏覽器上面的那個‘返回’或者‘關(guān)閉’按鈕)卻死活不行。網(wǎng)上最多的解決方案是這個,但是不知道是我使用方式有問題還是人品問題,壓根沒用,無論是微信開發(fā)者工具,還是安卓或者蘋果真機。
答案來自知乎:微信自帶瀏覽器環(huán)境內(nèi)左上角返回、關(guān)閉按鈕事件監(jiān)控?
pushHistory(); window.addEventListener("popstate", function(e) { alert("我監(jiān)聽到了瀏覽器的返回按鈕事件啦");//根據(jù)自己的需求實現(xiàn)自己的功能 }, false); function pushHistory() { var state = { title: "title", url: "#" }; window.history.pushState(state, "title", "#"); }
根據(jù)網(wǎng)上的方案,試了幾個(包括vue的生命周期函數(shù)),沒一個可行的。最后無奈之下,只能用一個蠢方法,用戶點擊每一題選項的時候,就把用戶當(dāng)前的記錄,通過接口發(fā)給后臺,讓后臺記錄。這個就是我該文章說的低級嚴(yán)重的失誤,想必大家也知道是怎么回事了。
3.失誤分析這次的答題活動,一共有三輪,每輪10道題,現(xiàn)場大概有500人答題。本來使用小程序開發(fā),不管用戶是沒答題就讓用戶可以開始答題,答題途中退出就記錄狀態(tài),答完題就顯示結(jié)果。在這個過程中,我跟后臺交互的只有兩次:一次是用戶進來的時候獲取用戶答題進度,一次是用戶答完了最后一題,發(fā)送用戶成績,讓后臺記錄;或者中途退出,發(fā)送用戶答題進度給后臺,讓后臺記錄。
但是后來我在網(wǎng)頁中,由于暫時沒法監(jiān)聽用戶是否退出,所以選擇了用戶回答完每一題的時候,把數(shù)據(jù)發(fā)給后臺,讓后臺答題進度。這樣請求數(shù)就多了N倍。服務(wù)器的壓力就大了很多。
由于用戶進來,無論是小程序還是網(wǎng)站,都要請求接口,獲取用戶答題數(shù)據(jù),這次不在對比范圍。這樣原本小程序只需要和后臺進行一次握手,但是在網(wǎng)頁中,采用了不合適的方式,和后臺握手次數(shù)變成了10次。足足多了9倍。如果是500人,每一輪從原本的500次,變成了5000次,三輪就從原本的1500次,變成了15000次!一般而言,10道題選擇題,是兩分鐘左右的回答時間,就相當(dāng)于在2分鐘內(nèi)服務(wù)器要響應(yīng)的次數(shù)多了9倍,這個擔(dān)子突然重了很多。而已這些請求,基本都有沒什么意義的,因為絕大部分的人,10道題,大概兩分鐘的答題時間里面,不會中途退出,相當(dāng)于我做了一件沒意義,又消耗服務(wù)器性能的事情。
讓我耿耿于懷的原因,我一向?qū)φ埱髷?shù)嚴(yán)格的控制,雖然現(xiàn)在公司不怎么考慮性能,服務(wù)器壓力。但是這會引起我的強迫癥。
4.解壓方案由于答題活動,9號要使用,而我是8號晚上洗完澡的時候和同事聊天的時候才想起,所以我沒時間改了,因為改了也是需要時間開發(fā),測試。9號由于同事請假,他的項目也由我負(fù)責(zé),也是比較趕的項目,我也沒那么多時間改。只能委屈一下服務(wù)器了。
說是這樣說,但是關(guān)于其他的給服務(wù)器減輕負(fù)擔(dān)的方案,還是有比較講一下,算是給自己提個醒,也算是給大家提個醒。開發(fā)要注意一點:不要急,不要急,不要急。
PS:當(dāng)時就是看著時間差不多是下午四點半了,然后還有兩個零散功能沒做,又要測試。找了很久的解決方案(監(jiān)聽微信的‘返回’或者‘關(guān)閉按鈕’)都沒下落的情況下,一下急了,腦袋放空,就想了那個方法。
cookie或者localstore記錄用戶的狀態(tài),這個應(yīng)該是最好的解決方案了,也應(yīng)該是最簡單的解決方案。
比如使用cookie記錄用戶的答題進度。在用戶每答一題的時候,就把cookie記錄到的數(shù)據(jù),更新一次。這樣只需要在用戶答完了最后一題的時候再把用戶的成績發(fā)給后臺就好,至于用戶中途退出也沒有,根據(jù)cookie判斷就好,如果cookie有記錄到用戶的數(shù)據(jù)。就顯示上次用戶退出時候的題目,讓用戶繼續(xù)答題。
原代碼:/** * @dedependson 點擊選項 * @index 題目索引 number * @item 當(dāng)前選項對象 object */ chooseDo(index,item){ /*其他代碼略*/ let _this=this; let _data={ qid:_this.qid,//答題輪次,如"2"代表第二輪答題 questions:_this.questions,//已答題目,"1,2,3"這個表示id為1,2,3的題目已經(jīng)回答了 totalScore:_this.totalScore//當(dāng)前得分 } //發(fā)送請求,讓后臺記錄用戶答題進度 this.$http.post(http_url.submit,_data,{emulateJSON:true}).then(res=>{ }) }
然后再到頁面加載的時候
mounted(){ this.$http.get(http_url.getQuestions,{ params:{ qid:this.qid } }).then(res=>{ res=res.body; //如果請求成功 if(res.code===0){ //如果用戶沒答完題 0-沒開始答題 1-沒答完題 2-答完題目 if(res.datas.status!==2){ //獲取答題的題目 this.questionList=res.datas.entryList; //如果題目小于10,就是開始答題了,但是沒答完(中途退出的原因) if(this.questionList.length<10){ //顯示答題頁面,讓用戶答題 this.questionListShow=true; } //否則就是沒答過題目,讓用戶答題 else{ //顯示開始答題頁面(答題首頁,用戶需要點擊開始答題) } } //如果用戶已經(jīng)答完題,顯示結(jié)果頁 else{ //代碼略 } } else{ alert(res.msg) } }) }cookie方案
chooseDo(index,item){ /*其他代碼略*/ let _this=this; let _data={ qid:_this.qid,//答題輪次,如"2"代表第二輪答題 questions:_this.questions,//已答題目,"1,2,3"這個表示id為1,2,3的題目已經(jīng)回答了 totalScore:_this.totalScore//當(dāng)前得分 } //保存cookie一天 //_this.qid作為答題輪次的標(biāo)識 setCookie("answer-qid"+_this.qid,_this.qid,1); setCookie("answer-questions"+_this.qid,_this.questions,1); setCookie("answer-totalScore"+_this.qid,_this.totalScore,1); }
cookie函數(shù)參考:ec-do
//設(shè)置cookie setCookie(name, value, iDay) { let oDate = new Date(); oDate.setDate(oDate.getDate() + iDay); document.cookie = name + "=" + value + ";expires=" + oDate; }, //獲取cookie getCookie(name) { let arr = document.cookie.split("; "),arr2; for (let i = 0; i < arr.length; i++) { arr2 = arr[i].split("="); if (arr2[0] == name) { return arr2[1]; } } return ""; }, //刪除cookie removeCookie(name) { this.setCookie(name, 1, -1); },
然后再到頁面加載的時候,處理方式的改變。
mounted(){ this.$http.get(http_url.getQuestions,{ params:{ qid:this.qid } }).then(res=>{ res=res.body; //如果請求成功 if(res.code===0){ //如果用戶沒答完題 0-沒開始答題 1-沒答完題 2-答完題目 if(res.datas.status!==2){ //記錄答題輪次 this.qid=res.datas.qid; //獲取答題的題目 this.questionList=res.datas.entryList; //如果用戶中途退出,我們沒有和后臺對接口,后臺無法記錄用戶答題進度,所以這次請求,返回的結(jié)果要么是沒開始答題,要么是答完題了。 //要還原用戶答題記錄,要使用cookie //如果存在cookie記錄,那么用戶肯定是至少答過一題,還原用戶答題進度 let _answerQid=getCookie("answer-qid"+this.qid) _answerQuestions=getCookie("answer-qid"+this.qid).split(","); //字符串轉(zhuǎn)整數(shù) _answerQuestions.map(item=>+item); if(_answerQid&&_answerQuestions){ this.questionList.fifler(item=>{ //item.id是題目的id //如果題目的id存在,就過濾掉 _answerQuestions.indexOf(item.id)===-1 }); //顯示答題頁面,讓用戶答題 this.questionListShow=true; } //否則就是沒答過題目,讓用戶答題 else{ //顯示開始答題頁面(答題首頁,用戶需要點擊開始答題) } } //如果用戶已經(jīng)答完題,顯示結(jié)果頁 else{ //代碼略 } } else{ alert(res.msg) } }) }
代碼上面,可能用了 cookie 會復(fù)雜些,但是就多了幾行而已,差不了多少,反倒是減輕了很多請求。
在小程序沒有使用這個方案,就是考慮到用戶退出小程序,可能會清除緩存,雖然這個幾率不大,所以使用生命周期函數(shù)進行unload()進行監(jiān)聽,用戶退出就把用戶答題進度提交給后臺,讓后臺記錄,這樣的情況不會很多,甚至沒有,請求不會很多,所以當(dāng)時就用了這個方案。沒有使用cookie或者localstore。
注意幾點:
1.無論什么情況,開發(fā)都需要一個清醒的頭腦,因為頭腦不清醒,寫的都是bug,那個活動是一個一次性的項目,如果是長期的,我肯定會重構(gòu)的,因為當(dāng)時寫的代碼太爛了。也容易犯一些低級的錯誤。
2.不要為了小概率的事件想得太多,給自己,同事,服務(wù)器都帶來麻煩,也影響項目進度。這次就是想得太多,結(jié)果提測的時間晚了,驗收的時間晚了,自己也犯了錯誤。想太多的后果可能就是撿了芝麻,漏了西瓜,甚至是偷雞不成蝕把米。
小結(jié)這次的的失誤就告一段落了,我也總結(jié)了一下,自己為什么會對這次失誤更更于懷。
1.最近一直在看怎么優(yōu)化代碼,讓代碼更有可讀性,可維護性。卻犯了請求數(shù)過多的錯。顧此失彼啊。
2.第二個就是因為這次失誤,導(dǎo)致的后果太嚴(yán)重了,直接多了90%的請求。以往失誤導(dǎo)致的后果沒怎么嚴(yán)重。
3.以往犯錯的時候,在項目上線之前能夠發(fā)現(xiàn),并且有時候改,這次不一樣,這次是發(fā)現(xiàn)了,但是沒時間改了。
4.那些以為不會有,不應(yīng)該犯的錯??赡芫驮陬^腦不清醒的時候,就會犯這些錯誤,無論什么時候都得留個神,這次也算是我自己提醒自己了。
不過結(jié)局是還算是好的,當(dāng)天因為時間關(guān)系,答題活動沒有進行,所以服務(wù)器沒有受到考驗。如果當(dāng)天服務(wù)器承受不住壓力,崩了,我也可能要引咎辭職了!
好了,故事就是這樣了,有點日記的感覺,希望大家諒解下。如果文章有什么地方寫錯了,也歡迎指點交流。
--------------------華麗的分割線-------------------
想了解更多,關(guān)注關(guān)注我的微信公眾號:守候書閣
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/93350.html
摘要:昨天看到一個大新聞拼多多在日凌晨出現(xiàn)漏洞,用戶可以領(lǐng)元無門檻優(yōu)惠券。拼多多本來就是家爭議頗大的公司,這次事件更是引發(fā)輿論熱議。有人估計全球為此花費的相關(guān)費用有數(shù)億美元。軟件發(fā)布測試版讓用戶使用,就屬于一種黑盒測試。 昨天看到一個大新聞: 拼多多在20日凌晨出現(xiàn)漏洞,用戶可以領(lǐng)100元無門檻優(yōu)惠券 。一夜之間,被黑產(chǎn)、羊毛黨和聞訊而來的吃瓜群眾薅了個底朝天,直到第二天上午9點才將優(yōu)惠券下...
摘要:但這并不意味著依賴版本是鎖死的。黃色表示不符合指定的語義化版本范圍,比如大版本升級,升級可能會遇到兼容性問題。文件可以列出不想打包的文件,避免把一些無關(guān)的文件發(fā)布到上。 作者: LeanCloud weakish 分享一些 npm 包管理工具的實用小竅門,希望能夠略微提高下前端、Node.js 開發(fā)者的生活質(zhì)量。 絕大多數(shù)前端和 Node.js 開發(fā)者每天的日常工作都離不開 npm,不...
摘要:當(dāng)奧巴馬贏得美國總統(tǒng)大選時,頁面活躍度刷新了記錄。對于每一個成因,都應(yīng)制定相應(yīng)的預(yù)防措施,以減輕大規(guī)模事故。這種故障會通過許多層面進入系統(tǒng)服務(wù)中,導(dǎo)致系統(tǒng)故障的發(fā)生。 作者介紹:Ben Maurer是Facebook的網(wǎng)絡(luò)基礎(chǔ)團隊的技術(shù)領(lǐng)先者,主要負(fù)責(zé)整個Facebook面向用戶產(chǎn)品的性能和可靠性。Ben于2010年正式加入Facebook,基礎(chǔ)設(shè)施團隊的成員。在加入Facebook之...
閱讀 3761·2021-10-13 09:39
閱讀 3810·2021-09-24 09:48
閱讀 1206·2021-09-01 10:30
閱讀 2537·2019-08-30 15:55
閱讀 1788·2019-08-29 16:39
閱讀 2307·2019-08-26 13:55
閱讀 3063·2019-08-26 12:23
閱讀 1645·2019-08-26 11:59