React 笔记

第一个例子


虚拟DOM的两种创建方式

使用jsx创建虚拟DOM

//1.创建虚拟DOM
const VDOM = (    /* 此处一定不要写引号,因为不是字符串 */
  

Hello,React

) //2.渲染虚拟DOM到页面 ReactDOM.render(VDOM,document.getElementById('test'))

使用js创建虚拟DOM

//1.创建虚拟DOM
const VDOM = React.createElement('h1',{id:'title'},
  React.createElement('span',{},'Hello,React'))
//2.渲染虚拟DOM到页面
ReactDOM.render(VDOM,document.getElementById('test'))

关于虚拟DOM

  1. 本质是Object类型的对象(一般对象)
  2. 虚拟DOM比较“轻”,真实DOM比较“重”,因为虚拟DOM是React内部在用,无需真实DOM上那么多的属性。
  3. 虚拟DOM最终会被React转化为真实DOM,呈现在页面上。

jsx语法规则:

  1. 定义虚拟DOM时,不要写引号。
  2. 标签中混入JS表达式时要用{}。(不能写非表达式的JS语句,如 for(){ } )
  3. 样式的类名指定不要用class,要用className。
  4. 内联样式,要用style={{key:value}}的形式去写。
  5. 只有一个根标签
  6. 标签必须闭合
  7. 标签首字母
    (1). 若小写字母开头,则将该标签转为html中同名元素,若html中无该标签对应的同名元素,则报错。
    (2). 若大写字母开头,react就去渲染对应的组件,若组件没有定义,则报错。

react中定义组件

函数式组件

//1.创建函数式组件
function MyComponent(){
  console.log(this);   //此处的this是undefined,因为babel编译后开启了严格模式
  return 

我是用函数定义的组件(适用于【简单组件】的定义)

} //2.渲染组件到页面 ReactDOM.render(,document.getElementById('test'))

MyComponent中的this是undefined,因为babel编译后开启了严格模式。
执行了ReactDOM.render(.......)之后,发生了什么?
1.React解析组件标签,找到了MyComponent组件。
2.发现组件是使用函数定义的,随后调用该函数,将返回的虚拟DOM转为真实DOM,随后呈现在页面中。

类式组件

//1.创建类式组件
class MyComponent extends React.Component {
  render(){
    //render是放在哪里的?—— MyComponent的原型对象上,供实例使用。
    //render中的this是谁?—— MyComponent的实例对象 <=> MyComponent组件实例对象。
    console.log('render中的this:',this);
    return 

我是用类定义的组件(适用于【复杂组件】的定义)

} } //2.渲染组件到页面 ReactDOM.render(,document.getElementById('test'))

执行了ReactDOM.render(.......之后,发生了什么?
1.React解析组件标签,找到了MyComponent组件。
2.发现组件是使用类定义的,随后new出来该类的实例,并通过该实例调用到原型上的render方法。
3.将render返回的虚拟DOM转为真实DOM,随后呈现在页面中。

组件实例属性:state

//1.创建组件
class Weather extends React.Component{  
  //构造器调用几次? ———— 1次
  constructor(props){
    console.log('constructor');
    super(props)
    //初始化状态
    this.state = {isHot:false,wind:'微风'}
    //解决changeWeather中this指向问题
    this.changeWeather = this.changeWeather.bind(this)
  }

  //render调用几次? ———— 1+n次 1是初始化的那次 n是状态更新的次数
  render(){
    console.log('render');
    //读取状态
    const {isHot,wind} = this.state
    return 

今天天气很{isHot ? '炎热' : '凉爽'},{wind}

} //changeWeather调用几次? ———— 点几次调几次 changeWeather(){ //changeWeather放在哪里? ———— Weather的原型对象上,供实例使用 //由于changeWeather是作为onClick的回调,所以不是通过实例调用的,是直接调用 //类中的方法默认开启了局部的严格模式,所以changeWeather中的this为undefined console.log('changeWeather'); //获取原来的isHot值 const isHot = this.state.isHot //严重注意:状态必须通过setState进行更新,且更新是一种合并,不是替换。 this.setState({isHot:!isHot}) console.log(this); //严重注意:状态(state)不可直接更改,下面这行就是直接更改!!! //this.state.isHot = !isHot //这是错误的写法 } } //2.渲染组件到页面 ReactDOM.render(,document.getElementById('test'))

由于changeWeather是作为onClick的回调,所以不是通过实例调用的,是直接调用。(所以this不是实例)
类中的方法默认开启了局部的严格模式,所以changeWeather中的this为undefined。(所以this不是window)

state 的简写方式

class Weather extends React.Component{
  //初始化状态
  state = {isHot:false,wind:'微风'}

  render(){
    const {isHot,wind} = this.state
    return 

今天天气很{isHot ? '炎热' : '凉爽'},{wind}

} //自定义方法————要用赋值语句的形式+箭头函数 changeWeather = () => { const isHot = this.state.isHot this.setState({isHot:!isHot}) } } //2.渲染组件到页面 ReactDOM.render(,document.getElementById('test'))

state总结

  1. 组件中 render 方法中的 this 为组件实例对象。
  2. 组件自定义的方法中 this 为 undefined,如何解决?
    a) 强制绑定 this: 通过函数对象的 bind()
    b) 箭头函数
  3. 状态数据,不能直接修改或更新,需要使用 setState() 方法。

组件实例属性:props

class Person extends React.Component{
  render(){
    // console.log(this);
    const {name,age,sex} = this.props
    return (
      
  • 姓名:{name}
  • 性别:{sex}
  • 年龄:{age+1}
) } } //渲染组件到页面 ReactDOM.render(,document.getElementById('test1')) ReactDOM.render(,document.getElementById('test2')) const p = {name:'老刘',age:18,sex:'女'} // console.log('@',...p); // ReactDOM.render(,document.getElementById('test3')) ReactDOM.render(,document.getElementById('test3'))

注意:
中的{...p}和JS中的...运算符无关,不符合JS语法。
展开运算符不能展开对象,但构造字面量对象时可以使用展开语法。代码如下:

//构造字面量对象时使用展开语法
let person = {name:'tom',age:18}
let person2 = {...person}
//console.log(...person); //报错,展开运算符不能展开对象

对props进行限制




// 对标签属性进行类型、必要性的限制
Person.propTypes = {
  name:PropTypes.string.isRequired, //限制name必传,且为字符串
  sex:PropTypes.string,//限制sex为字符串
  age:PropTypes.number,//限制age为数值
  speak:PropTypes.func,//限制speak为函数
}
//指定默认标签属性值
Person.defaultProps = {
  sex:'男',//sex默认值为男
  age:18 //age默认值为18
}

props的简写方式

class Person extends React.Component{
  constructor(props){ }

  static propTypes = {
    
  }
  static defaultProps = {
   
  }
  
  render(){ }
}

类式组件中的构造器

通常,在 React 中,构造函数仅用于以下两种情况:

  • 通过给 this.state 赋值对象来初始化 内部 state。
  • 为 事件处理函数 绑定实例。

总结:不写也行,通过其它方式去做

(视频至25)

你可能感兴趣的:(React 笔记)