《React進(jìn)階之路》第九章示例 bbs-redux-reselect 解讀

對應(yīng)源碼項(xiàng)目地址

能學(xué)到些啥窿冯?

作為學(xué)習(xí)的例子,只有三個(gè)頁面芙粱,但是麻雀雖小圣蝎,五臟俱全羡铲。通過該例子可以學(xué)習(xí)到以下知識(shí)點(diǎn):

  1. 項(xiàng)目的目錄結(jié)構(gòu)設(shè)計(jì)最佳實(shí)踐弧岳。
  2. 項(xiàng)目的 state 設(shè)計(jì)和模塊設(shè)計(jì)技巧譬正。
  3. 異步獲取 API 數(shù)據(jù),以及將獲取到的數(shù)據(jù)展示到頁面上盛撑。

三個(gè)核心頁面

  1. 登錄頁面碎节。
  2. 帖子列表頁面,僅展示帖子的基本信息抵卫。
  3. 帖子詳情頁面狮荔,展示帖子的詳細(xì)內(nèi)容,包括用戶的評論列表介粘。

測試賬號(hào)

  • 該bbs內(nèi)置三個(gè)用戶
    • tom
    • jack
    • steve
  • 密碼都是:123456

state 設(shè)計(jì)

{
  app: {
    requestQuantity: 0, // 當(dāng)前應(yīng)用中正在進(jìn)行的 API 請求數(shù)
    error: null         // 應(yīng)用全局錯(cuò)誤信息
  },
  auth: {
    userId: '59e5d570fe26fff867fc94c0', // 當(dāng)前登錄用戶的 id
    username: 'jack'                    // 當(dāng)前登錄用戶的用戶名
  },
  ui: {
    addDialogOpen: false, // 用于新增帖子的對話框的顯示狀態(tài)
    editDialogOpen: false // 用于編輯帖子的對話框的顯示狀態(tài)
  },
  posts: {
    allIds: [],  // 維護(hù)數(shù)據(jù)的有序性
    byId: {}     // 根據(jù) id 獲取帖子的相關(guān)內(nèi)容
    }
  },
  comments: {
    byPost: {
      '5c10b55d34ce789876fc00ed': [] // 帖子 id 與該帖子下的評價(jià) id 的映射殖氏。
    },
    byId: {}                         // 根據(jù)評論 id 獲取到的該條評論相關(guān)內(nèi)容。
    }
  },
  users: {}
  }
}

state 解讀

一共六個(gè)子 state

  1. app:記錄應(yīng)用業(yè)務(wù)狀態(tài)數(shù)據(jù)
    • requestQuantity:當(dāng)前應(yīng)用中正在進(jìn)行的 API 請求數(shù)姻采。
    • error:應(yīng)用全局錯(cuò)誤信息雅采。
  2. auth:登錄認(rèn)證狀態(tài)
    • userId:當(dāng)前登錄用戶的 id。
    • username:當(dāng)前登錄用戶的用戶名。
  3. ui:UI 狀態(tài)數(shù)據(jù)
    • addDialogOpen:用于新增帖子的對話框的顯示狀態(tài)婚瓜。
    • editDialogOpen:用于編輯帖子的對話框的顯示狀態(tài)宝鼓。
  4. posts:帖子列表
    • allIds:保存帖子列表的 id,維護(hù)數(shù)據(jù)的有序性巴刻。
    • byId:以帖子 id 為 key 的列表席函,每一個(gè)子項(xiàng)為帖子的相關(guān)內(nèi)容。
  5. comments
    • byPost 保存以某一個(gè)帖子的 id 為 key 的冈涧、該帖子 id 下的評論列表 id茂附,即:帖子 id 與該帖子下的評價(jià) id 的映射。
    • byId 與 posts 下的 byId 類似督弓,該項(xiàng)是以評論 id 為 key 的列表营曼,每一個(gè)子項(xiàng)為評論相關(guān)的內(nèi)容。
  6. users:當(dāng)前頁面相關(guān)的用戶信息列表

模塊設(shè)計(jì)

對應(yīng) state 的設(shè)計(jì)愚隧,模塊設(shè)計(jì)基本上也出來了蒂阱,除了對應(yīng)上面的六個(gè)子 state 都有相應(yīng)的模塊之外,還有一個(gè) Redux 模塊狂塘。

Redux 模塊录煤,位于 redux/modules 目錄下,各個(gè)功能相關(guān)的 reducer荞胡、action types妈踊、action creators 都定義到一個(gè)文件中。各個(gè)功能的 reducer 又通過 redux/modules/index.js 合并成一個(gè)根 reducer泪漂,以供 react-redux 創(chuàng)建 store 并進(jìn)行統(tǒng)一管理廊营。

運(yùn)行時(shí)數(shù)據(jù)

首頁,即帖子列表頁面

看一下運(yùn)行起來的 state萝勤,僅保留了兩篇帖子的數(shù)據(jù)露筒。

{
  app: {
    requestQuantity: 0,
    error: null
  },
  auth: {
    userId: '59e5d570fe26fff867fc94c0',
    username: 'jack'
  },
  ui: {
    addDialogOpen: false,
    editDialogOpen: false
  },
  posts: {
    allIds: [
      '5c10b579a41380ad2bf95cfd',
      '5c0f145bd76a23857943793c'
    ],
    byId: {
      '5c10b579a41380ad2bf95cfd': {
        id: '5c10b579a41380ad2bf95cfd',
        title: 'fs',
        vote: 0,
        updatedAt: '2018-12-12T07:16:10.863Z',
        author: '59e5d570fe26fff867fc94c0'
      },
      '5c0f145bd76a23857943793c': {
        id: '5c0f145bd76a23857943793c',
        title: '',
        vote: 0,
        updatedAt: '2018-12-11T01:35:23.187Z',
        author: '59e5d570fe26fff867fc94c0'
      }
    }
  },
  comments: {
    byPost: {},
    byId: {}
  },
  users: {
    '59e5d22f6722f75272b3bbcf': {
      id: '59e5d22f6722f75272b3bbcf',
      username: 'tom'
    },
    '59e5d570fe26fff867fc94c0': {
      id: '59e5d570fe26fff867fc94c0',
      username: 'jack'
    }
  }
}

帖子詳情頁面

再看有三條評論的帖子,進(jìn)入詳情頁面時(shí)敌卓,state 的內(nèi)容慎式。帖子是 jack 發(fā)的,三條評論都是 tom 發(fā)的趟径,所以users有兩個(gè)用戶的信息瘪吏。

{
  app: {
    requestQuantity: 0,
    error: null
  },
  auth: {
    userId: '59e5d570fe26fff867fc94c0',
    username: 'jack'
  },
  ui: {
    addDialogOpen: false,
    editDialogOpen: false
  },
  posts: {
    allIds: [],
    byId: {
      '5c10b55d34ce789876fc00ed': {
        id: '5c10b55d34ce789876fc00ed',
        title: 'asd',
        content: 'ads',
        vote: 0,
        updatedAt: '2018-12-12T07:14:37.395Z',
        author: '59e5d570fe26fff867fc94c0'
      }
    }
  },
  comments: {
    byPost: {
      '5c10b55d34ce789876fc00ed': [
        '5c1215a60dbbbd7c0a640389',
        '5c12159eb50852373a0d1a9e',
        '5c12151f9536aad30f94f59f'
      ]
    },
    byId: {
      '5c1215a60dbbbd7c0a640389': {
        id: '5c1215a60dbbbd7c0a640389',
        content: 'sadf',
        updatedAt: '2018-12-13T08:17:42.783Z',
        author: '59e5d22f6722f75272b3bbcf'
      },
      '5c12159eb50852373a0d1a9e': {
        id: '5c12159eb50852373a0d1a9e',
        content: 'sad',
        updatedAt: '2018-12-13T08:17:34.272Z',
        author: '59e5d22f6722f75272b3bbcf'
      },
      '5c12151f9536aad30f94f59f': {
        id: '5c12151f9536aad30f94f59f',
        content: 'sa',
        updatedAt: '2018-12-13T08:15:27.067Z',
        author: '59e5d22f6722f75272b3bbcf'
      }
    }
  },
  users: {
    '59e5d570fe26fff867fc94c0': {
      id: '59e5d570fe26fff867fc94c0',
      username: 'jack'
    },
    '59e5d22f6722f75272b3bbcf': {
      id: '59e5d22f6722f75272b3bbcf',
      username: 'tom'
    }
  }
}

對應(yīng)源碼結(jié)構(gòu)(模塊設(shè)計(jì))

源碼結(jié)構(gòu)與state設(shè)計(jì)是相輔相成的。

│  index.js
│
├─components // 全局通用組件
│  ├─Header
│  │      index.js
│  │      style.css
│  │
│  ├─Loading
│  │      index.js
│  │      style.css
│  │
│  └─ModalDialog
│          index.js
│          style.css
│
├─containers
│  ├─App
│  │      index.js
│  │
│  ├─Home
│  │      index.js
│  │
│  ├─Login
│  │      index.js
│  │      style.css
│  │
│  ├─Post
│  │  │  index.js // Post 容器組件
│  │  │  style.css
│  │  │
│  │  └─components // Post 專用組件
│  │      ├─CommentList
│  │      │      index.js
│  │      │      style.css
│  │      │
│  │      ├─CommentsView
│  │      │      index.js
│  │      │      style.css
│  │      │
│  │      ├─PostEditor
│  │      │      index.js
│  │      │      style.css
│  │      │
│  │      └─PostView
│  │              index.js
│  │              style.css
│  │
│  └─PostList
│      │  index.js // PostList 容器組件
│      │  style.css
│      │
│      └─components // PostList 專用組件
│          ├─PostItem
│          │      index.js
│          │      style.css
│          │
│          └─PostsView
│                  index.js
│
├─images
│      like-default.png
│      like.png
│
├─redux
│  │  configureStore.js
│  │
│  └─modules
│          app.js
│          auth.js
│          comments.js
│          index.js  // Redux 的根模塊舵抹,僅將其余模塊的 reducer 合并成一個(gè)根 reducer肪虎。
│          posts.js
│          ui.js
│          users.js
│
└─utils
        AsyncComponent.js
        connectRoute.js
        date.js
        request.js  // 對 fetch 的封裝
        SHA1.js
        url.js  // API 配置

關(guān)于目錄結(jié)構(gòu)設(shè)計(jì)的最佳實(shí)踐,請看:React+Redux工程目錄結(jié)構(gòu)惧蛹,最佳實(shí)踐

redux 模塊

redux 模塊扇救,指的是在 redux/modules 下定義的模塊刑枝。

一個(gè) redux 模塊,不僅包含 action types迅腔、action creators装畅、reducers,還包含從該模塊獲取 state 數(shù)據(jù)的 selectors 函數(shù)沧烈。

selectors 函數(shù)的使命:

  • 封裝:對外提供數(shù)據(jù)接口掠兄,外部調(diào)用者不需要知道內(nèi)部實(shí)現(xiàn)細(xì)節(jié),也不用關(guān)心內(nèi)部 state 的具體結(jié)構(gòu)锌雀。
  • 解耦:內(nèi)部 state 結(jié)構(gòu)如果有變化蚂夕,修改對外接口即可,不會(huì)影響到外部調(diào)用方腋逆,降低模塊間依賴關(guān)系婿牍,最大限度解耦。

全局 selector 函數(shù)

當(dāng)需要從多個(gè)模塊的 state 中獲取數(shù)據(jù)時(shí)惩歉,最好的做法等脂,是在 redux/module/index.js 文件中定義全局 selector 函數(shù),該 selector 再通過各個(gè)模塊的 selector 獲取需要的數(shù)據(jù)撑蚌。這樣上遥,容器組件通過調(diào)用全局 selector 函數(shù),可以非常便利地對全局?jǐn)?shù)據(jù)進(jìn)行處理争涌。

《React進(jìn)階之路》第九章示例原書代碼

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末粉楚,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子第煮,更是在濱河造成了極大的恐慌解幼,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,948評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件包警,死亡現(xiàn)場離奇詭異,居然都是意外死亡底靠,警方通過查閱死者的電腦和手機(jī)害晦,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,371評論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來暑中,“玉大人壹瘟,你說我怎么就攤上這事■猓” “怎么了稻轨?”我有些...
    開封第一講書人閱讀 157,490評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長雕凹。 經(jīng)常有香客問我殴俱,道長政冻,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,521評論 1 284
  • 正文 為了忘掉前任线欲,我火速辦了婚禮明场,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘李丰。我一直安慰自己苦锨,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,627評論 6 386
  • 文/花漫 我一把揭開白布趴泌。 她就那樣靜靜地躺著舟舒,像睡著了一般。 火紅的嫁衣襯著肌膚如雪嗜憔。 梳的紋絲不亂的頭發(fā)上魏蔗,一...
    開封第一講書人閱讀 49,842評論 1 290
  • 那天,我揣著相機(jī)與錄音痹筛,去河邊找鬼莺治。 笑死,一個(gè)胖子當(dāng)著我的面吹牛帚稠,可吹牛的內(nèi)容都是我干的谣旁。 我是一名探鬼主播,決...
    沈念sama閱讀 38,997評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼滋早,長吁一口氣:“原來是場噩夢啊……” “哼榄审!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起杆麸,我...
    開封第一講書人閱讀 37,741評論 0 268
  • 序言:老撾萬榮一對情侶失蹤搁进,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后昔头,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體饼问,經(jīng)...
    沈念sama閱讀 44,203評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,534評論 2 327
  • 正文 我和宋清朗相戀三年揭斧,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了莱革。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,673評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡讹开,死狀恐怖盅视,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情旦万,我是刑警寧澤闹击,帶...
    沈念sama閱讀 34,339評論 4 330
  • 正文 年R本政府宣布,位于F島的核電站成艘,受9級特大地震影響赏半,放射性物質(zhì)發(fā)生泄漏贺归。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,955評論 3 313
  • 文/蒙蒙 一除破、第九天 我趴在偏房一處隱蔽的房頂上張望牧氮。 院中可真熱鬧,春花似錦瑰枫、人聲如沸踱葛。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,770評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽尸诽。三九已至,卻和暖如春盯另,著一層夾襖步出監(jiān)牢的瞬間性含,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,000評論 1 266
  • 我被黑心中介騙來泰國打工鸳惯, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留商蕴,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,394評論 2 360
  • 正文 我出身青樓芝发,卻偏偏與公主長得像绪商,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子辅鲸,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,562評論 2 349

推薦閱讀更多精彩內(nèi)容