自己做网站用中文为什么是乱码,杭州公司建设网站,网站竞价推广,地图 添加到网站组件component
一.概念
Element VS Component (元素与组件)
//不成文的约定:元素小写#xff0c;组件大写
const divReact.createElement(div,...)
这是一个React元素(小写)
const Div()React.createElement(div,...)
这是一个React组件(大写)
什么是组件?
能跟其他…组件component
一.概念
Element VS Component (元素与组件)
//不成文的约定:元素小写组件大写
const divReact.createElement(div,...)
这是一个React元素(小写)
const Div()React.createElement(div,...)
这是一个React组件(大写)
什么是组件?
能跟其他物件组合起来的物件就是组件组件并没有明确的定义。 就目前而言一个返回值是React元素的函数就是组件
Vue里data、methods、render组合成的一个对象(构造选项)就可以表示一个组件
React两种组件
1.函数组件
function Welcome(props){return h1Hello,{props.name}/h1
}
使用方法: Welcome namefrank/
2.类组件
class Welcome extends React.Component {render() {return h1Hello, {this.props.name}/h1}
}
使用方法: Welcome namefrank/
Welcome /会被翻译成什么
div /翻译为React.createElement(div) div是元素 Welcome /翻译为React.createElement(Welcome) Welcome是函数
工具: babel online把标签翻译成JS React.createElement的逻辑
React.createElement目前接收几种参数:
1.如果传入一个字符串div,则会创建一个div(虚拟DOM元素)
2.如果传入一个函数,则会调用该函数获取其返回值
3.如果传入一个类,则在类前面加个new(这会导致执行 constructor),获取一个组件对象然后调用对象的render方法获取其返回值
class Welcome extends React.Component{ constructor(){super()this.state{n:0}}render(){return divhi/div}
}
使用类
new Welcome()
二.使用React的2种组件
React2种组件的书写方式: class类组件和function函数组件。
例子
import React from react;
import ReactDOM from react-dom;function App() { return ( div classNameApp 爸爸Son / /div);
}class Son extends React.Component { constructor() { super(); this.state { n: 0};}add() {// this.state.n 1 为什么不行this.setState({ n: this.state.n 1 }); }render() {return ( div classNameSon儿子 n: {this.state.n}button onClick{() this.add()}1/button Grandson //div);}
}const Grandson () {
//声明一个state初始值为0,用n代表0用setN对0进行修改每次setN时都会得到一个新的n(不是改变原有的n)const [n, setN] React.useState(0); //析构写法return (div classNameGrandson孙子 n:{n}button onClick{() setN(n 1)}1/button/div);
};const rootElement document.getElementById(root);
ReactDOM.render(App /, rootElement); 如何使用 props(外部数据)
类组件和函数组件使用 props:
类组件直接读取属性this.props.xxx
函数组件直接读取参数props.xxx
外部数据props不能写只能读。
一.class类组件添加外部数据props
1 传字符串
传: 接收方添加messageForSon字符串
接收: {this.props.messageForSon}
2 传变量
传: 接收方添加messageForSon{11}
接收: {this.props.messageForSon}
例子:爸爸传props给Son
function App() {
const a10return (div classNameApp爸爸Son messageForSon儿子你好 / //Son messageForSon{a} //div);
}
class Son extends React.Component {render() {return (div classNameSon我是儿子我爸对我说「{this.props.messageForSon}」Grandson messageForGrandson孙贼你好 //div);}
}
二.函数组件
同样的要么使用表示字符串要么使用{}表示JS的表达式。
1 传字符串
传: 接收方添加messageForGrandson字符串
接收: 需要传一个参数props 和 接收{props.messageForGrandson}
React会自动把messageForGrandson字符串变成一个对象然后放到第一个参数props里props可自定义,叫x也行一般叫props。
2 传变量
传: 接收方添加messageForGrandson{a}
接收: {props.messageForGrandson}
例子:儿子传props给Grandson
class Son extends React.Component {render() {var a10return (div classNameSon我是儿子我爸对我说「{this.props.messageForSon}」Grandson messageForGrandson孙贼你好 / //Grandson messageForGrandson{a} / 添加messageForGrandson/div);}
}
const Grandson props { var a10return (div classNameGrandson我是孙子我爸对我说「{props.messageForGrandson}」//接收string或变量/div);
}; 如何使用 state(内部数据)
类组件和函数组件使用 state:
类组件用this.state读this.setState写 函数组件用useState返回数组第一项读第二项写
一.类组件如何使用内部数据state
关注数据时关注它的3个条件1.如何初始化2.如何读3.如何写
class Son extends React.Component {constructor() {super();this.state { //1.初始化n: 0};}add() {// this.state.n 1 为什么不行this.setState({ n: this.state.n 1 }); //2.写}render() {return (div classNameSon儿子 n: {this.state.n} //3.读button onClick{() this.add()}1/buttonGrandson //div);}
}
this.state.n 1为什么不行
因为Vue会监听n,而React根本不会监听n,所以就算n变化了React也不知道.
那怎么才能通知React我修改了数据呢
用setState。 数据不可变的思维:不要改变以前的对象要改的话你就产生一个新的对象。React理念就是数据不可变你如果要改变就产生一个新的对象新的对象去容纳新的东西然后传给setState。
add() {//this.state.n 1//this.setState(this.state); //这种虽然也生效但不要用违反了数据不可变的思维this.setState({ n: this.state.n 1 }); }
这就是在类组件里如何使用state:
1.在constructor的super下面进行初始化
2.读的时候用this.state.n
3.写的时候用this.setState,最好用一个新的对象而不是在原有的对象上修改。
牛X的前端用setState(函数)他一般会写成一个函数。
add() {//this.setState({ n: this.state.n 1 }); //console.log(this.state.n) //打印出0,setState不会马上改变n的值而是等一会console执行完后才去覆盖n的值。//所以说setState是异步的更新UI的过程不会立即执行而是过一会执行新手不理解易bug,老手为了避开这个问题这样写,避免混淆新的、旧的state。this.setState( state { //旧的nconst nstate.n1console.log(n)return { n } //新的n}) }
这种写法能更好的理解旧state、新state而且不容易出错避免异步造成的误解。 使用state时注意一个事情setState不会马上改state最好使用函数去读新的值它会让你更容易区分新旧state。
二.函数组件如何使用内部数据state
关注数据时关注它的3个条件1.如何初始化2.如何读3.如何写
const Grandson () {const [n, setN] React.useState(0); //1.初始值//你返回的这个数组的第1项就是用来读的第2项就是用来写的。一个读接口一个写接口就2项。名字可写但一般就叫n、setN。return (div classNameGrandson孙子 n:{n} //2.读button onClick{() setN(n 1)}1/button //3.写,setN永远不会改变n,它会产生一个新的n。/div);
};
类组件的setState会等一会再改变state函数组件的setN永远都不会改变n。
函数组件注意事项
1.跟类组件类似的地方
也要通过setX(新值)来更新UI
2.跟类组件不同的地方
没有this,一律用参数和变量搞不懂this的就可以用函数组件。
Vue 对比 React
两种编程模型
Vue: 我对数据的修改会直接映射到UI上。Vue监听了n的变化当n变化时Vue就把n出现的地方全部都变一遍。
React: 一开始给我个state{n:0},我把这个state变成一个UI。 如果你想改变n,要注意不能修改之前的n。应该直接新建一个新的对象对应一个新的UI。
React就会将这2个UI进行对比(DOM diff)区别然后进行局部更新。
复杂 state 怎么处理
所谓复杂就是state是个对象如果state是个复杂的对象怎么办
如果state里不止有n怎么办
一.类组件里有n和m
1.类组件的setState会自动合并第一层属性:
当类组件里有n和m时setState时可以只setState一部分因为它会自动延用之前的属性。
2.类组件的setState不会合并第二层属性。
二.函数组件里有n和m
函数组件的setState不会自动合并当多个setState放一起时要记得用...操作符把之前的复制过来。函数组件的setState完全不会帮你自动合并不管你是第一层还是第二层要合并只能自己用...操作符。
一.类组件里有n和m
1.类组件的setState会自动合并第一层属性
例子
function App() {return (div classNameApp爸爸Son //div);
}class Son extends React.Component {constructor() {super();this.state {n: 0,m: 0};}addN() {this.setState({ n: this.state.n 1 });// m 会被覆盖为 undefined 吗}addM() {this.setState({ m: this.state.m 1 });// n 会被覆盖为 undefined 吗}render() {return (div classNameSon儿子 n: {this.state.n}button onClick{() this.addN()}n1/buttonm: {this.state.m}button onClick{() this.addM()}m1/buttonGrandson //div);}
}const Grandson () {const [n, setN] React.useState(0);return (div classNameGrandson孙子 n:{n}button onClick{() setN(n 1)}1/button/div);
};const rootElement document.getElementById(root);
ReactDOM.render(App /, rootElement); 总结:类组件里的setState,如果你对其中一部分进行修改那么其他的部分会自动延用上一次的值而不会被undefined覆盖。
addN() {this.setState({ n: this.state.n 1 });
/*相当于写this.setState({...this.state, n: this.state.n 1 });当只设置n时m会自动延用上一次的值m不会被覆盖为undefinedReact会自动帮你写...this.state:把之前所有值拷过来再将n变成n1
*/}
2.类组件的setState不会合并第二层属性
类组件的setState会自动合并第一层属性但是不会合并第二层属性。
解决方法: 使用Object.assign或者...操作符二选一。
1 使用...操作符(推荐)
class Son extends React.Component {constructor() {super();this.state {n: 0, //第1层设置n时不要管mm: 0,user: {name: frank, //第2层设置name时必须管ageage: 18}};}...changeUser() {this.setState({//m和n不会被置空user: {...this.state.user, //防止age被置空的解决方法name: jack//坑:age会被置空解决方法:...this.state.user}});}render() {return (div classNameSon儿子 n: {this.state.n}button onClick{() this.addN()}n1/buttonm: {this.state.m}button onClick{() this.addM()}m1/buttonhr /divuser.name: {this.state.user.name}/divdivuser.age: {this.state.user.age}/divbutton onClick{() this.changeUser()}change user/buttonGrandson //div);}
}
...操作法是React经常会用到的技巧 2 使用Object.assign
class Son extends React.Component {constructor() {super();this.state {n: 0,m: 0,user: {name: frank,age: 18}};}changeUser() {//Object.assign把之前user的所有属性复制到新对象上const user Object.assign({}, this.state.user); //等价于const user{...this.state.user}user.name jack;this.setState({// m 和 n 不会被置空user: user});}render() {return (div classNameSon儿子 n: {this.state.n}button onClick{() this.addN()}n1/buttonm: {this.state.m}button onClick{() this.addM()}m1/buttonhr /divuser.name: {this.state.user.name}/divdivuser.age: {this.state.user.age}/divbutton onClick{() this.changeUser()}change user/buttonGrandson //div);}
}const Grandson () {const [n, setN] React.useState(0);return (div classNameGrandson孙子 n:{n}button onClick{() setN(n 1)}1/button/div);
};
二.函数组件里有n和m
同样的需求下用函数组件更简便类组件已经过时了尽量用函数组件。 例子
//第1种写法
const Grandson () {const [n, setN] React.useState(0);const [m, setM] React.useState(0);return (div classNameGrandson孙子 n:{n}button onClick{() setN(n 1)}n1/buttonm:{m}button onClick{() setM(m 1)}m1/button/div);
};
函数组件另一种不推荐的写法你会发现m被质空:
//不要这样写当你setStaten时m就会undefined.
const Grandson () {const [state, setState] React.useState({n:0,m:0});return (div classNameGrandson孙子 n:{state.n}button onClick{() setState({n:state.n 1})}n1/buttonm:{state.m}button onClick{() setState({m:state.m 1})}m1/button/div);
}; 如果你要用函数组件的setState就不要搞个对象在这里。如果你非要弄个对象那每次赋值时要记得先拷贝之前的。
//第2种写法
const Grandson () {const [state, setState] React.useState({n:0,m:0});return (div classNameGrandson孙子 n:{state.n}button onClick{() setState({...state,n:state.n 1})}n1/buttonm:{state.m}button onClick{() setState({...state,m:state.m 1})}m1/button/div);
};
总结: 类组件会自动合并第1层不会合并第2层。函数组件不会自动合并。所以不要依赖自动合并。
事件绑定
React组件里事件绑定的各种写法:onClick,onKeyPress... 一.类组件的事件绑定
button onClick{()this.addN()}
示例
//最终写法:这种写法this不会变成window
class Son extends React.Component {addN() this.setState({n:this.state.n1}) //绑定
//上面的写法是下面的语法糖,2种写法完全等价
//constructor(){
// this.addN () this.setState({n:this.state.n1})
//}render(){button onClick{this.addN()} //使用}
}constructor里的thisthis.addN 指的是当前对象。 这种写法函数会被定义到对象本身身上这意味着每个Son组件都有自己的addN,如果有两个Son,就有2个addN。
//补充
//1.箭头函数上的this是不会变的因为箭头函数不支持this
addN(){this.setState({n:this.state.n1})}//2.constructor里的thisthis.addN 指的是当前对象Son。
class Son extends React.Component {constructor(){this.addN () this.setState({n:this.state.n1})}
}
复习this
1.this的值到底是什么给this定性
2.你怎么还没搞懂this? 如何确定this
3.JS里为什么会有this? 为什么要设计this
为什么this会变/不会变
所有函数的this都是参数由调用决定所以可变
唯独箭头函数的this不变因为箭头函数不接受this
React的特点:能不做的我都不做。
Vue的特点:能帮你做的都帮你做了。
this面试题1 this面试题2 总结
React是面向类和函数Vue更像是面向对象。
React提供的是JSX语法(2点):
1.普通属性用标签
2.非普通属性也就是跟JS相关的都写在{}里。
{}之外都是标签{}之内是JS。
面试题
React与Vue的区别
相同点
1.都是对视图的封装React是用类和函数表示一个组件而Vue是通过构造选项构造一个组件。
2.都提供了createElement的XML简写React提供的是JSX语法而Vue是提供的是template模版写法(语法巨多)
不同点
React是把HTML放在JS里写(HTML in JS),而Vue是把JS放在HTML里写(JS in HTML)