17、react之表格和分页组件、redux实际运用、react-redux实际应用。react虚拟DOM、优缺点、传值、组件的3个部分、ref、key、约束组件、setState、状态提声、生命周期及其2种组件实例、react.15.6.0.js源码外框、portal插槽、React.Children插槽、前端路由及触发条件
一、react之表格和分页组件
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title>react之表格和分页组件之React16.4.0版</title>
<script src="https://cdn.staticfile.org/react/16.4.0/umd/react.development.js"></script>
<script src="https://cdn.staticfile.org/react-dom/16.4.0/umd/react-dom.development.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
<style>
table{
border-collapse: collapse;
border: 1px solid #cbcbcb;
width:1000px;
background:#ffffff;
text-align:center;
}
table td,table th {
padding: 5px;
border: 1px solid #cbcbcb;
font-weight:100
}
div button{
color:gray;
margin-right:5px
}
.div{
color:red;
width:1000px;
padding:10px 0;
}
</style>
</head>
<body>
<div class="div">
<div>表格组件的功能:</div>
<div>(1)带过滤条件、</div>
<div>(2)表头可以合并、</div>
<div>(3)排序(暂不实现)、</div>
<div>(4)表体可以嵌套表格(暂不实现)、</div>
<div>(5)3种勾选(选择一项、选择一页、选择所有页)、</div>
<div>(6)翻页记忆、</div>
<div>(7)分页。</div>
</div>
<div id="container"></div>
</body>
<script type="text/babel">
const container = document.getElementById('container');
function TwoImg(props) {
var checkImg = {
yes: '',
no: '',
};
return (<img src={props.isTrue?checkImg.yes:checkImg.no} onClick={props.clickImg.bind(this)}/>)
}
class TablePage extends React.Component {
constructor(props) {
super(props);
this.state = {};
this.initThead = [];
this.dataIndexs = [];//把表头每一栏最后一项的dataIndex放到这里,作为表头和表体关联的依据
this.maxRowspan = 1;//表头最大跨行数
if(props.giveParentClickAllPages){
props.giveParentClickAllPages(this)
}
}
getDataIndexsAndMaxRowspan(columns,maxRowspan){
var that=this;
columns.forEach(function(item,index) {
if(item.children){
maxRowspan+=1;
if(maxRowspan>that.maxRowspan) that.maxRowspan = maxRowspan;
that.getDataIndexsAndMaxRowspan(item.children,maxRowspan)
}else{
that.dataIndexs.push(item.dataIndex)
}
});
}
addThisColumnRowspan(columns,maxRowspan){
var that=this;
columns.forEach(function(item,index) {
if(item.children){
item.thisRowspan=1;
that.addThisColumnRowspan(item.children,maxRowspan-1)
}else{
item.thisRowspan=maxRowspan;
}
});
}
addEachColumnColspan(columns){
var that=this;
columns.forEach(function(item) {
if(item.children){
if(!item.thisColspan)that.addThisColumnColspan(item,item.children)
}else{
item.thisColspan=1;
}
});
}
addThisColumnColspan(item,children){
var that=this;
children.forEach(function(child,index) {
var thisChildren= child.children;
if(thisChildren){
that.addThisColumnColspan(item,thisChildren);
that.addEachColumnColspan(children);
}else{
if(item.thisColspan){
item.thisColspan+=1;
}else{
item.thisColspan=1;
}
child.thisColspan=1;
}
});
}
getInitThead(){
for(var i=0;i<this.maxRowspan;i++){
this.initThead.push([]);
}
}
getCenterThead(columns,initThead,index){
var that=this;
columns.forEach(function(item,indexIn){
var itemTitle;
if(item.title){
itemTitle=item.title;
}else{
itemTitle=<TwoImg isTrue={that.props.checkSource.isSelectNowPage} clickImg={that.clickThisPage.bind(that,that.props.dataSource)}/>
}
initThead[index].push(<th key={indexIn+Math.random()} rowSpan={item.thisRowspan} colSpan={item.thisColspan} dataindex={item.dataIndex||''}>{itemTitle}</th>)
var children=item.children;
if(children){
that.getCenterThead(children,initThead,index+1)
}
})
}
getLastThead(thead,initThead){
var that=this;
initThead.forEach(function(item,index){
thead.push(<tr key={index}>{item}</tr>)
})
}
getTbody(dataSource,trBody){
var that=this;
dataSource.forEach(function(tr,index){
var trSingle=[];
for(var i=0;i<that.dataIndexs.length;i++){
var indexIn=that.dataIndexs[i];
var td;
if(indexIn === 'checkQC'){
td = <TwoImg isTrue={tr.state} clickImg={that.clickSingleItem.bind(that,tr,dataSource)}/>
}else{
td = tr[indexIn];
}
trSingle.push(<td key={indexIn}>{td}</td>)
}
trBody.push(<tr key={index}>{trSingle}</tr>);
});
}
componentWillUpdate(nextProps) {
this.signCheckbox(nextProps)
}
setAllState(){
this.props.checkboxClick({
allIncludedIds: this.props.checkSource.allIncludedIds,
allExcludedIds: this.props.checkSource.allExcludedIds,
isSelectNowPage: this.props.checkSource.isSelectNowPage,
isSelectAllPages: this.props.checkSource.isSelectAllPages,
textAllPages: this.props.checkSource.textAllPages,
})
}
clickAllPages(itemArray) {//所有页所有条目复选框被点击时执行的函数
if(this.props.checkSource.isSelectAllPages){
if(this.props.checkSource.allExcludedIds.length>0){
this.props.checkSource.isSelectAllPages = true;
this.props.checkSource.isSelectNowPage = true;
this.props.checkSource.textAllPages= '已启用,无排除项!';
itemArray.forEach(function (item) {
item.state = true;
});
}else if(this.props.checkSource.allExcludedIds.length==0){
this.props.checkSource.isSelectAllPages = false;
this.props.checkSource.isSelectNowPage = false;
this.props.checkSource.textAllPages= '未启用,无选择项!';
itemArray.forEach(function (item) {
item.state = false;
});
}
}else{
this.props.checkSource.isSelectAllPages = true;
this.props.checkSource.isSelectNowPage = true;
this.props.checkSource.textAllPages= '已启用,无排除项!';
itemArray.forEach(function (item) {
item.state = true;
});
}
this.props.checkSource.allExcludedIds = [];
this.props.checkSource.allIncludedIds = [];
this.setAllState()
}
clickThisPage(itemArray) {//当前页所有条目复选框被点击时执行的函数
//onClick={this.clickThisPage.bind(this,params.tableDatas)
var that = this;
this.props.checkSource.isSelectNowPage = !this.props.checkSource.isSelectNowPage;
itemArray.forEach(function (item) {
item.state = that.props.checkSource.isSelectNowPage;
if (item.state) {
that.delID(item[that.props.idKey], that.props.checkSource.allExcludedIds);
that.addID(item[that.props.idKey], that.props.checkSource.allIncludedIds);
} else {
that.delID(item[that.props.idKey], that.props.checkSource.allIncludedIds);
that.addID(item[that.props.idKey], that.props.checkSource.allExcludedIds);
}
});
if(this.props.checkSource.isSelectAllPages){
if(this.props.checkSource.isSelectNowPage && this.props.checkSource.allExcludedIds.length === 0){
this.props.checkSource.textAllPages = '已启用,无排除项!';
}else{
this.props.checkSource.textAllPages = '已启用,已排除'+ this.props.checkSource.allExcludedIds.length + '项!排除项的ID为:' + this.props.checkSource.allExcludedIds;
}
}else{
if(!this.props.checkSource.isSelectNowPage && this.props.checkSource.allIncludedIds.length === 0){
this.props.checkSource.textAllPages='未启用,无选择项!';
}else{
this.props.checkSource.textAllPages = '未启用,已选择' + this.props.checkSource.allIncludedIds.length + '项!选择项的ID为:' + this.props.checkSource.allIncludedIds;
}
}
this.setAllState()
}
clickSingleItem(item, itemArray) {//当前页单个条目复选框被点击时执行的函数
var that = this;
item.state = !item.state;
if (item.state) {
this.props.checkSource.isSelectNowPage = true;
this.addID(item[this.props.idKey], this.props.checkSource.allIncludedIds);
this.delID(item[this.props.idKey], this.props.checkSource.allExcludedIds);
itemArray.forEach(function (item) {
if (!item.state) {
that.props.checkSource.isSelectNowPage = false;
}
});
} else {
this.props.checkSource.isSelectNowPage = false;
this.addID(item[this.props.idKey], this.props.checkSource.allExcludedIds);
this.delID(item[this.props.idKey], this.props.checkSource.allIncludedIds);
}
if(this.props.checkSource.isSelectAllPages){
if(this.props.checkSource.isSelectNowPage && this.props.checkSource.allExcludedIds.length === 0){
this.props.checkSource.textAllPages = '已启用,无排除项!';
}else{
this.props.checkSource.textAllPages = '已启用,已排除'+ this.props.checkSource.allExcludedIds.length + '项!排除项的ID为:' + this.props.checkSource.allExcludedIds;
}
}else{
if(!this.props.checkSource.isSelectNowPage && this.props.checkSource.allIncludedIds.length === 0){
this.props.checkSource.textAllPages='未启用,无选择项!';
}else{
this.props.checkSource.textAllPages = '未启用,已选择' + this.props.checkSource.allIncludedIds.length + '项!选择项的ID为:' + this.props.checkSource.allIncludedIds;
}
}
this.setAllState()
}
signCheckbox(nextProps) {//标注当前页被选中的条目,在翻页成功后执行。
var that = this;
if(nextProps.checkSource.isSelectAllPages){
nextProps.checkSource.isSelectNowPage = true;
nextProps.dataSource.forEach(function (item) {
var thisID = item[nextProps.idKey];
var index = nextProps.checkSource.allExcludedIds.indexOf(thisID);
if (index > -1) {
item.state = false;
nextProps.checkSource.isSelectNowPage = false;
} else {
item.state = true;
}
});
}else{
nextProps.checkSource.isSelectNowPage = true;
nextProps.dataSource.forEach(function (item) {
var thisID = item[nextProps.idKey];
var index = nextProps.checkSource.allIncludedIds.indexOf(thisID);
if (index === -1) {
item.state = false;
nextProps.checkSource.isSelectNowPage = false;
} else {
item.state = true;
}
});
}
this.state.isSelectNowPage=nextProps.checkSource.isSelectNowPage;
}
addID(id, idArray) {
var index = idArray.indexOf(id);
if (index === -1) {
idArray.push(id);//如果当前页的单项既有勾选又有非勾选,这时勾选当前页,需要这个判断,以免重复添加
}
}
delID(id, idArray) {
var index = idArray.indexOf(id);
if (index > -1) {
idArray.splice(index, 1)
}
}
render() {
var that=this;
var thead=[];
var tbody=[];
var trBody=[];
var columns=this.props.columns;
var dataSource=this.props.dataSource;
this.initThead = [];
this.dataIndexs = [];
this.getDataIndexsAndMaxRowspan(columns,this.maxRowspan);
this.addThisColumnRowspan(columns,this.maxRowspan);
this.addEachColumnColspan(columns);
this.getInitThead();
this.getCenterThead(columns,this.initThead,0);
this.getLastThead(thead,this.initThead);
this.getTbody(dataSource,trBody);
return (
<div>
<table>
<thead>
{thead}
</thead>
<tbody>
{trBody}
</tbody>
</table>
</div>
)
}
}
//DevidePage可传入属性
//1、numOptions:每页显示多少项
//2、text:总项数前后的说明文字
class DevidePage extends React.Component {
constructor(props) {
super(props);
this.state = { };
}
componentDidMount(){
document.getElementById("inputQC").addEventListener("keydown", this.onKeyDown.bind(this))
}
componentDidUpdate(){
document.getElementById("inputQC").addEventListener("keydown", this.onKeyDown.bind(this))
}
componentWillUnmount(){
document.getElementById("inputQC").removeEventListener("keydown", this.onKeyDown.bind(this))
}
inputChange() {
var value = parseInt(this.refs.input.value);
var allPagesNum = this.props.divideSource.allPagesNum;
if(value < allPagesNum && value > 1) {
this.props.divideSource.inputValue = value
}else if(value >= allPagesNum) {
this.props.divideSource.inputValue = allPagesNum
}else{//包含 value <= 1和value=其它非数字字符
this.props.divideSource.inputValue = 1
}
}
clickButton(value){
var nowPageNum = null;
if(value === 'front'){
this.props.divideSource.nowPageNum--;
nowPageNum = this.props.divideSource.nowPageNum
}else if(value === 'back'){
this.props.divideSource.nowPageNum++;
nowPageNum = this.props.divideSource.nowPageNum
}else if(value === 'leap'){
this.inputChange();
nowPageNum = this.props.divideSource.inputValue
}else{
nowPageNum = value
}
this.refs.input.value = nowPageNum;
this.props.divideClick(nowPageNum,this.props.divideSource.eachPageItemsNum);
}
onKeyDown(event){
if(event.key === 'Enter'){
this.inputChange();
this.refs.input.value = this.props.divideSource.inputValue;
this.props.divideClick(this.props.divideSource.inputValue,this.props.divideSource.eachPageItemsNum);
}
}
pageNumLeap(){
var eachPageItemsNum = this.refs.select.value;
this.props.divideSource.eachPageItemsNum = eachPageItemsNum;
this.props.divideClick(1,eachPageItemsNum);
}
render() {
var numButton=[];
var allPagesNum = this.props.divideSource.allPagesNum;
var nowPageNum = this.props.divideSource.nowPageNum;
if (allPagesNum >= 1 && allPagesNum <= 10) {
for (var i = 1; i <= allPagesNum; i++) {
numButton.push(<button key={i} style={i==nowPageNum?{color:'red'}:{color:'gray'}} onClick={this.clickButton.bind(this,i)}>{i}</button>)
}
} else if (allPagesNum >= 11) {
if (nowPageNum > 8) {
numButton.push(<button key={1} onClick={this.clickButton.bind(this,1)}>{1}</button>);
numButton.push(<button key={2} onClick={this.clickButton.bind(this,2)}>{2}</button>);
numButton.push(<button key={3} onClick={this.clickButton.bind(this,3)}>{3}</button>);
numButton.push(<button key={'front'} disabled>{'...'}</button>);
numButton.push(<button key={nowPageNum-2} onClick={this.clickButton.bind(this,nowPageNum-2)}>{nowPageNum-2}</button>);
numButton.push(<button key={nowPageNum-1} onClick={this.clickButton.bind(this,nowPageNum-1)}>{nowPageNum-1}</button>);
numButton.push(<button key={nowPageNum} style={{color:'red'}} onClick={this.clickButton.bind(this,nowPageNum)}>{nowPageNum}</button>);
} else {
for (i = 1; i <= nowPageNum; i++) {
numButton.push(<button key={i} style={i==nowPageNum?{color:'red'}:{color:'gray'}} onClick={this.clickButton.bind(this,i)}>{i}</button>)
}
}
// 以上当前页的左边,以下当前页的右边
if (allPagesNum - nowPageNum >= 7) {
numButton.push(<button key={nowPageNum+1} onClick={this.clickButton.bind(this,nowPageNum+1)}>{nowPageNum+1}</button>);
numButton.push(<button key={nowPageNum+2} onClick={this.clickButton.bind(this,nowPageNum+2)}>{nowPageNum+2}</button>);
numButton.push(<button key={'back'} disabled>{'...'}</button>);
numButton.push(<button key={allPagesNum-2} onClick={this.clickButton.bind(this,allPagesNum-2)}>{allPagesNum-2}</button>);
numButton.push(<button key={allPagesNum-1} onClick={this.clickButton.bind(this,allPagesNum-1)}>{allPagesNum-1}</button>);
numButton.push(<button key={allPagesNum} onClick={this.clickButton.bind(this,allPagesNum)}>{allPagesNum}</button>);
} else {
for (var i = nowPageNum + 1; i <= allPagesNum; i++) {
numButton.push(<button key={i} onClick={this.clickButton.bind(this,i)}>{i}</button>)
}
}
}
var selectOption=[];
var numOptions=this.props.numOptions;
if(!numOptions){numOptions=[10,20,30,40,50]};
for(var i=0;i<numOptions.length;i++){
selectOption.push(<option value={numOptions[i]} key={i} >{numOptions[i]}</option>)
}
return (
<div style={{display:'block',display:"flex",width:"1000px",marginTop:"20px"}}>
<div style={{display:"flex"}}>
<button style={{marginRight:"5px"}} disabled={this.props.divideSource.nowPageNum===1} onClick={this.clickButton.bind(this,'front')}>上一页</button>
<div>{ numButton }</div>
<button disabled={this.props.divideSource.nowPageNum===this.props.divideSource.allPagesNum} onClick={this.clickButton.bind(this,'back')}>下一页</button>
</div>
<div style={{display:"flex", flex:1, justifyContent:"flex-end"}}>
<div style={{marginRight:"15px"}}>
<span>转到第</span>
<input id='inputQC' key={this.props.divideSource.nowPageNum==1?Math.random():'key'} type="text" style={{width:"30px",margin:"0 5px"}} ref="input" onChange={this.inputChange.bind(this)} onKeyDown={this.onKeyDown.bind(this,event)} defaultValue={this.props.divideSource.inputValue}/>
<span>页</span>
<button style={{margin:"0 5px"}} onClick={this.clickButton.bind(this,'leap')}>Go</button>
</div>
<div>
<span>每页显示</span>
<select style={{margin:"0 5px"}} ref="select" defaultValue={this.props.divideSource.eachPageItemsNum||10} onChange={this.pageNumLeap.bind(this)}>
{ selectOption }
</select>
<span>{(this.props.text&&this.props.text.unit)||"条"}</span>
</div>
<div>
<span>{this.props.text&&this.props.text.frontMoreText}</span>
<span>{(this.props.text&&this.props.text.totalText)||"共"}</span>
<span>{this.props.divideSource.allItemsNum||0}</span>
<span>{(this.props.text&&this.props.text.totalUnit)||"项"}</span>
<span>{this.props.text&&this.props.text.backMoreText}</span>
</div>
</div>
</div>
)
}
}
class WholePage extends React.Component {
constructor(props) {
super(props);
this.state = {
filter: {
input:'',
select:1
},
dataSource: [],
divideSource:{
nowPageNum: 1,
allPagesNum: 1,
allItemsNum: 1,
eachPageItemsNum: 10,
inputValue:1,
},
checkSource:{
allIncludedIds: [],
allExcludedIds: [],
isSelectAllPages: false,
isSelectNowPage: false,
textAllPages: '未启用,无选择项!',
},
};
};
componentWillMount() {
this.divideClick(1,10)
}
divideClick(nowPageNum,eachPageItemsNum) {
var data=[];
var allItemsNum = 193;
var nowPageNum = nowPageNum||1;
var eachPageItemsNum = eachPageItemsNum||10;
var allPagesNum = Math.ceil(allItemsNum/eachPageItemsNum);
for(var i=0;i<allItemsNum;i++){
var obj={
id:'id'+(i+1),
order:i+1,
name:'姓名'+(i+1),
age:(i+11)+'岁',
street:'第'+(i+3)+'大街',
building:(i+2)+'号楼',
number:((i+1)+'0'+5)+'室',
companyAddress:'公司地址'+(i+5),
companyName:'公司名称'+(i+6),
gender:(i%2===0?'男':'女')
};
data.push(obj)
};
var dataSource = data.slice((nowPageNum-1)*eachPageItemsNum,nowPageNum*eachPageItemsNum);
this.setState({
dataSource: dataSource,
divideSource: {
nowPageNum: nowPageNum,
allPagesNum: allPagesNum,
allItemsNum: allItemsNum,
eachPageItemsNum: eachPageItemsNum,
inputValue: nowPageNum,
},
})
}
checkboxClick(object) {
this.setState({
allIncludedIds: object.allIncludedIds,
allExcludedIds: object.allExcludedIds,
isSelectAllPages: object.isSelectAllPages,
isSelectNowPage: object.isSelectNowPage,
textAllPages: object.textAllPages,
})
}
getChildClickAllPages(that){
this.childRef=that;//把子组件的实例赋值给父组件的childRef属性。非常重要!!!
}
clickAllPages(dataSource){
this.childRef.clickAllPages(dataSource)//在父组件里调用子组件的方法。非常重要!!!
}
changeValue(key,event){
this.setState({
filter:{...this.state.filter,[key]:event.target.value}
})
}
render() {
var columns = [
{
title: '',
dataIndex: 'checkQC',
},
{
title: '序号',
dataIndex: 'order',
},
{
title: 'Name',
dataIndex: 'name',
},
{
title: 'Person',
children: [
{
title: 'Age',
dataIndex: 'age'
},
{
title: 'Family Address',
children: [
{
title: 'Street',
dataIndex: 'street'
},
{
title: 'Block',
children: [
{
title: 'Building',
dataIndex: 'building'
},
{
title: 'Door No.',
dataIndex: 'number'
},
],
},
],
},
],
},
{
title: 'Company',
children: [
{
title: 'Company Address',
dataIndex: 'companyAddress'
},
{
title: 'Company Name',
dataIndex: 'companyName'
},
],
},
{
title: 'Gender',
dataIndex: 'gender'
},
];
var {dataSource,divideSource,checkSource,filter}={...this.state};
return (
<div>
<div>以下是过滤示例</div>
<div style={{'border':'1px solid #cbcbcb','width':'980px','padding':'10px','margin':'6px 0'}}>
<div style={{'display':'flex'}}>
<div style={{'width':'212px'}}>
<label style={{'paddingRight':'4px'}}>输入框示例</label>
<input type='text' placeholder='请输入' onChange={this.changeValue.bind(this,'input')} style={{'border':'1px solid #cbcbcb','width':'100px'}}/>
</div>
<div style={{'width':'174px'}}>
<label style={{'paddingRight':'4px'}}>下拉框示例</label>
<select onChange={this.changeValue.bind(this,'select')}>
<option value={1}>1分钟</option>
<option value={5}>5分钟</option>
<option value={10}>10分钟</option>
<option value={30}>30分钟</option>
<option value={60}>60分钟</option>
</select>
</div>
<div style={{'width':'500px'}}>
<span>过滤条件为:{JSON.stringify(this.state.filter)}</span>
<button onClick={this.divideClick.bind(this,3,10)}>过滤</button>
<button onClick={this.divideClick.bind(this,1,10)}>刷新</button>
</div>
</div>
</div>
<div style={{'paddingBottom':'4px'}}>
<span style={{'paddingRight':'10px'}}><TwoImg isTrue={checkSource.isSelectAllPages && checkSource.allExcludedIds.length===0} clickImg={this.clickAllPages.bind(this,dataSource)}/></span>
<span>{checkSource.textAllPages}</span>
</div>
<TablePage
idKey='id'
columns={columns}
dataSource={dataSource}
checkSource={checkSource}
checkboxClick={this.checkboxClick.bind(this)}
giveParentClickAllPages={this.getChildClickAllPages.bind(this)}
/>
<DevidePage
divideSource={divideSource}
divideClick={this.divideClick.bind(this)}
/>
</div>
)
}
}
ReactDOM.render(<WholePage/>, container);
</script>
</html>
二、redux实际运用
1、createStore实际运用(伪代码)
function createStore(reducer, preloadedState, enhancer) {
return enhancer(createStore)(reducer, preloadedState);
}
createStore(
rootReducer,
preloadedState,
compose(functionA,functionB,functionC,functionD)//enhancer
)
function functionA(createStore3) {//7、接收functionB的返回值createStore3
return function createStore4(reducer4, preloadedState4, enhancer4) {//8、返回值为createStore4
//9、实际执行createStore4(reducer, preloadedState),此处加工参数reducer, preloadedState,传给下面的createStore3
var store3=createStore3(reducer3, preloadedState3, enhancer3);
//16、此处对createStore3的返回值store3进行加工,下面return的是createStore4的返回值,也是最终的返回值
return {
dispatch: dispatch3,
subscribe: subscribe3,
getState: getState3,
replaceReducer: replaceReducer3
}
}
};
function functionB(createStore2) {//5、接收functionC的返回值createStore2
return function createStore3(reducer3, preloadedState3, enhancer3) {//6、返回值为createStore3
//10、此处加工参数,传给下面的createStore2
var store2=createStore2(reducer2, preloadedState2, enhancer2);
//15、此处对createStore2的返回值store2进行加工,下面return的是createStore3的返回值
return {
dispatch: dispatch2,
subscribe: subscribe2,
getState: getState2,
replaceReducer: replaceReducer2
}
}
};
function functionC(createStore1) {//3、接收functionD的返回值createStore1
return function createStore2(reducer2, preloadedState2, enhancer2) {//4、返回值为createStore2
//11、此处加工参数,传给下面的createStore1
var store1=createStore1(reducer1, preloadedState1, enhancer1);
//14、此处对createStore1的返回值store1进行加工,下面return的是createStore2的返回值
return {
dispatch: dispatch1,
subscribe: subscribe1,
getState: getState1,
replaceReducer: replaceReducer1
}
}
};
function functionD(createStore0) {//1、实际执行functionD(createStore)
return function createStore1(reducer1, preloadedState1, enhancer1) {//2、返回值为createStore1
//12、此处加工参数,传给下面的createStore0
var store0=createStore0(reducer0, preloadedState0, enhancer0);
//13、此处对createStore0的返回值store0进行加工,下面return的是createStore1的返回值
return {
dispatch: dispatch0,
subscribe: subscribe0,
getState: getState0,
replaceReducer: replaceReducer0
}
}
};
2、compose实际运用(包裹但不执行其参数)
var arrayFunction = [functionA, functionB, functionC, functionD];
function compose(arrayFunction) {
return arrayFunction.reduce(function (a, b) {
//1、包裹
//第1次执行时,a是functionA,b是functionB,执行结果为functionOne
//第2次执行时,a是functionOne,b是functionC,执行结果为functionTwo
//第3次执行时,a是functionTwo,b是functionD,执行结果为functionThree
//2、给functionThree传参并执行
//第1次执行时,functionThree(createStore0),functionD执行,返回createStore1
//第2次执行时,functionTwo(createStore1),functionC执行,返回createStore2
//第3次执行时,functionOne(createStore2),functionB执行,返回createStore3
//第4次执行时,functionA(createStore3),返回createStore4
//3、给createStore4传参并执行,进而给createStore3、createStore2、createStore1、createStore0传参并执行
//4、给createStore0的返回值加工,进而给createStore1、createStore2、createStore3、createStore4的返回值加工,生成最终store
return function () {
return a(b.apply(undefined, arguments));
};
})
}
compose(arrayFunction)(createStore)(reducer, preloadedState);
//第一次循环后的结果如下
function functionOne() {
return functionA(functionB.apply(undefined, arguments));
}
//第二次循环后的结果如下
function functionTwo() {
return functionOne(functionC.apply(undefined, arguments));
}
//第三次循环后的结果如下
function functionThree() {
return functionTwo(functionD.apply(undefined, arguments));
}
//functionThree实际上就是enhancer,通过functionThree(createStore)(reducer, preloadedState)起作用。
3、createStore实际运用(真实案例)
import { createStore, applyMiddleware, compose } from 'redux';
import reduxThunk from 'redux-thunk';//从UI组件直接dispatch action。它的主要思想是扩展action,使得action从只能是一个对象变成还可以是一个函数。
import rootReducer from 'reducers/index';
import DevTools from 'containers/DevTools';
export default function configureStore(preloadedState) {
const store = createStore(
rootReducer,
preloadedState,
compose(
applyMiddleware(reduxThunk),
DevTools.instrument()
)
)
return store
}
4、相关源码解析
(1)createStore
function createStore(reducer, preloadedState, enhancer) {
var _ref2;
if (typeof preloadedState === 'function' && typeof enhancer === 'undefined') {
enhancer = preloadedState;
preloadedState = undefined;
}
if (typeof enhancer !== 'undefined') {
if (typeof enhancer !== 'function') {
throw new Error('Expected the enhancer to be a function.');
}
return enhancer(createStore)(reducer, preloadedState);////////////////////////
}
if (typeof reducer !== 'function') {
throw new Error('Expected the reducer to be a function.');
}
var currentReducer = reducer;
var currentState = preloadedState;
var currentListeners = [];
var nextListeners = currentListeners;
var isDispatching = false;
function ensureCanMutateNextListeners() {
if (nextListeners === currentListeners) {
nextListeners = currentListeners.slice();
}
}
function getState() {
return currentState;
}
function subscribe(listener) {
if (typeof listener !== 'function') {
throw new Error('Expected listener to be a function.');
}
var isSubscribed = true;
ensureCanMutateNextListeners();
nextListeners.push(listener);
return function unsubscribe() {
if (!isSubscribed) {
return;
}
isSubscribed = false;
ensureCanMutateNextListeners();
var index = nextListeners.indexOf(listener);
nextListeners.splice(index, 1);
};
}
function dispatch(action) {
if (!isPlainObject(action)) {
throw new Error('Actions must be plain objects. ' + 'Use custom middleware for async actions.');
}
if (typeof action.type === 'undefined') {
throw new Error('Actions may not have an undefined "type" property. ' + 'Have you misspelled a constant?');
}
if (isDispatching) {
throw new Error('Reducers may not dispatch actions.');
}
try {
isDispatching = true;
currentState = currentReducer(currentState, action);
} finally {
isDispatching = false;
}
var listeners = currentListeners = nextListeners;
for (var i = 0; i < listeners.length; i++) {
var listener = listeners[i];
listener();
}
return action;
}
function replaceReducer(nextReducer) {
if (typeof nextReducer !== 'function') {
throw new Error('Expected the nextReducer to be a function.');
}
currentReducer = nextReducer;
dispatch({ type: ActionTypes.INIT });
}
function observable() {
var _ref;
var outerSubscribe = subscribe;
return _ref = {
subscribe: function subscribe(observer) {
if (typeof observer !== 'object') {
throw new TypeError('Expected the observer to be an object.');
}
function observeState() {
if (observer.next) {
observer.next(getState());
}
}
observeState();
var unsubscribe = outerSubscribe(observeState);
return { unsubscribe: unsubscribe };
}
}, _ref[result] = function () {
return this;
}, _ref;
}
dispatch({ type: ActionTypes.INIT });
return _ref2 = {
dispatch: dispatch,
subscribe: subscribe,
getState: getState,
replaceReducer: replaceReducer
}, _ref2[result] = observable, _ref2;
}
(2)compose
function compose() {
for (var _len = arguments.length, funcs = Array(_len), _key = 0; _key < _len; _key++) {
funcs[_key] = arguments[_key];
}
if (funcs.length === 0) {
return function (arg) {
return arg;
};
}
if (funcs.length === 1) {
return funcs[0];
}
return funcs.reduce(function (a, b) {
return function () {
return a(b.apply(undefined, arguments));
};
});
}
(3)applyMiddleware
function applyMiddleware() {//这是一个加工dispatch的中间件
for (var _len = arguments.length, middlewares = Array(_len), _key = 0; _key < _len; _key++) {
middlewares[_key] = arguments[_key];
}
return function (createStore) {
return function (reducer, preloadedState, enhancer) {
var store = createStore(reducer, preloadedState, enhancer);
var _dispatch = store.dispatch;
var chain = [];
var middlewareAPI = {
getState: store.getState,
dispatch: function dispatch(action) {//此处为什么不是dispatch:store.dispatch
return _dispatch(action);
}
};
chain = middlewares.map(function (middleware) {
return middleware(middlewareAPI);//return reduxThunk(_ref)
});
_dispatch = compose.apply(undefined, chain)(store.dispatch);//_dispatch = (function (next){})(store.dispatch) ,这是dispatch的新定义。
return _extends({}, store, {//store里的dispatch,被这里的dispatch覆盖
dispatch: _dispatch
});
};
};
}
(4)combineReducers
function combineReducers(reducers) {
//下面是关于reducers的定义,它的key如fn1、fn2、fn3、fn4后来也成了state的key。此论依据“非常重要!”前面的代码
// finalReducers = reducers = {
// fn1 : function(){ return { key1 : "value1" } },
// fn2 : function(){ return { key2 : "value2" } },
// fn3 : function(){ return { key3 : "value3" } },
// fn4 : function(){ return { key4 : "value4" } },
// }
var reducerKeys = Object.keys(reducers);
var finalReducers = {};
for (var i = 0; i < reducerKeys.length; i++) {
var key = reducerKeys[i];
{
if (typeof reducers[key] === 'undefined') {
warning('No reducer provided for key "' + key + '"');
}
}
if (typeof reducers[key] === 'function') {
finalReducers[key] = reducers[key];
}
}
var finalReducerKeys = Object.keys(finalReducers);
var unexpectedKeyCache = void 0;
{
unexpectedKeyCache = {};
}
var shapeAssertionError = void 0;
try {
assertReducerShape(finalReducers);
} catch (e) {
shapeAssertionError = e;
}
return function combination() {//这个返回值就是createStore(reducer, preloadedState, enhancer)中的reducer
//下面是关于state的定义,它的key如fn1、fn2、fn3、fn4来自于reducers的key。此论依据“非常重要!”前面的代码
// nextState = state = { } = {
// fn1 : { key1 : "value1" },
// fn2 : { key2 : "value2" },
// fn3 : { key3 : "value3" },
// fn4 : { key4 : "value4" }
// }
var state = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
var action = arguments[1];
if (shapeAssertionError) {
throw shapeAssertionError;
}
{
var warningMessage = getUnexpectedStateShapeWarningMessage(state, finalReducers, action, unexpectedKeyCache);
if (warningMessage) {
warning(warningMessage);
}
}
var hasChanged = false;
var nextState = {};
for (var _i = 0; _i < finalReducerKeys.length; _i++) {
var _key = finalReducerKeys[_i];
var reducer = finalReducers[_key];
var previousStateForKey = state[_key];//非常重要!previousStateForKey可能为undefined
var nextStateForKey = reducer(previousStateForKey, action);//非常重要!可能执行reducer(undefined,action),即执行reducer(defaultState,action)。
if (typeof nextStateForKey === 'undefined') {
var errorMessage = getUndefinedStateErrorMessage(_key, action);
throw new Error(errorMessage);
}
nextState[_key] = nextStateForKey;
hasChanged = hasChanged || nextStateForKey !== previousStateForKey;
// const defaultState = {
// loading: true,
// };
// export default function accountReducer(state = defaultState, action) {
// switch (action.type) {
// case "isRegister":
// return {...state, loading: false }//action匹配成功,state的引用就改变了
// default:
// return state;//action匹配不成功,state的引用就不改变了
// }
// }
}
//遍历结束,一次性返回结果。
return hasChanged ? nextState : state;
};
}
(5)reduxThunk
//从UI组件直接dispatch action。它的主要思想是扩展action,使得action从只能是一个对象变成还可以是一个函数。
function reduxThunk(_ref) {//middleware(middlewareAPI)
var dispatch = _ref.dispatch;
var getState = _ref.getState;
return function (next) {//即return function(store.dispatch);next是dispatch的旧定义,即applyMiddleware函数的chain数组的下一项的执行结果。
return function (action) {//返回的函数是dispatch的新定义;其中action是参数,原本只能是对象,经改造后还可以是函数。
if (typeof action === 'function') {
return action(dispatch, getState);//把旧的dispatch、getState传进自定义函数参数里
}
return next(action);
};
};
}
(6)其它
function componentWillMount() {
const current = [
{ location: "222", url: "/" },
{ location: "333", url: "/carDetail" },
];
this.props.dispatch(menuChange(current))
}
export const menuChange = function (key) {
return function (dispatch) {
dispatch({
type: "menu_change",
key
})
}
}
(7)
(function webpackUniversalModuleDefinition(root, factory) {
if (typeof exports === 'object' && typeof module === 'object')
module.exports = factory();//common.js模块下执行,factory()的执行结果为null
else if (typeof define === 'function' && define.amd)
define([], factory);//require.js异步模块下执行
else if (typeof exports === 'object')
exports["ReduxThunk"] = factory();
else
root["ReduxThunk"] = factory();
})(
this,
function () {
return (function (modules) {
var installedModules = {};
function __webpack_require__(moduleId) {
if (installedModules[moduleId])
return installedModules[moduleId].exports;
var module = installedModules[moduleId] = {
exports: {},
id: moduleId,
loaded: false
};
modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
module.loaded = true;
return module.exports;
}
__webpack_require__.m = modules;
__webpack_require__.c = installedModules;
__webpack_require__.p = "";
return __webpack_require__(0);
})([function (module, exports, __webpack_require__) {
module.exports = __webpack_require__(1);
},function (module, exports) {
'use strict';
exports.__esModule = true;
exports['default'] = reduxThunk;//这是最终的注入
function reduxThunk(_ref) {
var dispatch = _ref.dispatch;
var getState = _ref.getState;
return function (next) {// return function (store.dispatch)
return function (action) {
if (typeof action === 'function') {
return action(dispatch, getState);
}
return next(action);
};
};
}
}
])
}
);
三、react-redux实际应用
1、实际运用情形
react-redux把redux状态和React-Router路由连起来 注意:前端单页面应用,就是把状态和路由连起来,同时注意嵌套
class App extends Component {
render() {
return(
<div>
{this.props.children}
</div>
)
}
}
let routes = (
// Route 组件 无 path,直接渲染,有 path,匹配渲染
<Route component={App}>
// App 组件下的 <div>{this.props.children}</div> 用来放置 Public (为空)及其内部第一个被匹配的组件
<Route component={Public}>/* 以下内容渲染在 Public 组件下的 <div>{this.props.children}</div> 里*/
//以下写法一:
<Route path="/" component={index}/>
<Route path="/carDetail" component={carDetail}/> //通过browserHistory.push('/carDetail')渲染
//以下写法二:
<Route path="/" component={index}>
<Route path="/carDetail" component={carDetail}/>
</Route>
</Route>
// 如果 Public 下,没有组件匹配成功,那么 Public 为空,不显示;NotFound 会被渲染
<Route path="*" component={NotFound} />//通过browserHistory.push('carDetail')渲染
</Route>
)
ReactDOM.render(
<Provider store={store}>
<div>
<Router onUpdate={() => window.scrollTo(0, 0)} history={browserHistory} children={routes}></Router>
</div>
</Provider>,
document.getElementById('app')
)
document.body.removeChild(document.getElementById('svganimate'));
2、React-Redux 将所有组件分成两大类:UI 组件 和 容器组件
(1)UI 组件:负责 UI 的呈现,即 UI本身、UI接收数据 和 UI派发行为,由用户提供。
(2)容器组件:负责管理数据和业务逻辑,由 React-Redux 自动生成。
3、UI组件StartList的定义
class StartList extends Component {
render() {
const { detail,close } = this.props;
//openDialogue 和 closeDialogue没有被解构赋值
//this.props包含3部分:
//(1)mapStateToProps遍历的内容
//(2)mapDispatchToProps遍历的内容,如果该项缺失,那么将被dispatch取代
//(3)容器标签携带的内容
return (
<div>
<div>页面内容</div>
<button onClick={ this.props.openDialogue }>{detail}</button>
</div>
<div>
<div>弹窗内容</div>
<button onClick={ this.props.closeDialogue }>{close}</button>
</div>
)
}
}
4、mapStateToProps 方法定义。负责输入逻辑,即将 state 映射到 UI 组件的 props 里,
function mapStateToProps (state){
return {
todos: state.todos//todos: getVisibleTodos(state.todos, state.visibilityFilter)
}
}
5、mapDispatchToProps 方法定义。负责输出逻辑,即将用户对 UI 组件的操作映射成 Action。
(1)为对象时,有key和value,key为组件的属性,value为function,即action creator,返回值为action
const mapDispatchToProps = {
closeDialogue: function(ownProps){
return {
type: 'dialogue',
filter: ownProps.filter
}
}
}
const mapDispatchToProps = {
closeDialogue: (ownProps) => {
type: 'dialogue',
filter: ownProps.filter
};
}
(2)为函数时,返回值为对象,有key和value,key为组件的属性,value为function,执行过程中会dispatch action
function mapDispatchToProps (dispatch, ownProps) {//ownProps(容器组件的props对象)
return {
closeDialogue: function(){
dispatch({
type: 'dialogue',
isShow: false
});
},
openDialogue: function(){
//1、同步请求时,此处只有下面这些代码
//2、异步请求时,此处将 ownProps 里的部分数据作为参数,向后台发送请求,获取返回值 result,在成功的回调里执行下面这些代码
dispatch({
type: 'dialogue',
isShow: true
});
dispatch({
type: 'detail',
data: result
});
},
};
}
6、容器组件LastList的定义。接收数据和派发行为,即把 mapStateToProps 和 mapDispatchToProps 传给 StartList 。
const LastList = connect(
mapStateToProps,
mapDispatchToProps
)(StartList)
<LastList detail="详情" close="关闭"/>
//附:简单实例
class ProjectNumber extends Component {
state = {
visible: false
};
showModal(){
const projectId = this.props.record.id;
this.props.dispatch(getCarInfo(projectId));
this.setState({
visible: true,
});
};
render() {
return (
<div>
<span onClick={this.showModal.bind(this)}>{this.props.record}</span>
<Modal visible={this.state.visible}></Modal>
</div>
);
}
}
一、真实DOM和虚拟DOM
1、真实DOM,用户每次操作DOM(文档对象模型),都会引起BOM(浏览器对象模型)重新渲染
2、虚拟DOM,用户每次操作DOM(文档对象模型),都会改变虚拟DOM(堆内存的一个对象),最后一次性地渲染到BOM上。
二、React的缺点和优点
1、缺点:
(1)React本身只是一个V,不是一个完整的框架,不是一套完整的框架
(2)需要加上React-Router和React-Redux才能成为一套完整的框架
2、优点:
(1)单向数据流动;
(2)虚拟DOM取代物理DOM作为操作对象;
(3)用JSX语法取代HTML模板,在JavaScript里声明式地描述UI。
三、react组件间的传值
(1)react父组件向子组件传值:属性传参
(2)react子组件向父组件传值:实例是一个对象,在父组件定义一个函数,给这个函数绑定父组件的this,在这个函数的函数体里,将这个函数的参数赋值给父组件实例的一个属性;然后通过属性传参的方式把这个函数传给子组件。在子组件里,给这个函数传参并执行,实现子组件向父组件传值!
四、React 组件
1、React 组件基本上由 3 个部分组成——属性(props)、状态(state)以及生命周期方法
2、React 组件一旦接收到的参数(即props)或自身状态有所改变,React 组件就会执行相应的生命周期方法,最后渲染。
3、整个过程完全符合传统组件所定义的组件职责(“属性更新”与“状态改变”)。
4、以上内容来自《深入React技术栈》第18和30页。
五、ref可以设置回调函数
1、<input type="text" ref="myInput"/>
this.refs.myInput.value ="22"; //this.$refs.myInput.value ="22" 减少获取dom节点的消耗
2、ref属性可以设置为一个回调函数,这也是官方强烈推荐的用法;这个函数执行的时机为:
3、组件被挂载后,回调函数被立即执行,回调函数的参数为该组件的具体实例。
4、组件被卸载或者原有的ref属性本身发生变化时,回调也会被立即执行,此时回调函数参数为null,以确保内存泄露。
六、为什么不能用index做key?
1、react会根据key来决定是否重新构建该组件
2、删除和添加操作,会使一个组件使用另一个的组件的index,进而key,进而data
七、约束性组件和非约束性组件(受控组件和非受控组件)
1、约束性组件
<input type="text" value={this.state.name} onChange={this.handleChange} />
handleChange: function(e) {
this.setState({name: e.target.value});
}
//用户输入内容A>触发onChange事件>handleChange中设置state.name="A"渲染input使他的value变成A
2、非约束性组件
<input type="text" defaultValue="a" />//用户输入A -> input 中显示A
3、React 把 input,textarea 和 select 三个组件做了抽象和封装,他们有统一的 value 属性 和 onChange 事件。
<input type='text' name='intro' id='intro' value={this.state.email} onChange={this.handleEmail} />
<textarea type='text' name='intro' id='intro' value={this.state.intro} onChange={this.handleIntro} />
<textarea type='text' name='intro' id='intro' value={this.state.intro} onChange={this.handleIntro} />
4、checkbox改变的不是 value ,而是 checked 状态。
<input type='radio' name='gender' checked={this.state.male==='MALE'} onChange={this.handleGender} value='MALE' />
<input type='radio' name='gender' checked={this.state.male==='FEMALE'} onChange={this.handleGender} value='FEMALE' />
handleGender(e){
this.setState({
male:e.target.value
})
}
八、setState
1、this.setState接收两种参数
(1)对象+函数(可选):传入的对象浅层合并到新的state中
(2)函数+函数(可选):第一个函数接受两个参数,第一个是当前state,第二个是当前props,该函数返回一个对象,和直接传递对象参数是一样的,就是要修改的state;第二个函数参数是state改变后触发的回调。
2、this.setState何时同步何时异步?
(1)异步,由react控制的事件处理程序
(2)同步,react控制之外的事件中调用setState是同步更新的。比如原生js绑定的事件,setTimeout/setInterval等。
(3)大部分开发中用到的都是react封装的事件,比如onChange、onClick、onTouchMove等,这些事件处理程序中的setState都是异步处理的。
3、this.setState何时渲染
(1)死循环:this.setState执行时,会根据_compositeLifeCycleState是否为null,来决定是否重新渲染,因此在有的生命周期里,会产生死循环。
(2)只生效一次:this.setState最后执行队列时,先用变量获取并存储state,后来直接修改变量,最后把变量赋值给state,页面渲染。
handleClick() {
//const count = this.state.count 下面 this.setState 多次执行,但只生效一次。因为似乎存在此行代码。
this.setState({
count: count + 1
})
this.setState({
count: count + 1
})
}
九、状态提声(父组件的函数作为属性传给子组件)
1、在父组件的constructor中定义状态
2、在父组件的方法中执行this.setState({})
3、把父组件的方法作为属性fromParent传给子组件
4、在子组件的方法中加上this.props.fromParent(e.target.value);
5、触发子组件的事件,执行子组件的方法,改变父组件的状态
十、React组件生命周期
A、ES5写法
1、ES5写法之实例化时期
(1)getDefaultProps
(2)getInitialState
(3)componentWillMount//执行setState会合并到初始化状态中;获取从属性生成的状态
执行到这里,生命状态会被重置为null
(4)render//执行setState会发起updateComponent导致死循环
(5)componentDidMount//执行setState会导致更新;这是发起异步请求去API获取数据的绝佳时期
2、ES5写法之存在期
(1)componentWillReceiveProps//执行setState会合并到状态中;
执行到这里,生命状态会被重置为null
(2)shouldComponentUpdate//执行setState会发起updateComponent导致死循环
(3)componentWillUpdate//执行setState会发起updateComponent导致死循环
(4)render//执行setState会发起updateComponent导致死循环
(5)componentDidUpdate//可以有条件地执行setState
3、ES5写法之销毁时期
(1)componentWillUnmount//等待页面卸载,改变state没意义。
B、ES6写法(旧,React v16.3-)
1、ES6写法之实例化时期
(1)static
(2)constructor
(3)componentWillMount//执行setState会合并到初始化状态中;获取从属性生成的状态
执行到这里,生命状态会被重置为null
(4)render//执行setState会发起updateComponent导致死循环
(5)componentDidMount//执行setState会导致更新;这是发起异步请求去API获取数据的绝佳时期
2、ES6写法之存在期
(1)componentWillReceiveProps//执行setState会合并到状态中;
执行到这里,生命状态会被重置为null
(2)shouldComponentUpdate//执行setState会发起updateComponent导致死循环
(3)componentWillUpdate//执行setState会发起updateComponent导致死循环
(4)render//执行setState会发起updateComponent导致死循环
(5)componentDidUpdate//可以有条件地执行setState
3、ES6写法之销毁时期
(1)componentWillUnmount//等待页面卸载,改变state没意义。
C、ES6写法(新,React v16.4+)
1、ES6写法之实例化时期
(1)constructor
(2)static getDerivedStateFromProps//执行setState会合并到初始化状态中;获取从属性生成的状态
执行到这里,生命状态会被重置为null
(3)render//执行setState会发起updateComponent导致死循环
(4)componentDidMount//执行setState会导致更新;这是发起异步请求去API获取数据的绝佳时期
2、ES6写法之存在期
(1)static getDerivedStateFromProps//执行setState会合并到状态中;
执行到这里,生命状态会被重置为null
(2)shouldComponentUpdate//执行setState会发起updateComponent导致死循环
(3)render//执行setState会发起updateComponent导致死循环
(4)getSnapshotBeforeUpdate//执行setState会发起updateComponent导致死循环
(5)componentDidUpdate//可以有条件地执行setState
3、销毁时期
(1)componentWillUnmount//等待页面卸载,改变state没意义。
4、另外====
(1)getDerivedStateFromProps//返回一个对象来更新state,如果返回 null 则不更新任何内容。
十一、React组件生命周期实例(es5写法)
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title>React实例</title>
<script src="https://lib.baomitu.com/react/15.6.1/react.js"></script>
<script src="https://lib.baomitu.com/react/15.6.1/react-dom.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
</head>
<body>
<div id="example"></div>
<script type="text/babel">
var Button = React.createClass({
setNewNumber(number,event) {
console.log(number)
console.log(event)
this.setState({number: this.state.number + 1})
},
//一、以下实例化时期
//1.getDefaultProps
getDefaultProps() {
return { name: "计数器" };
},
//2.getInitialState
getInitialState() {
return{number: 0};
},
render() {
return (
<div>
<button onClick = {this.setNewNumber.bind(null,this.state.number,event)}>点击{this.props.name}</button>
<Text myNumber = {this.state.number}></Text>
</div>
);
}
})
var Text = React.createClass({
//一、以下实例化时期
//1.getDefaultProps
getDefaultProps() {
console.log("1.getDefaultProps 获取默认属性");
return { };
},
//2.getInitialState
getInitialState() {
console.log("2.getInitialState 获取初始状态");
return { };
},
//3.componentWillMount
componentWillMount() {
console.log("3.componentWillMount 此组件将要被渲染到目标组件内部");
},
//4.render在下方
//5.componentWillMount
componentDidMount() {
console.log("5.componentWillMount 此组件已经被渲染到目标组件内部");
},
//二、以下存在时期
//6.componentWillReceiveProps
componentWillReceiveProps() {
console.log("6.componentWillReceiveProps 此组件将要收到属性");
},
//7.shouldComponentUpdate
shouldComponentUpdate(newProps, newState) {
console.log("7.shouldComponentUpdate 组件是否应该被更新");
return true;
},
//8.componentWillUpdate
componentWillUpdate() {
console.log("8.componentWillUpdate 组件将要被更新");
},
//9.render在下方
//10.componentDidUpdate
componentDidUpdate() {
console.log("10.componentDidUpdate 组件已经更新完成");
},
//三、以下销毁时期
//11.componentWillUnmount
componentWillUnmount() {
console.log("11.componentWillUnmount 组件将要销毁");
},
render() {
return (
<div>
<h3>{this.props.myNumber}</h3>
</div>
);
}
})
ReactDOM.render(
<div>
<Button />
</div>,
document.getElementById('example')
);
</script>
</body>
</html>
十二、React组件生命周期实例(es6写法)
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title>React实例</title>
<script src="https://cdn.staticfile.org/react/16.4.0/umd/react.development.js"></script>
<script src="https://cdn.staticfile.org/react-dom/16.4.0/umd/react-dom.development.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
</head>
<body>
<div id="example"></div>
<script type="text/babel">
class Button extends React.Component {
name="计算器";//这样也行,二选一
state = {number: 0};//这样也行,二选一
constructor(props) {
super(props);
this.name="计算器";//这样也行,二选一
this.state = {number: 0};//这样也行,二选一
this.setNewNumber = this.setNewNumber.bind(this);
};
setNewNumber(number,event) {
console.log(number)
console.log(event)
this.setState({number: this.state.number + 1})
};
render() {
return (
<div>
<button onClick = {this.setNewNumber.bind(this,this.state.number,event)}>点击{this.name}</button>
<Text myNumber = {this.state.number}></Text>
</div>
);
}
}
class Text extends React.Component {
//一、以下实例化时期
//1.和2.constructor
constructor(props) {
super(props);
console.log("2.constructor 获取初始状态");
}
//3.componentWillMount
componentWillMount() {
console.log("3.componentWillMount 此组件将要被渲染到目标组件内部");
}
//4.render在下方
//5.componentWillMount
componentDidMount() {
console.log("5.componentWillMount 此组件已经被渲染到目标组件内部");
}
//二、以下存在时期
//6.componentWillReceiveProps
componentWillReceiveProps() {
console.log("6.componentWillReceiveProps 此组件将要收到属性");
}
//7.shouldComponentUpdate
shouldComponentUpdate(newProps, newState) {
console.log("7.shouldComponentUpdate 组件是否应该被更新");
return true;
}
//8.componentWillUpdate
componentWillUpdate() {
console.log("8.componentWillUpdate 组件将要被更新");
}
//9.render在下方
//10.componentDidUpdate
componentDidUpdate() {
console.log("10.componentDidUpdate 组件已经更新完成");
}
//三、以下销毁时期
//11.componentWillUnmount
componentWillUnmount() {
console.log("11.componentWillUnmount 组件将要销毁");
}
render() {
return (
<div>
<h3>{this.props.myNumber}</h3>
</div>
);
}
}
ReactDOM.render(
<div>
<Button />
</div>,
document.getElementById('example')
);
</script>
</body>
</html>
来源 https://www.runoob.com/try/try.php?filename=try_react_life_cycle2
十三、React.15.6.0源码外框
1、理解一
/**
* React v15.6.0
*/
(function (allFn) {
if (typeof exports === "object" && typeof module !== "undefined") {
module.exports = allFn()
} else if (typeof define === "function" && define.amd) {
define([], allFn)
} else {
var tempGlobal;
if (typeof window !== "undefined") {
tempGlobal = window
} else if (typeof global !== "undefined") {
tempGlobal = global
} else if (typeof self !== "undefined") {
tempGlobal = self
} else {
tempGlobal = this
}
tempGlobal.React = allFn()
}
})(function () {
var define, module, exports;
return (function outerFn(first, second, third) {
function recursion(number) {
if (!second[number]) {
if (!first[number]) {
var error = new Error("Cannot find module '" + number + "'");
throw error.code = "MODULE_NOT_FOUND", error
}
var module = second[number] = {
exports: {}
};
first[number][0].call(module.exports, function (key) {
var value = first[number][1][key];
return recursion(value ? value : key)
}, module, module.exports, outerFn, first, second, third)
}
return second[number].exports
}
for (var number = 0; number < third.length; number++) recursion(third[number]);
return recursion
//以上两行可以合并为下面一行
//return recursion(third[0]);
//同时下面的"(48)" 应当删掉
})({ 2: [function (_dereq_, module, exports) { console.log(_dereq_) }, { "25": 25, "30": 30 }], }, {}, [16])(16)
});
2、理解二
/**
* react v15.6.0
*/
(function (f) {
if (typeof exports === "object" && typeof module !== "undefined") {
module.exports = f()
} else if (typeof define === "function" && define.amd) {
define([], f)
} else {
var g;
if (typeof window !== "undefined") {
g = window
} else if (typeof global !== "undefined") {
g = global
} else if (typeof self !== "undefined") {
g = self
} else {
g = this
}
g.React = f()
}
})(function () {
var define, module, exports;
return (function e(t, n, r) {
///////////////////////////////////////////////////////////////////////////
function fn(num) {
//fn的作用:
//(1)定义接收容器。
//(2)填充接收容器,即给接收容器m下的exports属性和第2个实参对象n下的[num]属性下的exports属性,增加属性值(对象类型);
//(3)暴露接收容器的内容。
//t[num][0]的作用:填充接收容器。
//function(e){}的作用:填充接收容器时,获取所需依赖。
if (!n[num]) {
if (!t[num]) {/* 抛错 */}
var m = n[num] = { exports: {} };
t[num][0].call(m.exports, function (e) {
var number = t[num][1][e];
return fn(number ? number : e)
}, m, m.exports)
}
return n[num].exports//在react实例化的过程中,这行代码不但因获取依赖而多次执行,而且还因获取react实例而最后执行。
}
///////////////////////////////////////////////////////////////////////////
for (var i = 0; i < r.length; i++){fn(r[i])/* fn(16)第1次执行,执行结果没有变量接收 */};
return fn // 执行到这,整个逻辑就快结束了
})(
/* 第1个实参 */{1:[function(_dereq_, module, exports){},{"5":5,"6":6}],2:[],3:[],4:[],5:[],...},
/* 第2个实参 */{/* 存储依赖 */},
/* 第3个实参 */[16]
)(16)// fn(16)第2次执行,因为n[num]为真,所以直接返回n[num].exports并被挂在g.React上
});
十四、portal插槽
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title>React插槽实例</title>
<script src="https://cdn.staticfile.org/react/16.4.0/umd/react.development.js"></script>
<script src="https://cdn.staticfile.org/react-dom/16.4.0/umd/react-dom.development.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
</head>
<body>
<div id="container"></div>
<div id="outer"></div>
</body>
<script type="text/babel">
const container = document.getElementById('container');
const outer = document.getElementById('outer');
class Modal extends React.Component {
constructor(props) {
super(props);
this.div = document.createElement('div');
}
componentDidMount() {
outer.appendChild(this.div);
}
componentWillUnmount() {
outer.removeChild(this.div);
}
render() {
return ReactDOM.createPortal(
this.props.children,
this.div
);
}
}
class Parent extends React.Component {
constructor(props) {
super(props);
this.state = {clicks: 0};
this.handleClick = this.handleClick.bind(this);
}
handleClick() {
this.setState(state => ({
clicks: state.clicks + 1
}));
}
render() {
return (
<div onClick={this.handleClick}>
<div>{this.state.clicks}</div>
<div><button>Click</button></div>
<Modal>
<div>
<button className="modal">Click</button>
</div>
</Modal>
</div>
);
}
}
ReactDOM.render(<Parent />, container);
</script>
十五、React.Children插槽
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title>React插槽实例</title>
<script src="https://cdn.staticfile.org/react/16.4.0/umd/react.development.js"></script>
<script src="https://cdn.staticfile.org/react-dom/16.4.0/umd/react-dom.development.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
</head>
<body>
<div id="container"></div>
<div id="outer"></div>
</body>
<script type="text/babel">
class Parent extends React.Component {
constructor(props) {
super(props);
this.state = {clicks: 0};
this.handleClick = this.handleClick.bind(this);
}
handleClick(event) {
console.log(event)
this.setState(state => ({
clicks: state.clicks + 1
}));
}
render() {
var that = this;
return (
<div>
<div>{this.state.clicks}</div>
<div><button onClick={this.handleClick}>clicks</button></div>
<ul>
{
React.Children.map(this.props.children,function(item,index){
if(index !=1){
return <li onClick={that.handleClick}>{item}</li>
}else{
return <li onClick={that.handleClick}>{item}---{index+1}</li>
}
})
}
</ul>
</div>
);
}
}
ReactDOM.render(<Parent>
<span style={{cursor:'pointer',userSelect: 'none'}}>插槽一</span>
<span style={{cursor:'pointer',userSelect: 'none'}}>插槽二</span>
<span style={{cursor:'pointer',userSelect: 'none'}}>插槽三</span>
</Parent>, document.getElementById('container'));
</script>
</html>
十六、前端路由:url有变化,但不向后台发送请求,
1、它的作用是
(1)记录当前页面的状态;
(2)可以使用浏览器的前进后退功能;
2、hash模式帮我们实现这两个功能,因为它能做到:
(1)改变url且不让浏览器向服务器发出请求;
(2)监测 url 的变化;
(3)截获 url 地址,并解析出需要的信息来匹配路由规则。
3、history 模式改变 url 的方式会导致浏览器向服务器发送请求。
4、react前端路由触发条件
(1)<Link to={item.url}>{item.location}</Link>
(2)browserHistory.push('/aaa')