React基础

个人博客搭建完成,欢迎大家来访问哦
黎默丶lymoo的博客

什么是React

React是一个用于构建用户界面的JavaScript库,而不是一个MVC框架,但可以使用React作为MVC架构的View层轻易的在已有项目中使用。React起源于Facebook的内部项目,用来架设Instagram的网站,并于2013年5月开源。React拥有较高的性能,代码逻辑非常简单,越来越多的人已开始关注和使用它。

React的特点

虚拟DOM结构

以前没有ajax技术的时候,web页面从服务端整体渲染出html输出到浏览器端进行渲染,同样的,用户的一个改变页面的操作也会刷新整个页面来完成。直到有了ajax出现,实现页面局部刷新,带来的高效和分离让web开发者们惊叹不已。但随之而来的问题是,复杂的用户交互及展现需要通过大量的DOM操作来完成,这让页面的性能以及开发的效率又出现了新的瓶颈。
时至今日,谈到前端性能优化,减少DOM元素、减少reflow和repaint、编码过程中尽量减少DOM的查询等手段是大家耳熟能详的。而页面任何UI的变化都是通过整体刷新来完成的。幸运的是,React通过自己实现的DOM Diff算法,计算出虚拟页面当前版本和新版本之间的差异,最小化重绘,避免不必要的DOM操作,解决了这两个公认的前端性能瓶颈,实现高效DOM渲染。
我们知道,频繁的操作DOM所带来的性能消耗是很大的,而React之所以快,是因为它不直接操作DOM,而是引进虚拟DOM的实现来解决这个问题
对于页面的更新,React通过自己实现的DOM Diff算法来进行差异对比、差异更新,反映到页面上就是只重绘了更新的部分,从而提高渲染效率。

组件化

在业务开发中,遇到公共的模板部分,我们不得不将模板和规定的数据格式耦合在一起来实现组件。而在React中,我们可以使用JSX语法来封装组件,将组件的结构、数据逻辑甚至样式都聚合在一起,更加简单、明了、直观的定义组件。
有了组件化的实现,我们可以很直观的将一个复杂的页面分割成若干个独立组件,再将这些独立组件组合完成一个复杂的页面。这样既减少了逻辑复杂度,又实现了代码的重用。

配置React的开发环境

获取相关文件

首先我们需要获取到React的开发文件,可以从React官网下载相关配置文件,由于React用到的是JSX语法,浏览器是不能够直接进行解析的,还需要配置一个将JSX转换成JavaScript的文件,将链接中的代码复制到自己本地的任意新建的一个文件中保存完成配置,最后配置文件如下图所示:

React所需的环境配置文件

配置React开发文件模板

新建一个HTML文件进行如下配置,当这个文件配置好之后,以后需要开发React的程序都可以以这个为模板进行开发。

<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8">
        <title></title>
        <!--引入react核心库-->
        <script type="text/javascript" src="build/react.js"></script>
        <!--引入react的DOM操作-->
        <script type="text/javascript" src="build/react-dom.js"></script>
        <!--把JSX语法转换成浏览器能识别的JavaScript-->
        <script type="text/javascript" src="build/browser.min.js"></script>
    </head>
    <body>
        <!--把通过react生成的DOM放置在#container里面-->
        <div id="container"></div>
    </body>
    <!--写JSX的地方需要设置为text/babel-->
    <script type="text/babel">
        // 在这里写react代码

    </script>
</html>

JSX的认识

上文提到的JSX是什么东西呢?其实JSX是一种语法,不是一门新的语言,它既包括JavaScript和Xhtml语言,XHTML用标签来进行表示,JavaScript用{}展示。
这里主要注意的是写JSX的地方需要设置为text/babel,以及内部的书写规范

简单的React示例

这里举一个例子展示一个最简单的React程序
因为React放置的是虚拟的DOM结构,我们需要将它渲染至页面上,它提供的语法为ReactDOM.render()
它里面可以有三个参数,第一个为我们想要渲染至页面上的DOM元素,第二个为我们想在页面的哪个节点元素中进行渲染,第三个为渲染成功以后的回调

<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8">
        <title></title>
        <script type="text/javascript" src="build/react.js"></script>
        <script type="text/javascript" src="build/react-dom.js"></script>
        <script type="text/javascript" src="build/browser.min.js"></script>
    </head>
    <body>
        <div id="container"></div>
    </body>
    <script type="text/babel">
        var name = "zhangsan";
        ReactDOM.render(
            <h1>hello {name}</h1>,
           document.getElementById("container")
        );
        // 这里有第二种写法
        ReactDOM.render(
            React.createElement("h1", null, "hello " + name),
            document.getElementById("container")
        )
    </script>
</html>

展示的效果为

React示例

如果想要渲染出多个相同标签的不同内容,React也提供了一种方式,用key来确定标签的不同,以下展示部分script代码

var names = ["Alice", "Cindy", "Rose"];
ReactDOM.render(
    <ul>
        {
            names.map(function (name, key) {
                return <li key={key}>{name}</li>
            })
        }
    </ul>,
    document.getElementById("container")
)

展示的效果为
React示例

React组件

在react中将一些重用的dom结构进行了封装,就称为组件,它便于我们进行代码管理以及代码复用

组件的命名规则

首字母必须大写,使用驼峰命名法

组件的构建方法

React提供了一个命令为React.createClass()来创建一个组件,它内部包含一个必须要有的方法render,用来返回一个或多个的DOM结构

封装一个简单的组件示例

var HelloMessage = React.createClass({
    render: function () {
        return <h1>hello message</h1>
    }
});

这样就简单的封装了一个名为HelloMessage的组件,如果想要调用这个组件,方法为

ReactDOM.render(
    <HelloMessage/>,
    document.getElementById("container")
)

复合组件

复合组件简单来说即是组件套用组件,我们来尝试封装一个复合组件

var Component1 = React.createClass({
    render: function () {
        return <h1>测试</h1>
    }
});
var Component2 = React.createClass({
    render: function () {
        return <a href="http://www.baidu.com" target="_blank">去百度</a>
    }
});
var Component3 = React.createClass({
    render: function () {
        return (
            <div>
                <Component1/>
                <Component2/>
            </div>
        )
    }
});

React的CSS样式

React有三种添加样式的方法

1.内联样式

var StyleTest = React.createClass({
    render: function () {
        return <div style={{backgroundColor: "red", border: "10px solid #ccc"}}>Test</div>
    }
});
ReactDOM.render(
    <StyleTest/>,
    document.getElementById("container")
)

效果展示:
React示例

2.对象样式

注意这里的styles都是以对象的格式进行书写的,千万不要以前的习惯后面写封号,字段名不加引号等等

var styles = {
    color: "blue",
    backgroundColor: "green"
};
var StyleTest = React.createClass({
    render: function () {
        return <div style={styles}>Test</div>
    }
});
ReactDOM.render(
    <StyleTest/>,
    document.getElementById("container")
)

效果展示:
React示例

3.选择器样式

这个就和我们原来的CSS书写差不多,但是类名的添加不再是class,注意这里是className

<style>
    .styles{
        color: yellow;
        background-color: black;
    }
</style>
var StyleTest = React.createClass({
    render: function () {
        return <div className="styles">Test</div>
    }
});
ReactDOM.render(
        <StyleTest/>,
    document.getElementById("container")
)

效果展示:
React示例

React的属性props

props简单来说就是一个组件的私有属性,从父组件传递一些需要的值到子组件中来使用
使用方法为this.props,这里的this便代表它的父组件,props则代表了父组件上的所有元素,是一个数组
用法示例:

var PropsTest = React.createClass({
    render: function () {
        return <h1>hello {this.props.name}</h1>
    }
});
ReactDOM.render(
    <PropsTest name="zhangsan"/>,  // 这里的name值会传递到h1的标签中相应的位置
    document.getElementById("container")
)

这里有一个特例为this.props.children,它表示的是组件的所有子节点

var PropsTest = React.createClass({
    render: function () {
        return {
            <ol>
                {
                    React.Children.map(this.props.children, function (child)) {
                        return <li>{child}</li>
                    }
                }
            </ol>
        }
    }
})
ReactDOM.render({
    <PropsTest>
        <span>hello</span>
        <span>world</span>
    </PropsTest>,
    document.getElementById("conotainer")
})

效果展示:
React示例

设置默认值

React提供一个专门设置props的默认值命令getDefaultProps
具体用法为:

var PropsTest = React.createClass({
    getDefaultProps: function () {
        // 设置该组件属性的初始值
        return {
            name: "lisi"
        }
    },
    render: function () {
        return <h1>hello {this.props.name}</h1>
    }
});
ReactDOM.render(
    <PropsTest/>,
    document.getElementById("container")
);

效果展示:
React示例

设置props数据类型

var PropsTest = React.createClass({
    propTypes: {
        // 设置数据的类型
        name: React.PropTypes.string.isRequired
    },
    getDefaultProps: function () {
        return {
            name: "lisi"
        }
    },
    render: function () {
        return <h1>hello {this.props.name}</h1>
    }
});
ReactDOM.render(
    <PropsTest/>,
    document.getElementById("container")
);

React的状态state

React把组件看成是一个状态机(State Machines)。通过与用户的交互,实现不同状态,然后渲染UI,让用户界面和数据保持一致。
React里,只需更新组件的state,然后根据新的state重新渲染用户界面(不要操作DOM)。
getInitialState方法用于定义初始状态,也就是一个对象,这个对象可以通过this.state属性读取。当用户点击组件,导致状态变化,this.setState方法就修改状态值,每次修改以后,自动调用this.render方法,再次渲染组件。

var LikeButton = React.createClass({
    getInitialState: function() {
        return {liked: false};
    },
    handleClick: function(event) {
        this.setState({liked: !this.state.liked});
    },
    render: function() {
        var text = this.state.liked ? '喜欢' : '不喜欢';
        return (
            <p onClick={this.handleClick}>
                你<b>{text}</b>我。点击切换状态。
            </p>
        );
    }
});
ReactDOM.render(
    <LikeButton />,
    document.getElementById("container")
);

React组件生命周期

React的生命周期可以分为三个状态:
1.Mounting:已插入真实DOM
2.Updating:正在被重新渲染
3.Unmounting:已移出真实DOM

生命周期的方法

componentWillMount:在渲染前调用,在客户端也在服务端。
componentDidMount:在第一次渲染后调用,只在客户端。之后组件已经生成了对应的DOM结构,可以通过this.getDOMNode()来进行访问。 如果你想和其他JavaScript框架一起使用,可以在这个方法中调用setTimeout, setInterval或者发送AJAX请求等操作(防止异部操作阻塞UI)。
componentWillReceiveProps:在组件接收到一个新的prop时被调用。这个方法在初始化render时不会被调用。
shouldComponentUpdate:返回一个布尔值。在组件接收到新的props或者state时被调用。在初始化时或者使用forceUpdate时不被调用。
可以在你确认不需要更新组件时使用。
componentWillUpdate:在组件接收到新的props或者state但还没有render时被调用。在初始化时不会被调用。
componentDidUpdate:在组件完成更新后立即调用。在初始化时不会被调用。
componentWillUnmount:在组件从DOM中移除的时候立刻被调用。
以下示例在Hello组件加载以后,通过componentDidMount方法设置一个定时器,每隔100毫秒重新设置组件的透明度,并重新渲染

var Hello = React.createClass({
    getInitialState: function () {
        return {
            opacity: 1.0
        };
    },
    componentDidMount: function () {
        this.timer = setInterval(function () {
            var opacity = this.state.opacity;
            opacity -= .05;
            if (opacity < 0.1) {
                opacity = 1.0;
            }
            this.setState({
                opacity: opacity
            });
        }.bind(this), 100);
    },
    render: function () {
        return (
            <div style={{opacity: this.state.opacity}}>
                Hello {this.props.name}
            </div>
        );
    }
});
ReactDOM.render(
    <Hello name="world"/>,
    document.getElementById('example')
);

原文链接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容