ReactRouterv6抢先看
在撰写本文时,React Router v6 仍处于 alpha 阶段,但现在正是开始使用它并探索未来发展的好时机。 本指南将让您了解新功能/更改!
如您所知,主要维护者在 2018 年初分叉了 React Router 项目,以创建一个名为 Reach Router 的轻量级替代方案。
在此期间,这两个库都在增长,但是 Reach Router 的积极开发似乎将停止,并将合并到 即将推出的 React Router v6 🛣
随着即将发布,这里先睹为快!
跳到…
正在变得
这个顶级组件将被重命名。 但是,它的功能大部分保持不变。
// v5
import {
BrowserRouter,
Switch,
Route
} from 'react-router-dom';
function App() {
return (
<BrowserRouter>
<Switch>
<Route exact path="/"><Home /></Route>
<Route path="/profile"><Profile /></Route>
</Switch>
</BrowserRouter>
);
}
就放下在那里:
// v6
import {
BrowserRouter,
Routes,
Route
} from 'react-router-dom';
function App() {
return (
<BrowserRouter>
<Routes>
<Route path="/" element={<Home />} />
<Route path="profile/*" element={<Profile />} />
</Routes>
</BrowserRouter>
);
}
大变化
在 v6 中组件正在接受最大的检修。 幸运的是,这些新变化实际上会使其更易于使用!
component/render 属性将替换 element 属性:
import Profile from './Profile';
// v5
<Route path=":userId" component={Profile} />
<Route
path=":userId"
render={routeProps => (
<Profile routeProps={routeProps} animate={true} />
)}
/>
// v6
<Route path=":userId" element={<Profile />} />
<Route path=":userId" element={<Profile animate={true} />} />
如果您注意到,在 v6 中,现在传递道具要容易得多。 这否定了 v5 中 render 属性的使用。
嵌套路由更简单
v5 中的嵌套路由必须非常明确地定义。 这需要在这些组件中包含大量字符串匹配逻辑。 看 :
// v5
import {
BrowserRouter,
Switch,
Route,
Link,
useRouteMatch
} from 'react-router-dom';
function App() {
return (
<BrowserRouter>
<Switch>
<Route exact path="/" component={Home} />
<Route path="/profile" component={Profile} />
</Switch>
</BrowserRouter>
);
}
function Profile() {
let match = useRouteMatch();
return (
<div>
<nav>
<Link to={`${match.url}/me`}>My Profile</Link>
</nav>
<Switch>
<Route path={`${match.path}/me`}>
<MyProfile />
</Route>
<Route path={`${match.path}/:id`}>
<OthersProfile />
</Route>
</Switch>
</div>
);
}
在 v6 中,您可以删除字符串匹配逻辑。 也不需要useRouteMatch()! 结果非常小:
// v6
import {
BrowserRouter,
Routes,
Route,
Link,
Outlet
} from 'react-router-dom';
// Approach #1
function App() {
return (
<BrowserRouter>
<Routes>
<Route path="/" element={<Home />} />
<Route path="profile/*" element={<Profile/>} />
</Routes>
</BrowserRouter>
);
}
function Profile() {
return (
<div>
<nav>
<Link to="me">My Profile</Link>
</nav>
<Routes>
<Route path="me" element={<MyProfile />} />
<Route path=":id" element={<OthersProfile />} />
</Routes>
</div>
);
}
// Approach #2
// You can also define all
// <Route> in a single place
function App() {
return (
<BrowserRouter>
<Routes>
<Route path="/" element={<Home />} />
<Route path="profile" element={<Profile />}>
<Route path=":id" element={<MyProfile />} />
<Route path="me" element={<OthersProfile />} />
</Route>
</Routes>
</BrowserRouter>
);
}
function Profile() {
return (
<div>
<nav>
<Link to="me">My Profile</Link>
</nav>
<Outlet />
</div>
)
}
注意: 组件的使用方式如下{this.props.children}在反应路由器 v6 中。 这是 Reach Router 的一个非常受欢迎的功能!
有时您需要以编程方式导航。 例如,在用户提交表单后,他们需要被重定向到确认页面。 这是 v5 中的 useHistory 库,在 v6 中已重命名为 useNavigate:
// v5
import { useHistory } from 'react-router-dom';
function MyButton() {
let history = useHistory();
function handleClick() {
history.push('/home');
};
return <button onClick={handleClick}>Submit</button>;
};
现在 history.push() 将替换为 navigate():
// v6
import { useNavigate } from 'react-router-dom';
function MyButton() {
let navigate = useNavigate();
function handleClick() {
navigate('/home');
};
return <button onClick={handleClick}>Submit</button>;
};
在某些情况下,您需要替换浏览器历史记录中的 URL,而不是推送新 URL。 这在 v6 中略有改变:
// v5
history.push('/home');
history.replace('/home');
// v6
navigate('/home');
navigate('/home', {replace: true});
从 20kb 到 8kb
通过所有这些更改,您会期望捆绑包的大小会增加,但实际上它减少了一半! v5 的压缩包约为 20kb,而 v6 仅约为 8kb。
捆绑包大小是使用 BundlePhobia 工具计算的。
结论
我对 React Router v6 的发布感到非常兴奋。 希望这篇文章能让您了解它发布时会发生什么(应该很快)! 您可以在 最新版本说明 中阅读更多关于 React Router v6 的信息📝
有关新功能的完整列表,请参阅官方 React Router v6 迁移指南 🚏