這裏只是對本身的學習進行一次總結,也是爲了讓本身之後若是長時間不使用快速記憶起來的筆記,若是想要學習,仍是去官網看文檔比較好一些。、html
注意 下面的代碼的 script標籤的type類型都爲 「text/babel」vue
目錄react
1、無狀態組件es6
2、函數式聲明chrome
3、this以及事件對象瀏覽器
4、操做dom元素服務器
5、父子組件傳值babel
只是普通的變量是沒有狀態的, 只有組件纔有狀態或無狀態dom
react是單向數據流
狀態其實就是讓數據驅動視圖的響應式數據
1 let a = '這裏是div元素111'; 2 3 setTimeout(() => { 4 console.log('函數執行了'); 5 a = '改變以後的值' 6 },2000); 7 8 const element = ( 9 <div>{a}</div> 10 ); 11 12 ReactDOM.render( 13 element, 14 document.getElementById('app') 15 )
若是你學習過vue的話,那種雙向數據綁定用的 Object.defineProperty() 叫數據劫持,而後用了訂閱發佈(觀察者)那種模式,數據改變以後,會驅動着視圖去改變
react必需要經過 setState() 這個組件的實例化對象去調用去改變自身的state纔會去驅動視圖是改變的,因此上面的那種就算數據改變了,可是視圖是沒有進行改變的。
看下面的es6的這種寫法
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 this.state = { // 只有在state裏面的內容纔是有無狀態的 5 a: '這裏是div元素' 6 }; 7 8 setTimeout(() => { 9 this.setState({ // 這裏必要要用 setState這個方法來改變值纔會驅動視圖去改變 若是 this.state.a = 'sdfsd' 這樣是不會驅動視圖去改變的 10 a: '改變以後的值' 11 }) 12 }, 1000) 13 } 14 render() { 15 return ( 16 <div> 17 {this.state.a} 18 </div> 19 ); 20 } 21 } 22 23 ReactDOM.render( 24 <Element/>, 25 document.getElementById('app') 26 )
其實以前一直都是使用的變量去指代html元素,沒有使用過 class 那種,如今先使用ES5的寫法去寫一寫什麼是函數式聲明。
先看一下以前的寫法吧
1 const Head = ( 2 <h2>Head頭部</h2> 3 ); 4 const Foot = ( 5 <h2>底部</h2> 6 ); 7 const element = ( 8 <div> 9 { 10 Head 11 } 12 <h2>Element內容</h2> 13 { 14 Foot 15 } 16 </div> 17 ); 18 19 ReactDOM.render( 20 element, 21 document.getElementById('app') 22 )
函數式聲明是什麼樣子的呢
1 const Head = function () { 2 return ( 3 <h2>Head頭部</h2> 4 ) 5 }; 6 const Foot = function () { 7 return ( 8 <h2>底部</h2> 9 ) 10 }; 11 const Element = function() { 12 return ( 13 <div> 14 <Head/> 15 <h2>Element內容</h2> 16 <Foot/> 17 </div> 18 ); 19 }; 20 21 ReactDOM.render( 22 <Element></Element>, 23 document.getElementById('app') 24 )
能夠很清晰的看到他們兩個的區別,一個是使用變量來表明的,另外的一個是使用的組件形式表明的,開發中的寫法都是使用ES6的class繼承聲明來寫組件的
看一下函數聲明的傳值
1 const Head = function (props) { 2 return ( 3 <h2>{props.title}</h2> 4 ) 5 }; 6 const Foot = function (props) { 7 return ( 8 <h2>{props.con}</h2> 9 ) 10 }; 11 const Element = function() { 12 return ( 13 <div> 14 <Head 15 title={'頭部信息啊'} 16 /> 17 <h2>Element內容</h2> 18 <Foot 19 con={'底部的內容啊'} 20 /> 21 </div> 22 ); 23 }; 24 25 ReactDOM.render( 26 <Element></Element>, 27 document.getElementById('app') 28 )
這個樣子和vue實際上是挺像的,在標籤上面寫自定義的屬性,子組件就能夠接收到,等先這樣看看,後面會使用class寫的,class寫起來感受還容易理解一些。
注意: 函數式聲明組件
1. 組件的首字母必須是大寫
2. 不要使用H5新標籤
下面看一個狀態改變驅動視圖的例子
1 const Head = function (props) { 2 return ( 3 <h2>{props.title}</h2> 4 ) 5 }; 6 const Foot = function (props) { 7 return ( 8 <h2>{props.con}</h2> 9 ) 10 }; 11 12 class Element extends React.Component{ 13 constructor() { 14 super(); 15 this.state = { 16 title: '頭部信息', 17 con: '底部信息' 18 }; 19 20 setTimeout(() => { 21 this.setState({ 22 title: '頭部信息改變' 23 }) 24 },2000) 25 } 26 render() { 27 return ( 28 <div> 29 <Head 30 title={this.state.title} 31 /> 32 <h2>Element內容</h2> 33 <Foot 34 con={this.state.con} 35 /> 36 </div> 37 ); 38 } 39 } 40 41 ReactDOM.render( 42 <Element></Element>, 43 document.getElementById('app') 44 )
這樣就能夠和上面那種結合起來了,之後全部的組件都用class來寫了,就不用function那種形式了。
react組件中的this和事件對象的不一樣寫法有的寫法是須要綁定this的
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 } 5 6 btn() { 7 console.log(this); // 能夠看到這裏打印的是 undefined 8 } 9 10 render() { 11 return ( 12 <div> 13 <h1>title</h1> 14 <button onClick={this.btn}>按鈕</button> 15 </div> 16 ); 17 } 18 } 19 20 ReactDOM.render( 21 <Element></Element>, 22 document.getElementById('app') 23 )
看到上面打印的this是undefined,下面有幾種寫法能夠更正this指向實例化對象的
第一種 更正this和事件對象
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 this.btn = this.btn.bind(this); // 在這裏更正,官方推薦的也是這種寫法 5 } 6 7 btn(e) { 8 console.log(this); 9 console.log(e.target) 10 } 11 12 render() { 13 return ( 14 <div> 15 <h1>title</h1> 16 <button onClick={this.btn}>按鈕</button> 17 </div> 18 ); 19 } 20 } 21 22 ReactDOM.render( 23 <Element></Element>, 24 document.getElementById('app') 25 )
須要注意的是上面的那種寫法若是傳遞參數的話,那麼事件對象就是最後一個沒有對象參數的和js中的bind同樣
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 this.btn = this.btn.bind(this); // 在這裏更正,官方推薦的也是這種寫法 5 } 6 7 btn() { 8 console.log(this); 9 console.log(arguments) // 由於下面的函數傳遞了參數,這裏第二個參數纔是事件對象 10 } 11 12 render() { 13 return ( 14 <div> 15 <h1>title</h1> 16 <button onClick={this.btn(1)}>按鈕</button> 17 </div> 18 ); 19 } 20 } 21 22 ReactDOM.render( 23 <Element></Element>, 24 document.getElementById('app') 25 )
第二種 在行內bind
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 } 5 6 btn() { 7 console.log(this); 8 console.log(arguments) 9 } 10 11 render() { 12 return ( 13 <div> 14 <h1>title</h1> 15 <button onClick={this.btn.bind(this, 1)}>按鈕</button> // 直接寫到行內,在這裏改變this的指向,一樣的問題,那個參數的事件對象是最後一個 16 </div> 17 ); 18 } 19 } 20 21 ReactDOM.render( 22 <Element></Element>, 23 document.getElementById('app') 24 )
第三種 在一個函數裏面執行它
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 } 5 6 btn(e) { 7 console.log(this); 8 console.log(arguments) 9 } 10 11 render() { 12 return ( 13 <div> 14 <h1>title</h1> 15 <button onClick={(e) => { 16 this.btn(e,11); 17 }}>按鈕</button> 18 </div> 19 ); 20 } 21 } 22 23 ReactDOM.render( 24 <Element></Element>, 25 document.getElementById('app') 26 )
第四種 改變函數的寫法
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 } 5 6 btn = (e) => { 7 console.log(this); 8 console.log(e.target) 9 } 10 11 render() { 12 return ( 13 <div> 14 <h1>title</h1> 15 <button onClick={this.btn}>按鈕</button> 16 </div> 17 ); 18 } 19 } 20 21 ReactDOM.render( 22 <Element></Element>, 23 document.getElementById('app') 24 )
固然了這種方式若是要傳遞參數的話仍是須要bind的。
若是想要在react中操做dom元素的話,有幾種方法
第一種 直接使用js的方法
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 5 } 6 7 btn = () => { 8 const ele = document.querySelector('.h1'); 9 console.log(ele); 10 11 }; 12 13 render() { 14 return ( 15 <div> 16 <h1 className='h1'>title</h1> 17 <button onClick={this.btn}>按鈕</button> 18 </div> 19 ); 20 } 21 } 22 23 ReactDOM.render( 24 <Element></Element>, 25 document.getElementById('app') 26 )
第二種 經過事件對象
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 5 } 6 7 btn = (e) => { 8 console.log(e.target); 9 }; 10 11 render() { 12 return ( 13 <div> 14 <h1 className='h1'>title</h1> 15 <button onClick={this.btn}>按鈕</button> 16 </div> 17 ); 18 } 19 } 20 21 ReactDOM.render( 22 <Element></Element>, 23 document.getElementById('app') 24 )
第三種 ref
和vue同樣
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 5 } 6 7 btn = (e) => { 8 console.log(this.refs.abc) // 在這裏獲取它 9 }; 10 11 render() { 12 return ( 13 <div> 14 <h1 ref="abc">title</h1> // 這裏定義一個字符串的名字 15 <button onClick={this.btn}>按鈕</button> 16 </div> 17 ); 18 } 19 } 20 21 ReactDOM.render( 22 <Element></Element>, 23 document.getElementById('app') 24 )
不過最新版本的不建議上面的那種寫法了,也是ref建議下面的寫法
之後咱們就能夠用下面這種最經常使用的方式
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 5 } 6 7 btn = (e) => { 8 console.log(this.abc) 9 }; 10 11 render() { 12 return ( 13 <div> 14 <h1 ref={abc => {this.abc = abc}}>title</h1> 15 <button onClick={this.btn}>按鈕</button> 16 </div> 17 ); 18 } 19 } 20 21 ReactDOM.render( 22 <Element></Element>, 23 document.getElementById('app') 24 )
第四種 ReactDOM.findDOMNode
1 class Element extends React.Component{ 2 constructor() { 3 super(); 4 5 } 6 7 btn = (e) => { 8 console.log(this.abc) 9 10 console.log(ReactDOM.findDOMNode(this.abc)); 11 }; 12 13 render() { 14 return ( 15 <div> 16 <h1 ref={abc => {this.abc = abc}}>title</h1> 17 <button onClick={this.btn}>按鈕</button> 18 </div> 19 ); 20 } 21 } 22 23 ReactDOM.render( 24 <Element></Element>, 25 document.getElementById('app') 26 )
ref若是標記的是組件,那麼ref他就是組件,可是findDOMNode這種方式是獲取DOM元素的,就算裏面的參數是組件,也是得到DOM元素
核心思想
父組件給子組件傳值 直接在組件上面添加屬性就能夠了 子組件經過props訪問,獲得 實際上是構造函數實例化的時候傳過去了
子組件給父組件傳值 實際上是父組件給子組件傳一個函數,子組件調用的時候把 要傳遞的數據 放到 父組件傳遞過來的函數 的參數裏面,而後父組件再去作他本身的操做
Item子組件
1 class Item extends React.Component { 2 constructor (props) { 3 super(props); 4 this.state = { 5 sub: this.props.data 6 } 7 } 8 9 componentWillReceiveProps(nextProps) { 10 console.log('執行了嗎'); 11 console.log(nextProps); 12 } 13 14 render() { 15 console.log('子組件打印this'); 16 console.log(this); 17 console.log(this.props); 18 19 return ( 20 <div> 21 <p>Item組件</p> 22 <h1>{this.state.sub.res}</h1> 23 <input type="button" defaultValue="這裏" onClick={this.props.supFn.bind(this,'子組件參數')} /> 24 <input type="button" 25 defaultValue="22" 26 onClick={() => { 27 this.props.supFn('參數2') 28 }} 29 30 /> 31 </div> 32 ); 33 } 34 }
父組件
1 class Main extends React.Component { 2 constructor() { 3 super(); 4 this.superFn = this.superFn.bind(this); 5 6 } 7 state = { 8 data: { 9 res: '裏面的數據' 10 } 11 }; 12 13 superFn(a) { 14 console.log('父組件的函數'); 15 console.log(`${a}`); 16 let data = { 17 res: '修改事後' 18 }; 19 console.log(data); 20 21 this.setState({ 22 data: data 23 }) 24 } 25 render() { 26 return ( 27 <div> 28 <p>主頁面</p> 29 <p>{this.state.data.res}</p> 30 <Item supFn={this.superFn} data={this.state.data} aaa="傳值啊"></Item> 31 </div> 32 ); 33 } 34 } 35 36 ReactDOM.render( 37 <Main />, 38 document.getElementById('app') 39 )
要在服務器環境下打開文件
在谷歌瀏覽器輸入框中輸入
而後就可使用模塊導出導出功能了
1 // 2.js 2 const a = 1; 3 const b = 2; 4 5 export default {a, b};
1 <script type="module"> 2 // 要服務器環境下打開 http服務 3 // chrome//flags 開啓那個權限 4 import a from './2.js'; 5 console.log(a) 6 </script>
若是你看了個人文章有了一些收穫我會很是高興的,因爲能力有限,文章有的部分解釋的不到位,但願在之後的日子裏能慢慢提升本身能力,若是不足之處,還望指正。