当前位置:   article > 正文

JSX语法详解_jsx大括号的含义

jsx大括号的含义

JSX是React的核心组成部分,它使用XML标记的方式去直接声明界面,界面组件之间可以互相嵌套。可以理解为在JS中编写与XML类似的语言,一种定义带属性树结构(DOM结构)的语法,它的目的不是要在浏览器或者引擎中实现,它的目的是通过各种编译器将这些标记编译成标准的JS语言。

虽然你可以完全不使用JSX语法,只使用JS语法,但还是推荐使用JSX,可以定义包含属性的树状结构的语法,类似HTML标签那样的使用,而且更便于代码的阅读。

使用JSX语法后,你必须要引入babel的JSX解析器,把JSX转化成JS语法,这个工作会由babel自动完成。同时引入babel后,你就可以使用新的es6语法,babel会帮你把es6语法转化成es5语法,兼容更多的浏览器。

我们从最简单的一个官网例子helloworld开始:

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="UTF-8" />
  5. <title>Hello React!</title>
  6. <script src="vendor-js/react.js"></script>
  7. <script src="vendor-js/react-dom.js"></script>
  8. <script src="vendor-js/babel-core/browser.min.js"></script>
  9. </head>
  10. <body>
  11. <div id="example"></div>
  12. <script type="text/babel">
  13. ReactDOM.render(
  14. <h1>Hello, world!</h1>,
  15. document.getElementById('example')
  16. );
  17. </script>
  18. </body>
  19. </html>

在这个简单的例子中,看不出来有任何jsx语法的地方,当其中<h1>Hello,world</h1>就是使用到了jsx语法。HTML 语言直接写在 JavaScript 语言之中,不加任何引号,这就是 JSX 的语法,它允许 HTML 与 JavaScript 的混写。如果转化成纯JavaScript 的话,就是:

  1. <script type="text/javascript">
  2. ReactDOM.render(
  3. React.DOM.h1(null,'hello,world!'),
  4. document.getElementById('example')
  5. );
  6. </script>

在上述JSX语法中有两个要注意的地方:

  • <script> 标签的 type 属性为 text/babel,这是React 独有的 JSX 语法,跟 JavaScript 不兼容。凡是在页面中直接使用 JSX 的地方,都要加上 type="text/babel"。
  • 一共用了三个库: react.js 、react-dom.js 和 browser.min.js ,它们必须首先加载。其中,react.js 是 React 的核心库,react-dom.js 是提供与 DOM 相关的功能, browser.min.js的作用是将 JSX 语法转为 JavaScript 语法。

    将 JSX 语法转为 JavaScript 语法,这一步很消耗时间。现在前端项目,都会使用前端工程化,不会直接在html页面中直接写js代码,写好的js代码都会使用工具进行编译压缩等。这样的话,我们的jsx也会通过编译直接转化成js语法,让浏览器直接使用。

ReactDOM.render 是 React 的最基本方法,将模板转为HTML语言,并插入指定的 DOM 节点。

JSX的特点:

  1. 类XML语法容易接受,结构清晰
  2. 增强JS语义
  3. 抽象程度高,屏蔽DOM操作,跨平台
  4. 代码模块化

JSX基本语法规则:

JSX本身就和XML语法类似,可以定义属性以及子元素。唯一特殊的是可以用大括号来加入JavaScript表达式。遇到 HTML 标签(以 < 开头),就用 HTML 规则解析;遇到代码块(以 { 开头),就用 JavaScript 规则解析。

  1. var arr = [
  2. <h1>Hello world!</h1>,
  3. <h2>React is awesome</h2>,
  4. ];
  5. ReactDOM.render(
  6. <div>{arr}</div>,
  7. document.getElementById('example')
  8. );

这个就是一个简单的html与js混用的例子。arr变量中存在html元素,div中又使用了arr这个js变量。转化成纯javascript的话:

  1. var h1=React.DOM.h1(null,'Hello world!');
  2. var h2=React.DOM.h1(null,'React is awesome');
  3. var div=React.DOM.div(null,h1,h2);
  4. ReactDOM.render(
  5. div,
  6. document.getElementById('example')
  7. );

React组件

我们使用jsx来将代码封装成React组件,然后像插入普通 HTML 标签一样,在其他地方插入这个组件。使用React.createClass用于生成一个组件。

  1. var MyComponent=React.createClass({
  2. render: function() {
  3. return <h1>Hello world!</h1>;
  4. }
  5. });
  6. ReactDOM.render(
  7. <MyComponent />,
  8. document.getElementById('example')
  9. );

上面代码中,变量 MyComponent就是一个组件类。模板插入 <MyComponent /> 时,会自动生成 MyComponent 的一个实例(下文的"组件"都指组件类的实例)。所有组件类都必须有自己的 render 方法,用于输出组件。

React 可以渲染 HTML 标签 (strings) 或 React 组件 (classes)。

  • 在react中通常约定组件类的第一个字母必须大写,html标签都是小写。
    1. //要渲染 HTML 标签,只需在 JSX 里使用小写字母开头的标签名。
    2. var myDivElement = <div className="foo" />;
    3. React.render(myDivElement, document.getElementById('example'));
    4. //要渲染 React 组件,只需创建一个大写字母开头的本地变量。
    5. var MyComponent = React.createClass({/*...*/});
    6. var myElement = <MyComponent />;
    7. React.render(myElement, document.getElementById('example'));
  • 还有一个注意点:组件类只能包含一个顶层标签,否则会报错。
    1. //var myDivElement =<h1>你好</h1><h1>hello</h1>;
    2. //上述写法是会报错的,要写成只有一个顶层标签:
    3. var myDivElement =<div><h1>你好</h1><h1>hello</h1></div>;

上述代码一个静态的组件,下面看一个动态组件:

  1. var MyComponent=React.createClass({
  2. getInitialState: function() {
  3. return {clickNum: 0};
  4. },
  5. handleClick:function(){
  6. var num=this.state.clickNum;
  7. num++;
  8. this.setState({clickNum:num});
  9. },
  10. render: function() {
  11. return (
  12. <div>
  13. <h1 onClick={this.handleClick}>Hello {this.props.name}!</h1>
  14. <h2 style={{color:'red'}}>点击{this.props.name}次数:{this.state.clickNum}</h2>
  15. </div>
  16. );
  17. }
  18. });
  19. ReactDOM.render(
  20. <div>
  21. <MyComponent name="张三" />
  22. <hr/>
  23. <MyComponent name="李四" />
  24. </div>,
  25. document.getElementById('example')
  26. );

上面代码中定义的MyComponent组件包含属性,状态和事件,是一个简单的比较完整的组件。使用props通过父组件进行传递值,使用state定义组件自己的状态,组件支持的大部分的DOM操作事件。

关于属性props:

  • class 属性需要写成 className ,for 属性需要写成 htmlFor ,这是因为 class 和 for 是 JavaScript 的保留字。
  • 直接在标签上使用style属性时,要写成style={{}}是两个大括号,外层大括号是告知jsx这里是js语法,和真实DOM不同的是,属性值不能是字符串而必须为对象,需要注意的是属性名同样需要驼峰命名法。即margin-top要写成marginTop。
  • this.props.children 不要children作为把对象的属性名。因为this.props.children获取的该标签下的所有子标签。this.props.children 的值有三种可能:如果当前组件没有子节点,它就是 undefined ;如果有一个子节点,数据类型是 object ;如果有多个子节点,数据类型就是 array 。所以,处理 this.props.children 的时候要小心。官方建议使用React.Children.map来遍历子节点,而不用担心数据类型。

关于状态state:

  • 组件免不了要与用户互动,React 将组件看成是一个状态机,一开始有一个初始状态,然后用户互动,导致状态变化,从而触发重新渲染 UI。
  • getInitialState 方法用于定义初始状态,也就是一个对象,这个对象可以通过 this.state 属性读取。当用户点击组件,导致状态变化,this.setState 方法就修改状态值,每次修改以后,自动调用 this.render 方法,再次渲染组件。

由于 this.props 和 this.state 都用于描述组件的特性,可能会产生混淆。一个简单的区分方法是,this.props 表示那些一旦定义,就不再改变的特性,而 this.state 是会随着用户互动而产生变化的特性。

注意点:
如果往原生 HTML 元素里传入 HTML 规范里不存在的属性,React 不会显示它们。如果需要使用自定义属性,要加 data- 前缀。

<div data-custom-attribute="foo" />

PropTypes

组件的属性props可以接受任意值,字符串、对象、函数等等都可以。有时,我们需要一种机制,验证别人使用组件时,提供的参数是否符合要求。React中使用PropTypes进行参数的校验。

  1. var MyTitle = React.createClass({
  2. propTypes: {
  3. title: React.PropTypes.string.isRequired,
  4. },
  5. render: function() {
  6. return <h1> {this.props.title} </h1>;
  7. }
  8. });

上面的Mytitle组件有一个title属性。PropTypes 告诉 React,这个 title 属性是必须的,而且它的值必须是字符串。当我们给title传递一个数字时,控制台就会报错:

Warning: Failed propType: Invalid prop `title` of type `number` supplied to `MyTitle`, expected `string`.

此外,getDefaultProps 方法可以用来设置组件属性的默认值。

获取真实的DOM节点

组件并不是真实的 DOM 节点,而是存在于内存之中的一种数据结构,叫做虚拟 DOM (virtual DOM)。只有当它插入文档以后,才会变成真实的 DOM 。
有时需要从组件获取真实 DOM 的节点,这时就要用到 ref 属性。

  1. var MyComponent = React.createClass({
  2. handleClick: function() {
  3. this.refs.myTextInput.focus();
  4. },
  5. render: function() {
  6. return (
  7. <div>
  8. <input type="text" ref="myTextInput" />
  9. <input type="button" value="Focus the text input" onClick={this.handleClick} />
  10. </div>
  11. );
  12. }
  13. });

为了获取真是DOM节点,html元素必须有一个 ref 属性,然后 this.refs.[refName] 就会返回这个真实的 DOM 节点。需要注意的是,由于 this.refs.[refName] 属性获取的是真实 DOM ,所以必须等到虚拟 DOM 插入文档以后,才能使用这个属性,否则会报错。

求值表达式

要使用 JavaScript 表达式作为属性值,只需把这个表达式用一对大括号 ( { } ) 包起来,不要用引号 ( " " )。
在编写JSX时,在 { } 中不能使用语句(if语句、for语句等等),但可以使用求值表达式,这本身与JSX没有多大关系,是JS中的特性,它是会返回值的表达式。我们不能直接使用语句,但可以把语句包裹在函数求值表达式中运用。

条件判断的写法

你没法在JSX中使用 if-else 语句,因为 JSX 只是函数调用和对象创建的语法糖。在 { } 中使用,是不合法的JS代码,不过可以采用三元操作表达式

  1. var HelloMessage = React.createClass({
  2. render: function() {
  3. return <div>Hello {this.props.name ? this.props.name : "World"}</div>;
  4. }
  5. });
  6. ReactDOM.render(<HelloMessage name="xiaowang" />, document.body);

可以使用比较运算符“ || ”来书写,如果左边的值为真,则直接返回左边的值,否则返回右边的值,与if的效果相同。

  1. var HelloMessage = React.createClass({
  2. render: function() {
  3. return <div>Hello {this.props.name || "World"}</div>;
  4. }
  5. });
函数表达式

( )有强制运算的作用

  1. var HelloMessage = React.createClass({
  2. render: function() {
  3. return <div>Hello {
  4. (function(obj){
  5. if(obj.props.name)
  6. return obj.props.name
  7. else
  8. return "World"
  9. }(this))
  10. }</div>;
  11. }
  12. });
  13. ReactDOM.render(<HelloMessage name="xiaowang" />, document.body);

外括号“ )”放在外面和里面都可以执行。唯一的区别是括号放里面执行完毕拿到的是函数的引用,然后再调用“function(){}(this)()”;括号放在外面的时候拿到的事返回值。

组件的生命周期

组件的生命周期分成三个状态:

  1. * Mounting:已插入真实 DOM
  2. * Updating:正在被重新渲染
  3. * Unmounting:已移出真实 DOM

React 为每个状态都提供了两种处理函数,will 函数在进入状态之前调用,did 函数在进入状态之后调用,三种状态共计五种处理函数。

  1. * componentWillMount()
  2. * componentDidMount()
  3. * componentWillUpdate(object nextProps, object nextState)
  4. * componentDidUpdate(object prevProps, object prevState)
  5. * componentWillUnmount()

此外,React 还提供两种特殊状态的处理函数。

  1. * componentWillReceiveProps(object nextProps):已加载组件收到新的参数时调用
  2. * shouldComponentUpdate(object nextProps, object nextState):组件判断是否重新渲染时调用

注释

JSX 里添加注释很容易;它们只是 JS 表达式而已。你只需要在一个标签的子节点内(非最外层)小心地用 {} 包围要注释的部分。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/494412
推荐阅读
相关标签
  

闽ICP备14008679号