摘要:兄弟組件之間無法直接通信,它們需要利用同一層的上級作為中轉(zhuǎn)站。在兩個地方會用到,一是通過提交后需要拿到里面的數(shù)據(jù),二是利用監(jiān)聽到發(fā)生變化后調(diào)用它來獲取新的數(shù)據(jù)。
前言
在學(xué)習(xí)react的過程中,深深的被react的函數(shù)式編程的模式所吸引,一切皆組件,所有的東西都是JavaScript。React框架其實功能很單一,主要負責(zé)渲染的功能,但是社區(qū)很活躍,衍生出了很多優(yōu)秀的庫和工具。個人覺得,想要做好一個項目,往往需要其他庫和工具的配合,例如redux管理數(shù)據(jù),react-router管理路由等,掌握基本的webpack配置和es6語法,然后想要提高性能,還有配合react的鉤子函數(shù)和immutable.js,什么時候組件不需要重新渲染,next.js服務(wù)端渲染等等...
一直有一個想法就是重構(gòu)自己的博客,剛好這段時間放假,又剛好學(xué)習(xí)了react,于是就有了這個項目。
項目地址:https://github.com/k-water/re...技術(shù)棧
如果覺得不錯的話,您可以點右上角 "Star" 支持一下 謝謝! ^_^
前端
react
react-redux
react-thunk
react-router
axios
eslint
maked
highlight.js
antd
es6/7/8
后臺
spring boot
此項目采用前后端分離的實現(xiàn),后臺接口基于RESTful規(guī)范設(shè)計,只提供數(shù)據(jù),前端負責(zé)路由跳轉(zhuǎn),權(quán)限限制,渲染數(shù)據(jù)等。PS:由于我是個前端er,所以這里主要講的是前端。
實現(xiàn)的功能[x] admin增刪查改博客
[x] 博客標簽
[x] 博客內(nèi)容markdown
[x] 博客內(nèi)容頁展示目錄
[x] 返回頂部
[x] markdown代碼高亮
[x] 用戶登錄注冊
[x] 用戶評論
[x] 響應(yīng)式
TODO[ ] 博客分類
[ ] 點擊標簽搜索相關(guān)博客
[ ] 優(yōu)化首頁側(cè)邊欄
[ ] 完善歸檔
效果預(yù)覽 首頁 內(nèi)容頁 用戶登錄 用戶評論 后臺管理 個人總結(jié) markdown渲染在前端渲染markdown的時候遇到了一點問題,相關(guān)的包很多,但是各種包解析的結(jié)果都有差異,react周邊社區(qū)推薦的是 react-markdown,使用方法也很簡單
import ReactMarkdown from "react-markdown" const input = "# This is a header And this is a paragraph" ReactDOM.render(, document.getElementById("container") )
但是發(fā)現(xiàn)react-markdown對表格的支持不太友好,最后采用了marked,結(jié)合highlight.js對代碼部分實現(xiàn)高亮
import marked from "marked" import hljs from "highlight.js" componentWillMount() { marked.setOptions({ highlight: code => hljs.highlightAuto(code).value }) }
最后解析出來的是一個字符串,還需要將它插入dom中,由于安全問題,React不提倡將字符串直接插入dom中,但React保留了一個API,可以這樣做:
React組件化
react的組件由dom視圖和state組成,state是數(shù)據(jù)中心,它的狀態(tài)決定著視圖的狀態(tài)。react只負責(zé)UI的渲染,與其他框架監(jiān)聽數(shù)據(jù)動態(tài)改變dom不同,react采用setState來控制視圖的更新。setState會自動調(diào)用render函數(shù),觸發(fā)視圖的重新渲染,如果僅僅只是state數(shù)據(jù)的變化而沒有調(diào)用setState,并不會觸發(fā)更新。說到組件,就必須了解react組件的生命周期,官方的圖解如下:
關(guān)于這部分的解釋網(wǎng)上有很多,可以自行查閱。而我在開發(fā)過程用的最多的就是
componentWillMount()
componentDidMount()
shouldComponentUpdate(nextProps, nextState)
這幾個鉤子函數(shù)了,關(guān)于性能優(yōu)化,可以在shouldComponentUpdate上作文章,由于shouldComponentUpdate默認返回true,簡單的方法可以通過比較更新前后的數(shù)據(jù)結(jié)構(gòu)是否相同來判斷組件是否需要重新渲染,這時候就可以采用immutable.js了。
組件之間通信react是單向數(shù)據(jù)流,自上而下的傳遞數(shù)據(jù)。解決復(fù)雜組件之間通信的方法有很多。一般父子組件通信是最簡單的,父組件將一個回調(diào)函數(shù)傳遞給子組件,子組件通過this.props直接調(diào)用該函數(shù)與父組件通信。
如果組件之間嵌套很深,可以使用上下文getChildContext來傳遞信息,這樣在不需要將函數(shù)一層層往下傳,任何一層的子級都可以通過this.context直接訪問,react-redux內(nèi)部實現(xiàn)就是利用此方法。
兄弟組件之間無法直接通信,它們需要利用同一層的上級作為中轉(zhuǎn)站。
Reduxredux不是必須的,如果不是復(fù)雜的組件通信,邏輯簡單,用context就行。redux并不是react特有的,其他框架也可以使用redux。當初為了學(xué)習(xí)redux花費了不少時間,一開始并不理解redux中間的操作,看了很多前輩們寫的文章才逐漸明白。簡單說說redux。
redux由三部分組成:store, reducer, action
store是一個對象,它主要由三個方法:
dispatch
用于action的分發(fā),當action傳入dispatch會立即執(zhí)行,有些時候我們不想它立刻觸發(fā),可以在createStore中使用middleware中間件對dispatch進行改造,例如redux-thunk,不過這是react-radux做的事了。
subscribe
顧名思義,監(jiān)聽器,監(jiān)聽state的變化,這個函數(shù)在store調(diào)用dispatch時會注冊一個listener監(jiān)聽state變化。
getState
獲取store中的state,當我們用action觸發(fā)reducer改變了state時,需要拿到新的state里面的數(shù)據(jù)。getState在兩個地方會用到,一是通過dispatch提交action后store需要拿到state里面的數(shù)據(jù),二是利用subscribe監(jiān)聽到state發(fā)生變化后調(diào)用它來獲取新的state數(shù)據(jù)。
說了這么多,store的核心代碼其實很短:
/** * 應(yīng)用觀察者模式 * @param {Object} state * @param {Function} reducer */ function createStore(reducer) { let state = null const listeners = [] const subscribe = listener => listeners.push(listener) const getState = () => state const dispatch = action => { // 覆蓋原對象 state = reducer(state, action) listeners.forEach(listener => listener()) } // 初始化state dispatch({}) return { getState, dispatch, subscribe } }
另一部分,reducer是一個純函數(shù)(pure function),它接收一個state和action作為參數(shù),根據(jù)action的type返回一個新的state,如果傳入的action type沒有匹配到,則返回默認的state,簡單實現(xiàn)如下:
function reducer(state, action) { if (!state) { return { title: { text: "water make redux", color: "red" }, content: { text: "water make redux", color: "green" } } } switch (action.type) { case "UPDATE_TITLE_TEXT": return { ...state, title: { ...state.title, text: action.text } } case "UPDATE_TITLE_COLOR": return { ...state, title: { ...state.title, color: action.color } } default: return state } }
action比較簡單,它返回一個對象,其中type屬性是必須的,同時也可以傳入一些其他的數(shù)據(jù)。
使用例子如下:
/ 生成store const store = createStore(reducer) let oldState = store.getState() // 監(jiān)聽數(shù)據(jù)變化重新渲頁面 store.subscribe(() => { const newState = store.getState() renderApp(newState, oldState) oldState = newState }) // 首次渲染頁面 renderApp(store.getState()) store.dispatch({ type: "UPDATE_TITLE_TEXT", text: "water is fighting" }) store.dispatch({ type: "UPDATE_TITLE_COLOR", color: "#f00" })React-redux
react-redux則是對redux做了封裝,可以在react中直接使用,并且提供了Provider和connect。
Provider是一個組件,它接受store作為props,然后通過context往下傳,這樣react中任何組件都可以通過context獲取store。
connect是一個函數(shù),也是一個高階組件(HOC),通過傳入state和dispatch返回一個新的組件,它的寫法是如下:
connect(mapStateToProps, mapDispatchToProps, mergeProps, options)(component)
也可以采用裝飾器的寫法,這需要babel的支持:
@connect( state, { func } )
具體的不多介紹,迷你實現(xiàn)可以看看這個項目:https://github.com/k-water/ma...
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/107245.html
摘要:前言此項目是基于螞蟻金服開源的之上,用全家桶的進行再次開發(fā)的,項目已經(jīng)開源,項目地址在上。項目地址開源不易,如果覺得該項目不錯或者對你有所幫助,歡迎到上給個,謝謝。 showImg(https://segmentfault.com/img/remote/1460000017116745); 前言 此 blog-react-admin 項目是基于 螞蟻金服開源的 ant design p...
摘要:希望幫助更多的前端愛好者學(xué)習(xí)。前端開發(fā)者指南作者科迪林黎,由前端大師傾情贊助。翻譯最佳實踐譯者張捷滬江前端開發(fā)工程師當你問起有關(guān)與時,老司機們首先就會告訴你其實是個沒有網(wǎng)絡(luò)請求功能的庫。 前端基礎(chǔ)面試題(JS部分) 前端基礎(chǔ)面試題(JS部分) 學(xué)習(xí) React.js 比你想象的要簡單 原文地址:Learning React.js is easier than you think 原文作...
摘要:單頁博客應(yīng)用編寫總結(jié)很久之前就想寫一個博客應(yīng)用在一開始想要直接用和模板直接寫但是暑假一開始的時候不小心入了的坑所以就一不做二不休直接用寫那既然用了不寫個單頁應(yīng)用也過意不去了不前前后后寫了將近兩個星期現(xiàn)在看來這其實是一個很容易的應(yīng)用但是鑒于 React-Express單頁博客應(yīng)用編寫總結(jié) 很久之前就想寫一個博客應(yīng)用.在一開始想要直接用express和ejs模板直接寫, 但是暑假一開始的時...
閱讀 1712·2021-10-28 09:32
閱讀 615·2021-09-24 09:47
閱讀 2937·2021-09-02 15:11
閱讀 2742·2021-08-09 13:46
閱讀 2893·2019-08-30 15:55
閱讀 1078·2019-08-30 15:54
閱讀 3313·2019-08-29 14:12
閱讀 816·2019-08-26 13:40