这篇文章尝试将 Vue 中一些常见的功能在 React 中实现一遍,如果你恰巧是 Vue 转 React,期待对你有些帮助。

条件渲染

  1. import React, { useState } from "react"
  2. export default function Vif (){
  3. const [ isShow, setIsShow ] = useState(true)
  4. const onToggleShow = () => {
  5. setIsShow(!isShow)
  6. }
  7. return (
  8. <div className="v-if">
  9. <button onClick={ onToggleShow }>切换</button>
  10. {/* 也可以用三目运算符 */}
  11. {/* { isShow ? <div>前端胖头鱼 显示出来啦</div> : null } */}
  12. {
  13. isShow && <div>前端胖头鱼 显示出来啦</div>
  14. }
  15. </div>
  16. )
  17. }
  1. import React, { useState } from "react"
  2. export default function VShow (){
  3. const [ isShow, setIsShow ] = useState(true)
  4. const onToggleShow = () => {
  5. setIsShow(!isShow)
  6. }
  7. return (
  8. <div className="v-show">
  9. <button onClick={ onToggleShow }>切换</button>
  10. {
  11. <div style={{ display: isShow ? '' : 'none' }}>前端胖头鱼 显示出来啦</div>
  12. }
  13. </div>
  14. )
  15. }

列表渲染

在 React 没有类型 Vue 的 v-for 指令,我们可以采用 map 遍历的方式实现类似功能

  1. import React, { useState } from "react"
  2. export default function VFor (){
  3. const [ list, setList ] = useState([
  4. {
  5. id: 1,
  6. name: '前端',
  7. },
  8. {
  9. id: 2,
  10. name: '后端',
  11. },
  12. {
  13. id: 3,
  14. name: 'android',
  15. },
  16. {
  17. id: 4,
  18. name: 'ios',
  19. },
  20. ])
  21. return (
  22. <div className="v-for">
  23. {
  24. list.map((item) => {
  25. return <div className="v-for-item" key={ item.id }>{ item.name }</div>
  26. })
  27. }
  28. </div>
  29. )
  30. }

计算属性

React没有计算属性,但是我们可以通过useMemo这个hook来实现,和Vue computed不太一样的地方在于,我们必须手动维护依赖

  1. import React, { useMemo, useState } from "react"
  2. export default function Computed (){
  3. const [ num1, setNum1 ] = useState(10)
  4. const [ num2, setNum2 ] = useState(10)
  5. const num3 = useMemo((a, b) => {
  6. return num1 + num2
  7. }, [ num1, num2 ])
  8. const onAdd = () => {
  9. setNum1(num1 + 10)
  10. }
  11. return (
  12. <div className="computed">
  13. <button onClick={ onAdd }>+10</button>
  14. <div>计算结果:{ num3 }</div>
  15. </div>
  16. )
  17. }

侦听器

React中要实现监听某些数据的变化执行响应的动作,可以使用useEffect

  1. import React, { useState, useMemo, useEffect } from "react"
  2. import './watch.css'
  3. export default function Watch() {
  4. const [fetching, setFetching] = useState(false)
  5. const [selects, setSelects] = useState([
  6. 'boy',
  7. 'girl'
  8. ])
  9. const [selectValue, setSelectValue] = useState('')
  10. const result = useMemo(() => {
  11. return fetching ? '请求中' : `请求结果: 选中${selectValue || '~'}`
  12. }, [ fetching ])
  13. const onSelect = (value) => {
  14. setSelectValue(value)
  15. }
  16. const fetch = () => {
  17. if (!fetching) {
  18. setFetching(true)
  19. setTimeout(() => {
  20. setFetching(false)
  21. }, 1000)
  22. }
  23. }
  24. useEffect(() => {
  25. fetch()
  26. }, [ selectValue ])
  27. return (
  28. <div className="watch">
  29. <div className="selects">
  30. {
  31. selects.map((item, i) => {
  32. return <button key={ i } onClick={ () => onSelect(item) }>{ item }</button>
  33. })
  34. }
  35. </div>
  36. <div className="result">
  37. { result }
  38. </div>
  39. </div>
  40. )
  41. }

Class 与 Style 绑定

有时候难免要给元素动态添加样式style,Vue和React都给我们提供了方便的使用方式。
在使用上基本大同小异:

相同点:
CSS property 名可以用驼峰式 (camelCase) 或短横线分隔 (kebab-case,记得用引号括起来) 来命名

不同点:

  1. Vue可以通过数组语法绑定多个样式对象,React主要是单个对象的形式(这点Vue也可以)
  2. React 会自动添加 ”px”(这点Vue不会自动处理) 后缀到内联样式为数字的属性,其他单位手动需要手动指定
  3. React样式不会自动补齐前缀。如需支持旧版浏览器,需手动补充对应的样式属性。Vue中当 v-bind:style 使用需要添加浏览器引擎前缀的 CSS property 时,如 transform,Vue.js 会自动侦测并添加相应的前缀。
  1. import React from "react"
  2. export default function Style (){
  3. const style = {
  4. width: '100%',
  5. height: '500px',
  6. }
  7. const style2 = {
  8. backgroundImage: 'linear-gradient(120deg, #84fab0 0%, #8fd3f4 100%)',
  9. borderRadius: '10px',
  10. }
  11. return (
  12. <div className="style" style={ { ...style, ...style2 } } ></div>
  13. )
  14. }
  1. import React, { useMemo, useState } from "react"
  2. import './class.css' // 此处样式与上面是一样的
  3. export default function Class (){
  4. const [ isActive, setIsActive ] = useState(false)
  5. const buttonText = useMemo(() => {
  6. return isActive ? '已选中' : '未选中'
  7. }, [ isActive ])
  8. const buttonClass = useMemo(() => {
  9. // 和Vue中不太一样的是我们需要手动join一下,变成'button active'形式
  10. return [ 'button', isActive ? 'active' : '' ].join(' ')
  11. }, [ isActive ])
  12. const onClickActive = () => {
  13. setIsActive(!isActive)
  14. }
  15. return (
  16. <div className={ buttonClass } onClick={onClickActive}>{ buttonText }</div>
  17. )
  18. }

基础 - 图1

provide/inject

Vue和React中对于全局状态的管理都有各自好的解决方案,比如Vue中的Vuex,React中的redux和Mobx,当然小型项目中引入这些有点大材小用了,有没有其他解决方案呢?

Vue中可以使用provide/inject
React中则可以使用Context

假设全局有有一个用户信息userInfo的变量,需要在各个组件中都能便捷的访问到,在Vue和React中该如何实现呢?

  1. import { createContext } from "react";
  2. export const UserInfoContext = createContext({
  3. userInfo: {
  4. name: ''
  5. }
  6. })
  1. import { UserInfoContext } from './context/index'
  2. function App() {
  3. return (
  4. <BrowserRouter>
  5. // 注意这里
  6. <UserInfoContext.Provider
  7. value={{ userInfo: { name: '前端胖头鱼' } }}
  8. >
  9. <div className="title">我是React栗子</div>
  10. <Routes>
  11. <Route path="/v-if" element={<Vif />} />
  12. <Route path="/v-show" element={<VShow />} />
  13. <Route path="/v-for" element={<VFor />} />
  14. <Route path="/computed" element={<Computed />} />
  15. <Route path="/watch" element={<Watch />} />
  16. <Route path="/style" element={<Style />} />
  17. <Route path="/class" element={<Class />} />
  18. <Route path="/slot" element={<Slot />} />
  19. <Route path="/nameSlot" element={<NameSlot />} />
  20. <Route path="/scopeSlot" element={<ScopeSlot />} />
  21. <Route path="/provide" element={<Provide />} />
  22. </Routes>
  23. </UserInfoContext.Provider>
  24. </BrowserRouter>
  25. );
  26. }
  1. import React, { useContext } from "react"
  2. import { UserInfoContext } from '../context/index'
  3. export default function Provide() {
  4. // 通过userContext,使用定义好的UserInfoContext
  5. const { userInfo } = useContext(UserInfoContext)
  6. return (
  7. <div class="provide-inject">{ userInfo.name }</div>
  8. )
  9. }

基础 - 图2

插槽

https://mp.weixin.qq.com/s/ABXZLhnWJrQoJgz01hI4zA