大家好,我是编程小6,很高兴遇见你,有问题可以及时留言哦。
React是近几年来前端项目开发非常火的一个框架,其背景是Facebook团队的技术支持,市场占有率也很高。很多初学者纠结一开始是学react还是vue。个人觉得,有时间的话,最好两个都掌握一下。从学习难度上来说,react要比vue稍难一些。万事开头难,但是掌握了react对于大幅提高前端技能还是非常有帮助的。本文一步步详细梳理了从创建react、精简项目、集成插件、初步优化等过程。对于react开发者来说,能够节省很多探索的时间。下面请跟着我来一步步操作。
正式开始前,先看下通过本次分享,能掌握什么?
即使你是新手,跟着操作一遍以后,也可以快速上手React项目啦!
※注:本文代码区域每行开头的“+”表示新增,“-”表示删除,“M”表示修改;代码中的“...”表示省略。
为了加速npm下载速度,先把npm设置为淘宝镜像地址。
npm config set registry http://registry.npm.taobao.org/
也可以使用cnpm,根据喜好自行选择。
通过官方的create-react-app,找个喜欢的目录,执行:
npx create-react-app react-app
命令最后的react-app是项目的名称,可以自行更改。
稍等片刻即可完成安装。安装完成后,可以使用npm或者yarn启动项目。
进入项目目录,并启动项目:
cd react-app
yarn start (或者使用npm start)
如果没有安装yarn,可以前往yarn中文网站安装:
yarn.bootcss.com/
启动后,可以通过以下地址访问项目:
http://localhost:3000/
接下来,删除一般项目中用不到的文件,最简化项目。
├─ /node_modules
├─ package.json
├─ /public
| ├─ favicon.ico
| ├─ index.html
- | ├─ logo192.png
- | ├─ logo512.png
- | ├─ mainfest.json
- | └─ robots.txt
├─ README.md
├─ /src
- | ├─ App.css
| ├─ App.js
- | ├─ App.test.js (jTest自动化测试)
- | ├─ index.css
- | ├─ index.js
- | ├─ logo.svg
- | ├─ serviceWorker.js
- | └─ setuoTests.js (PWA)
└─ yarn.lock
以上文件删除后,页面会报错。这是因为相应的文件引用已不存在。需要继续修改代码。
现在目录结构如下,清爽许多:
├─ /node_modules
├─ package.json
├─ /public
| ├─ favicon.ico
| └─ index.html
├─ README.md
├─ /src
| ├─ App.js
| └─ index.js
└─ yarn.lock
逐个修改以下文件:
src/App.js代码简化如下:
import React from 'react'
function App() {
return (
<div className="App"> <h1>This is React App.</h1> </div>
)
}
export default App
src/index.js代码简化如下:
import React from 'react'
import ReactDOM from 'react-dom'
import App from './App'
ReactDOM.render(<App />, document.getElementById('root'))
public/index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" href="%PUBLIC_URL%/favicon.ico" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>React App</title>
</head>
<body>
<noscript>You need to enable JavaScript to run this app.</noscript>
<div id="root"></div>
</body>
</html>
运行效果如下:
react要求每个组件HTML的最外层必须是由一个标签包裹,且不能存在并列的标签。例如,在src/App.js中,如果是这样就会报错:
// 以下代码将会报错,最外层不能存在并列的标签。
function App() {
return (
<div className="App"> <h1>This is React App.</h1> </div>
<div className="App-other"> <h1>This is React App-other.</h1> </div>
)
}
如果确实需要这样的HTML,并且不想再添加一个父级标签,可以使用Fragment作为最外层。代码修改如下:
M import React, { Fragment } from 'react'
function App() {
return (
+ <Fragment> <div className="App"> <h1>This is React App.</h1> </div> <div className="App-other"> <h1>This is React App-ohter.</h1> </div> + </Fragment>
)
}
export default App
以上仅为了说明Fragment的使用效果。在某些组件嵌套的使用场景下,非常适合使用Fragment。例如父组件是<table>
,而子组件可以用<Fragment>
包裹多个<tr>
。
项目目录结构可根据项目实际灵活制定。这里分享下我常用的结构,仅供参考。
├─ /node_modules
├─ package.json
├─ /public
| ├─ favicon.ico <-- 网页图标
| └─ index.html <-- HTML页模板
├─ README.md
├─ /src
| ├─ /common <-- 全局公用目录
| | ├─ /fonts <-- 字体文件目录
| | ├─ /images <-- 图片文件目录
| | ├─ /js <-- 公用js文件目录
| | └─ /style <-- 公用样式文件目录
| | | ├─ frame.css <-- 全部公用样式(import其他css)
| | | ├─ reset.css <-- 清零样式
| | | └─ global.css <-- 全局公用样式
| ├─ /components <-- 公共模块组件目录
| | ├─ /header <-- 头部导航模块
| | | ├─ index.js <-- header主文件
| | | └─ header.css <-- header样式文件
| | └─ ... <-- 其他模块
| ├─ /pages <-- 页面组件目录
| | ├─ /home <-- home页目录
| | | ├─ index.js <-- home主文件
| | | └─ home.css <-- home样式文件
| | ├─ /login <-- login页目录
| | | ├─ index.js <-- login主文件
| | | └─ login.css <-- login样式文件
| | └─ ... <-- 其他页面
| ├─ App.js <-- 项目主模块
| └─ index.js <-- 项目入口文件
└─ yarn.lock
在frame.css里引入其他公用样式: src/common/style/frame.css
@import './reset.css';
@import './global.css';
然后在src/index.js里引入frame.css
import React from 'react'
import ReactDOM from 'react-dom'
import App from './App'
+ import './common/style/frame.css'
ReactDOM.render(<App />, document.getElementById('root'))
如图,页面全局样式已生效。
工欲善其事必先利其器,这么高大上的react怎能好意思用最原始的css搭配呢?create-react-app默认情况下未暴露配置文件。要更灵活配置项目,需要将配置文件暴露出来。
执行以下命令,暴露配置文件:
※注意:暴露配置的文件的操作不可逆!
npm run eject
如果之前没有提及git的话,可能会报以下错误:
Remove untracked files, stash or commit any changes, and try again
需要先在项目根目录下执行:
git add .
git commit -m "初始化项目(备注)"
稍等片刻,eject成功,目录变化如下:
+ ├─ /config
+ | ├─ /jest
+ | ├─ env.js
+ | ├─ module.js
+ | ├─ paths.js
+ | ├─ pnpTs.js
+ | ├─ webpack.config.js <-- webpack配置文件
+ | └─ webpackDevServer.config.js
├─ /node_modules
├─ package.json
├─ /public
| ├─ favicon.ico
| └─ index.html
├─ README.md
+ ├─ /scripts
+ | ├─ build.js
+ | ├─ start.js
+ | └─ test.js
├─ /src
| ├─ /common <-- 全局公用目录
| ├─ /components <-- 公共模块组件目录
| ├─ /pages <-- 页面组件目录
| ├─ App.js <-- 项目主模块
| └─ index.js <-- 项目入口文件
└─ yarn.lock
eject后,虽然package.json以及webpack.config.js里有了sass相关代码,但是要正确使用Sass/Scss,还要再安装node-sass。
执行以下命令:
npm install node-sass --save-dev
安装完成后,项目已支持Sass/Scss,可以将原css文件后缀名修改为sacc/scss,别忘了把src/index.js
中引入的frame.css后缀名修改为sacc/scss。
支持Less稍微多一点步骤,首先安装less和less-loader:
npm install less less-loader --save-dev
然后修改config/webpack.config.js:
// style files regexes
const cssRegex = /\.css$/;
const cssModuleRegex = /\.module\.css$/;
const sassRegex = /\.(scss|sass)$/;
const sassModuleRegex = /\.module\.(scss|sass)$/;
+ const lessRegex = /\.less$/;
+ const lessModuleRegex = /\.module\.less$/;
...(略)
// Opt-in support for SASS (using .scss or .sass extensions).
// By default we support SASS Modules with the
// extensions .module.scss or .module.sass
{
test: sassRegex,
exclude: sassModuleRegex,
use: getStyleLoaders(
{
importLoaders: 2,
sourceMap: isEnvProduction && shouldUseSourceMap,
},
'sass-loader'
),
// Don't consider CSS imports dead code even if the
// containing package claims to have no side effects.
// Remove this when webpack adds a warning or an error for this.
// See https://github.com/webpack/webpack/issues/6571
sideEffects: true,
},
// Adds support for CSS Modules, but using SASS
// using the extension .module.scss or .module.sass
{
test: sassModuleRegex,
use: getStyleLoaders(
{
importLoaders: 2,
sourceMap: isEnvProduction && shouldUseSourceMap,
modules: {
getLocalIdent: getCSSModuleLocalIdent,
},
},
'sass-loader'
),
},
// 以下这里仿照上面sass的代码,配置下less。
+ {
+ test: lessRegex,
+ exclude: lessModuleRegex,
+ use: getStyleLoaders(
+ {
+ importLoaders: 2,
+ sourceMap: isEnvProduction && shouldUseSourceMap,
+ },
+ 'less-loader'
+ ),
+ sideEffects: true,
+ },
+ {
+ test: lessModuleRegex,
+ use: getStyleLoaders(
+ {
+ importLoaders: 2,
+ sourceMap: isEnvProduction && shouldUseSourceMap,
+ modules: {
+ getLocalIdent: getCSSModuleLocalIdent,
+ },
+ },
+ 'less-loader'
+ ),
+ },
修改后需要执行yarn start
重启项目。
然后将原css文件的后缀名修改为less,src/index.js
中引入的frame.less,页面已正常解析less。
支持Stylus跟Less完全一样,首先安装stylus和stylus-loader:
执行以下命令:
npm install stylus stylus-loader --save-dev
安装完成后,按照上一小节介绍的支持less的方法,修改config/webpack.config.js。完成后重启项目,引入stylus文件可以正常解析了。
我个人习惯使用Stylus,因此后续的讲解中使用Stylus。同时,把src/common/下的style目录也更名为stylus。
├─ /config
├─ /node_modules
├─ package.json
├─ /public
├─ README.md
├─ /scripts
├─ /src
| ├─ /common <-- 全局公用目录
| | ├─ /fonts
| | ├─ /images
| | ├─ /js
M | | └─ /stylus
M | | | ├─ frame.styl
M | | | ├─ reset.styl
M | | | └─ global.styl
| ├─ /components <-- 公共模块组件目录
| ├─ /pages <-- 页面组件目录
| ├─ App.js <-- 项目主模块
| └─ index.js <-- 项目入口文件
└─ yarn.lock
frame.styl代码如下:
@import './reset.styl';
@import './global.styl';
src/index.js代码修改如下:
import React from 'react'
import ReactDOM from 'react-dom'
import App from './App'
M import './common/style/frame.styl'
ReactDOM.render(<App />, document.getElementById('root'))
最基本的配置搞定了,接下来要开始引入页面(pages)了。页面的切换需要使用路由(Router),请继续阅读下面的章节。
首先,构建home和login页面。
src/pages/home/index.js代码:
import React, { Component } from 'react'
import './home.styl'
class Home extends Component {
render() {
return (
<div className="P-home">
<h1>Home page</h1>
</div>
)
}
}
export default Home
src/pages/home/home.styl代码
.P-home
h1
padding: 20px 0
font-size: 30px
color: #fff
background: #67C23A
text-align: center
src/pages/login/index.js代码:
import React, { Component } from 'react'
import './login.styl'
class Login extends Component {
render() {
return (
<div className="P-login">
<h1>Login page</h1>
</div>
)
}
}
export default Login
src/pages/login/login.styl代码
.P-login
h1
background: #E6A23C
我个人的习惯是,仅供参考:
全局公用级别(不需要模块化)的className,用G-xxx。例如G-autocut(截字)、G-color-red(文字红色)。
页面级别的className,用P-xxx。
模块级别的className,用M-xxx。
接下来,我们使用react-router-dom实现路由。
执行安装命令:
npm install react-router-dom --save
修改src/App.js,代码如下:
import React, { Fragment } from 'react'
import Login from './pages/login'
import Home from './pages/home'
import { HashRouter, Route, Switch, Redirect } from 'react-router-dom'
function App() {
return (
<Fragment> <HashRouter> <Switch> <Route path="/login" component={Login} /> <Route path="/home" component={Home} /> <Route exact path="/" component={Home} /> <Redirect to={"/home"} /> </Switch> </HashRouter> </Fragment>
)
}
export default App
App.js引入了Home和Login两个页面级组件。然后使用react-router-dom分别设置了路径。
import的机制是默认寻找index.js,所以每个组件的主文件名设为index.js,在引用的时候就可以省略文件名。
这里说明一下<Route>
的属性:
path表示路径,这个很好理解。
component表示绑定的组件。
exact表示是否精确匹配。
如果没有设置exact,那么:
localhost:3000/
会显示Home页,
localhost:3000/abc
也会显示Home页。
因为匹配到了前面的"/",路由就会成功。
最后的<Redirect>
表示以上都没有匹配成功的会,默认跳转的路由。
来看下效果:
访问http://localhost:3000/#/login
效果:
访问http://localhost:3000/#/home
效果:
接下来,简单介绍下如果在页面之间进行路由跳转。
在Login页面添加一个用于跳转至Home页的按钮,代码修改如下:
import React, { Component } from 'react'
import './login.styl'
class Login extends Component {
render() {
return (
<div className="P-login"> <h1>Login page</h1> + <button onClick={this.gotoHome.bind(this)}>跳转Home页</button> </div>
)
}
+ gotoHome() {
+ this.props.history.push('/home')
+ }
}
export default Login
注意button的onClick里要bind(this),否则,在gotoHome里的this是undefined。
当然,也可以这么写:
<button onClick={() => {this.gotoHome()}}>跳转Home页</button>
最终目的都是要让gotoHome中的this指向正确的作用域。
这章节内容也很容易,接触过vue的同学应该也很清楚,为了教程的完整性,还是简单说一下。下面来简单实现一个公用的头部组件。
目录结构如下:
| ├─ /components <-- 公共模块组件目录
+ | | ├─ /header <-- 公用header组件
+ | | | ├─ index.js
+ | | | └─ header.styl
src/components/header/index.js代码:
import React, { Component } from 'react'
import './header.styl'
class Header extends Component {
render() {
return (
<div className="M-header">
Header
</div>
)
}
}
export default Header
src/components/header/header.styl代码:
.M-header
height: 40px
line-height: 40px
font-size: 36px
color: #fff
background: #409EFF
然后,在Home和Login页面里引入Header组件。
以Home页面为例,修改src/pages/home/index.js:
import React, { Component } from 'react'
+ import Header from '../../components/header'
import './home.styl'
class Home extends Component {
render() {
return (
<div className="P-home"> + <Header /> <h1>Home page</h1> </div>
)
}
}
export default Home
同样的方式在Login页面也引入Header组件,代码就不放出了。效果如下:
使用过vue的同学都知道,vue组件有data和props。对应react的是state和props。
react向子组件传参使用以下方式:
<Header
param1="abc"
param2="c"
func1={()=>{console.log('func1')}}
/>
在Header组件内部,直接使用this.props就可以接收。例如:this.props.param1。
本次分享主要是介绍流程性的内容,篇幅有限,关于react的state和props请查阅官方文档。
为了更方便调试react项目,建议安装chrome插件。
先科学上网,在chrome网上应用店里搜索“React Developer Tools”并安装。
安装完成后,打开chrome调试工具,可以清晰的看到react项目代码结构。
本次分享涉及的项目代码已全部上传至GitHub,有需要的同学可前往自行下载:
github.com/Yuezi32/rea…
本次分享(上篇)暂告一段落。在下篇中,将继续讲解以下内容:
7 Redux及相关插件
7.1 安装redux
7.2 安装react-redux
7.3 安装redux-thunk
7.4 安装浏览器Redux插件
7.5 创建store
7.6 复杂项目store分解
7.7 对接react-redux与store
7.8 启动Redux DevTools
7.9 安装使用immutable
8 Mock.js安装与使用
9 解决本地开发跨域问题
10 其他常用工具
11 附赠章节:集成Ant Design
11.1 安装Ant Design
11.2 实现按需加载
11.3 自定义主题颜色
敬请阅读:
《超全面详细一条龙教程!从零搭建React项目全家桶(下篇)》
欢迎关注我的个人微信公众号,随时获取最新文章。