亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

vue和react中props變化后如何修改state

 更新時間:2022年08月08日 09:56:53   作者:前端精髓  
這篇文章主要介紹了vue和react中props變化后如何修改state,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

vue和react中props變化后修改state

如果只想在 state 更改時重新計算某些數(shù)據(jù),比如搜索框案例。

vue

<template>
  <div>
  	<input type="text" v-model="filterText">
    <ul>
      <li v-for="item in filteredList" :key="item.id">
        {{ item.text }}
      </li>
    </ul>
  </div>
</template>
<script>
export default {
  props: {
    list: {
      type: Array,
      default: () => ([])
    }
  },
  data () {
    return {
      filterText: ''
    }
  },
  computed: {
    filteredList () {
      return this.list.filter(item => item.text.includes(this.filterText))
    }
  }
}
</script>

react

import React, { PureComponent } from 'react';
class Example extends PureComponent {
  state = {
    filterText: ''
  };
  handleChange = event => {
    this.setState({
      filterText: event.target.value
    })
  }
  render() {
    const filteredList = this.filter(this.props.list, this.state.filterText)
    return (
      <>
        <input
          type="text"
          onChange={this.handleChange}
          value={this.state.filterText} />
        <ul>
          {
            filteredList.map(
              item => <li key={item.id}>{item.text}</li>
            )
          }
        </ul>
      </>
    );
  }
}

如果你想在 prop 更改時“重置”某些 state,比如隨機默認值案例

vue

Vue提供了一種更通用的方式來觀察和響應(yīng)Vue實例上的數(shù)據(jù)變動:偵聽屬性 watch。

<template>
  <div>
    <input type="text" v-model="text">
  </div>
</template>
<script>
export default {
  props: {
    email: {
      type: String,
      default: ''
    }
  },
  data () {
    return {
      text: ''
    }
  },
  watch: {
    email: {
      immediate: true,
      handler (value) {
        this.text = value
      }
    }
  }
}
</script>

react

React生命周期 getDerivedStateFromProps 會在調(diào)用 render 方法之前調(diào)用,并且在初始掛載及后續(xù)更新時都會被調(diào)用。它應(yīng)返回一個對象來更新 state,如果返回 null 則不更新任何內(nèi)容。

父組件重新渲染時觸發(fā),請注意,不管原因是什么,都會在每次渲染前觸發(fā)此方法。

class Example extends Component {
  state = {
    text: ''
  };
  handleChange = (event) => {
    this.setState({
      text: event.target.value
    })
  }
  static getDerivedStateFromProps(nextProps, prevState) {
    if (prevState.email !== nextProps.email) {
      return {
        text: nextProps.email,
        email: nextProps.email
      }
    }
    return {text: prevState.text}
  }
  render() {
    return (
      <>
        <input
          type="text"
          onChange={this.handleChange}
          value={this.state.text} />
      </>
    );
  }
}

改進

直接復(fù)制 prop 到 state 是一個非常糟糕的想法。這兩者的關(guān)鍵在于,任何數(shù)據(jù),都要保證只有一個數(shù)據(jù)來源,而且避免直接復(fù)制它。

vue

<template>
  <div>
    <input type="text" :value="value" @input="handleInput">
  </div>
</template>
<script>
export default {
  props: {
    value: {
      type: String,
      default: ''
    }
  },
  methods: {
    handleInput (e) {
      this.$emit('input', e.target.value)
    }
  }
}
</script>
<template>
  <div id="app">
    <Example v-model="email"/>
    <button @click="handleClick">默認值</button>
  </div>
</template>
<script>
import Example from './components/Example.vue'
export default {
  components: {
    Example
  },
  data () {
    return {
      email: ''
    }
  },
  methods: {
    handleClick () {
      this.email = String(Math.random())
    }
  }
}
</script>

react

function Example (props) {
  return <input onChange={props.onChange} value={props.email} />;
}
class App extends React.Component {
  state = {
    email: ''
  }
  handleClick = () => {
    this.setState({
      email: String(Math.random())
    })
  }
  handleChange = (event) => {
    this.setState({
      email: event.target.value
    })
  }
  render() {
    return (
      <>
        <Example email={this.state.email} onChange={this.handleChange} />
        <div>
          <button onClick={this.handleClick}>默認值</button>
        </div>
      </>
    );
  }
}

react改變state必須知道的知識點

react可以通過this.state.xx的方式直接獲取state,但是當我們修改state的時候,往往有許多的坑。

1.不能直接修改state

組件修改state,并不會重新觸發(fā)render。例如:

?//錯誤
this.state.title='attend';
//正確
this.setState({title:'attend'});

2.state的更新是異步的

調(diào)用setState時,組件state并不會立即改變,只是把要修改的狀態(tài)放入事件隊列當中,為了彌補這個問題,使用另一種 setState() 的形式,接受一個函數(shù)。

這個函數(shù)將接收前一個狀態(tài)作為第一個參數(shù),應(yīng)用更新時的 props 作為第二個參數(shù),代碼如下:

//正確
this.setState((prevState, props)=>({
?? ?counter: prevState.counter + 1
}))

3.state的更新是一個合并的過程

當調(diào)用setState()修改組件的狀態(tài)時,只需要傳入發(fā)生改變的state,而不是完整的state,因為組件state的更新是一個合并的過程:

this.state = {
?? ?title: 'React',
?? ?content: 'React is an wondeful JS library'
}

當只需要修改title時,只需要將修改的title傳給setState即可:

this.setState({title:'ReactJs'});

react會合并最新的title到原來的狀態(tài),同時保留原來狀態(tài)的content,最終合并state為:

this.state = {
?? ?title: 'ReactJs',
?? ?content: 'React is an wondeful Js library'
}

state與不可變對象

react官方把state當成不可變對象,一方面直接修改this.state,組件并不會重新render;另一方面,state中包含的所有狀態(tài)都應(yīng)該是不可變的對象,state當中的某一個狀態(tài)發(fā)生變化時,應(yīng)該重新創(chuàng)建這個狀態(tài)對象,而不是直接修改原來的state狀態(tài),那么當狀態(tài)發(fā)生變化時,如何去創(chuàng)建新的狀態(tài)呢,我們根據(jù)狀態(tài)類型可以分為下面三種情況:

狀態(tài)類型為不可變類型

number、string、boolean、null、undefined

這種情況最簡單,因為狀態(tài)是不可變類型,所以直接給要修改的狀態(tài)賦一個新值即可,例如我們要修改的count為number型,title(string),success(boolean)三個狀態(tài):

this.setState({
?? ?count:1,
?? ?title:'React',
?? ?success:true
})

狀態(tài)類型為數(shù)組

假如有一個數(shù)組類型的狀態(tài)books,當向books中增加一本書時。

//方法一:使用preState,concat創(chuàng)建新數(shù)組
?this.setState((prevState)=>({
?? ?books: prevState.books.concat(['React Guide'])
}))
//方法二:ES6 spread syntax
this.setState(prevState=>({
?? ?books:[...prevState,'React Guide']
}))

當我們從books中截取部分元素作為新狀態(tài)時,可以用數(shù)組的slice方法:

this.setState(prevState=>({
?? ?books: prevState.books.slice(1,3);
}))

當從books中過濾部分元素后,作為新狀態(tài)時,可以使用filter方法:

this.setState(prevState=>({
?? ?books: prevState.books.filter(item=>{
?? ? ?return item!='React';
?? ?})
}))

【注意】不要使用push,pop,shift,unshift,splice等方法修改數(shù)組類型的狀態(tài),因為這些方法都是在原數(shù)組的基礎(chǔ)上修改的,而concat,slice,filter會返回一個新的數(shù)組。

狀態(tài)的類型是普通對象

(1) 使用es6的Object.assgin()方法

this.setState({
?? ?onwer: Object.assgin({},preState.onwer,{name:'Jason'});
})

(2) 使用對象擴展語法(Object spread properties):

this.setState(preState=>{
?? ?owner: {...preState.owner, name:'Jason'}
})

總結(jié)

創(chuàng)建新的狀態(tài)的關(guān)鍵是避免使用直接修改原對象的方法,這種方法在vue中稱為變異方法,而是使用可以返回一個新對象的方法,當然可以使用Immutable的JS庫(Immutable.js)實現(xiàn)類似的效果。

思考

為什么React推薦組件狀態(tài)的修改是不可變對象呢?

(1) 不可變對象的修改會返回一個新的對象,不用擔心原對象在不小心的情況下修改導(dǎo)致的錯誤,方便程序的管理和調(diào)試。

(2) 處于性能的考慮,對象組件的狀態(tài)是不可變對象時,在組件的shouldComponentUpdate方法中僅需要比較前后兩次狀態(tài)對象的引用就可以判斷狀態(tài)是否真的改變,從而避免不必要的render調(diào)用。

進階

除了以上方法改變react組件的狀態(tài)之外,我們還經(jīng)常會用到replaceState()改變組件的狀態(tài)。

replaceState()方法與setState()類似,但是方法只會保留nextState中狀態(tài),原state不在nextState中的狀態(tài)都會被刪除。

//使用語法:
replaceState(object nextState[, function callback])
  • nextState,將要設(shè)置的新狀態(tài),該狀態(tài)會替換當前的state。
  • callback,可選參數(shù),回調(diào)函數(shù)。該函數(shù)會在replaceState設(shè)置成功,且組件重新渲染后調(diào)用。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • vue動態(tài)路由匹配和路由懶加載多種方法詳解

    vue動態(tài)路由匹配和路由懶加載多種方法詳解

    這篇文章主要介紹了vue動態(tài)路由匹配和路由懶加載,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-06-06
  • vue2 vue3中使用Echarts詳細

    vue2 vue3中使用Echarts詳細

    這篇文章主要給大家介紹的是vue2 vue3中使用Echarts的相關(guān)資料,下面文章 會詳細介紹該內(nèi)容,感興趣的小伙伴不要錯過喲
    2021-09-09
  • Ant?Design?Vue?修改表格頭部樣式的詳細代碼

    Ant?Design?Vue?修改表格頭部樣式的詳細代碼

    這篇文章主要介紹了Ant?Design?Vue?修改表格頭部樣式,首先用到的是customHeaderRow這個API,類型是一個函數(shù),本文通過完整代碼給大家詳細講解,需要的朋友可以參考下
    2022-10-10
  • 使用element-ui實現(xiàn)行合并過程

    使用element-ui實現(xiàn)行合并過程

    這篇文章主要介紹了使用element-ui實現(xiàn)行合并過程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • Vue Router去掉url中默認的錨點#

    Vue Router去掉url中默認的錨點#

    vue-router默認hash模式——使用URL的hash來模擬一個完整的URL,于是當URL改變時,頁面不會重新加載。這篇文章主要介紹了Vue Router去掉url中默認的錨點#,需要的朋友可以參考下
    2018-08-08
  • Vue組件通信之父傳子與子傳父詳細講解

    Vue組件通信之父傳子與子傳父詳細講解

    這篇文章主要介紹了React中父子組件通信詳解,在父組件中,為子組件添加屬性數(shù)據(jù),即可實現(xiàn)父組件向子組件通信,文章通過圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-10-10
  • vue引用public目錄下文件的方式詳解

    vue引用public目錄下文件的方式詳解

    由于一些演示需要對一些簡單頁面進行配置,由于打包build后的vue項目基本已經(jīng)看不出原樣,因此需要創(chuàng)建一個文件,并在打包的時候不會進行編譯,所以文件放在public,下面這篇文章主要給大家介紹了關(guān)于vue引用public目錄下文件的相關(guān)資料,需要的朋友可以參考下
    2022-08-08
  • vue實現(xiàn)pdf文檔在線預(yù)覽功能

    vue實現(xiàn)pdf文檔在線預(yù)覽功能

    這篇文章主要為大家詳細介紹了vue實現(xiàn)pdf文檔在線預(yù)覽功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-11-11
  • vue中實現(xiàn)checkbox回顯效果

    vue中實現(xiàn)checkbox回顯效果

    這篇文章主要介紹了vue中實現(xiàn)checkbox回顯效果,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • Vue2?模版指令元素綁定事件執(zhí)行順序解析

    Vue2?模版指令元素綁定事件執(zhí)行順序解析

    這篇文章主要為大家介紹了Vue2?模版指令元素綁定事件執(zhí)行順序解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08

最新評論