Reference version

metro.config.js

编辑页面

Metro 中可用配置的参考。


For the complete documentation index, see llms.txt. Use this Use this file to discover all available pages.

查看更多关于 metro.config.js 的信息,请参阅 自定义 Metro 指南

环境变量

Expo CLI 可以从 .env 文件中加载环境变量。有关如何在 Expo CLI 中使用环境变量的更多信息,请参阅 环境变量指南

EAS CLI 对环境变量使用不同的机制,除非它在编译和打包时调用 Expo CLI。了解更多关于 EAS 中的环境变量

如果你正在迁移一个较旧的项目,那么你应该通过在 .gitignore 中添加以下内容来忽略本地 env 文件:

.gitignore
# 本地 env 文件 .env*.local

禁用 dotenv 文件

可以在调用任何 Expo CLI 命令之前,通过启用 EXPO_NO_DOTENV 环境变量,来完全禁用 Expo CLI 加载 dotenv 文件。

Terminal
# 所有用户都可以运行 cross-env,然后接着运行 Expo CLI 命令
npx cross-env EXPO_NO_DOTENV=1 expo start
# 或者,macOS 和 Linux 用户可以先定义环境变量,然后运行 npx,再接着运行 Expo CLI 命令
EXPO_NO_DOTENV=1 npx expo start

禁用以 EXPO_PUBLIC_ 为前缀的客户端环境变量

EXPO_PUBLIC_ 为前缀的环境变量会在构建时暴露给应用。例如,EXPO_PUBLIC_API_KEY 将可作为 process.env.EXPO_PUBLIC_API_KEY 使用。

可以使用环境变量 EXPO_NO_CLIENT_ENV_VARS=1 来禁用客户端环境变量内联,这必须在进行任何打包之前定义。

Terminal
# 所有用户都可以运行 cross-env,然后接着运行 Expo CLI 命令
npx cross-env EXPO_NO_CLIENT_ENV_VARS=1 expo start
# 或者,macOS 和 Linux 用户可以先定义环境变量,然后运行 npx,再接着运行 Expo CLI 命令
EXPO_NO_CLIENT_ENV_VARS=1 npx expo start

CSS

信息 CSS 支持仍在开发中,目前仅适用于 web。

Expo 支持你项目中的 CSS。你可以从任何组件中导入 CSS 文件。CSS Modules 也受支持。

CSS 支持默认启用。你可以通过在 Metro 配置中设置 isCSSEnabled 来禁用该功能。

metro.config.js
/** @type {import('expo/metro-config').MetroConfig} */ const config = getDefaultConfig(__dirname, { // 禁用 CSS 支持。 isCSSEnabled: false, });

全局 CSS

警告 全局样式仅限 web 使用,使用它会导致你的应用在原生平台上的视觉表现发生偏差。

你可以从任何组件导入一个 CSS 文件。该 CSS 将应用到整个页面。

这里,我们将为类名 .container 定义一个全局样式:

styles.css
.container { background-color: red; }

然后我们可以通过导入样式表并使用 .container 来在组件中使用这个类名:

App.js
import './styles.css'; import { View } from 'react-native'; export default function App() { return ( <> {/* 使用 `className` 为 React DOM 组件分配样式。 */} <div className="container">Hello World</div> {/* 使用以下语法配合 `style`,为 React Native web 追加类名。 */} <View style={{ $$css: true, _: 'container', }}> Hello World </View> </> ); }

你也可以像导入任何 node 模块一样,导入库中捆绑的样式表:

index.js
// 将样式应用到整个应用。 import 'emoji-mart/css/emoji-mart.css';
  • 在原生平台上,所有全局样式表都会被自动忽略。
  • 全局样式表支持热重载,只需保存文件,变更就会应用。

警告 使用 Expo Router 时,请始终在根目录的 _layout.tsx 中导入全局 CSS。Expo Router 会从根布局开始遍历依赖图。在嵌套布局中导入 CSS 会导致 node_modules 中的 CSS 在你的自定义样式之前加载,从而可能破坏你预期的样式顺序。

CSS Modules

警告 原生平台的 CSS Modules 仍在开发中,目前仅适用于 web。

CSS Modules 是一种将 CSS 限定在特定组件范围内的方法。这有助于避免命名冲突,并确保样式只应用于预期的组件。

在 Expo 中,CSS Modules 通过创建一个扩展名为 .module.css 的文件来定义。该文件可以从任何组件中导入。导出的值是一个对象,类名作为键,仅限 web 的作用域名称作为值。可以使用 unstable_styles 导入来访问对 react-native-web 安全的样式。

CSS Modules 支持平台扩展名,允许你为不同平台定义不同的样式。例如,你可以定义 module.ios.cssmodule.android.css 文件,分别为 Android 和 iOS 定义样式。你需要在导入时不带扩展名,例如:

例如,反转扩展名,App.ios.module.css 将不起作用,并会生成一个名为 App.ios.module 的通用模块。

你不能将样式传递给 React Native 或 React Native for web 组件的 className 属性。相反,你必须使用 style 属性。

App.js
import styles, { unstable_styles } from './App.module.css'; export default function Page() { return ( <> <Text style={{ // 这就是 react-native-web 类名的应用方式 $$css: true, _: styles.text, }}> Hello World </Text> <Text style={unstable_styles.text}>Hello World</Text> {/* 仅限 web 的用法: */} <p className={styles.text}>Hello World</p> </> ); }
App.module.css
.text { color: red; }
  • 在 web 上,所有 CSS 值都可用。CSS 不会像 React Native Web 的 StyleSheet API 那样被处理或自动添加前缀。你可以使用 postcss.config.js 为你的 CSS 自动添加前缀。
  • CSS Modules 底层使用 lightningcss,请查看 这些 issue 了解不受支持的功能。

PostCSS

可以通过在项目根目录添加 postcss.config.json 文件来自定义 PostCSS。该文件应导出一个返回 PostCSS 配置对象的函数。例如:

postcss.config.json
{ "plugins": { "tailwindcss": {} } }

postcss.config.jsonpostcss.config.js 都受支持,但 postcss.config.json 可以实现更好的缓存。

Expo CLI 会借助对 browserslist 的内置支持,自动处理 CSS 厂商前缀。请避免添加 autoprefixer,因为这会重复该功能并减慢打包速度。

更新后重置缓存

更改 Post CSS 或 browserslist 配置后,需要清除 Metro 缓存:

Terminal
npx expo start --clear
npx expo export --clear

browserslist

Expo 通过基于 Rust 的 CSS 解析器自动支持 browserslist。你可以通过在 package.json 文件中添加 browserslist 字段来自定义 CSS 厂商前缀和浏览器支持。例如:

package.json
{ "browserslist": [">0.2%", "not dead", "not op_mini all"] }

SASS

Expo Metro 对 SCSS/SASS 提供 部分 支持。

要进行设置,请在项目中安装 sass 包:

Terminal
yarn add -D sass

然后,确保在 metro.config.js 文件中已完成 CSS 设置

  • 当安装了 sass 时,不带扩展名的模块将按以下顺序解析:scsssasscss
  • 仅在 sass 文件中使用预期的语法。
  • 当前不支持从 scss/sass 文件内部导入其他文件。

Tailwind

信息 标准 Tailwind CSS 仅支持 web 平台。若要获得通用支持,请使用类似 NativeWindUniwind 的库,它们允许使用 Tailwind CSS 创建带样式的 React Native 组件。

Tailwind CSS

了解如何在你的 Expo 项目中配置和使用 Tailwind CSS。

扩展 Babel 转换器

Expo 的 Metro 配置使用自定义的 transformer.babelTransformerPath 值,以确保始终使用 expo-babel-preset 并支持 web/Node.js 环境。

如果你想扩展 Babel 转换器,请从 @expo/metro-config/babel-transformer 导入上游转换器,而不是 metro-react-native-babel-transformer。例如:

metro.transformer.js
const upstreamTransformer = require('@expo/metro-config/babel-transformer'); module.exports.transform = async ({ src, filename, options }) => { // 对 SVG 文件执行一些自定义处理... if (filename.endsWith('.svg')) { src = '...'; } // 将源代码传递给上游 Expo 转换器。 return upstreamTransformer.transform({ src, filename, options }); };

自定义解析

Expo CLI 扩展了默认的 Metro 解析器,以添加 Web、Server 和 tsconfig 别名支持等功能。你也可以通过链式调用 config.resolver.resolveRequest 函数来类似地自定义 Metro 的默认解析行为。

metro.config.js
const { getDefaultConfig } = require('expo/metro-config'); /** @type {import('expo/metro-config').MetroConfig} */ const config = getDefaultConfig(__dirname); config.resolver.resolveRequest = (context, moduleName, platform) => { if (moduleName.startsWith('my-custom-resolver:')) { // 将模块名解析为文件路径的逻辑... // 注意:如果无法解析,请抛出错误。 return { filePath: 'path/to/file', type: 'sourceFile', }; } // 确保你调用默认解析器。 return context.resolveRequest(context, moduleName, platform); }; module.exports = config;

与传统打包器不同,Metro 在所有平台上共享同一个解析器函数。因此,你可以使用 context 对象在每次请求时动态修改解析设置。

模拟模块

如果你希望某个平台上的某个模块为空,可以从解析器返回 type: 'empty' 对象。下面的示例会使 lodash 在 web 上为空:

metro.config.js
const { getDefaultConfig } = require('expo/metro-config'); /** @type {import('expo/metro-config').MetroConfig} */ const config = getDefaultConfig(__dirname); config.resolver.resolveRequest = (context, moduleName, platform) => { if (platform === 'web' && moduleName === 'lodash') { return { type: 'empty', }; } // 确保你调用默认解析器。 return context.resolveRequest(context, moduleName, platform); }; module.exports = config;

这种技术等同于在 Webpack 或 Vite 中使用空外部依赖,但额外的好处是可以针对特定平台。

虚拟模块

Metro 目前不支持虚拟模块。你可以使用的一种类似方式,是在 node_modules/.cache/... 目录中创建一个模块,并将解析重定向到该文件。

下面的示例会在 node_modules/.cache/virtual/virtual-module.js 中创建一个模块,并将 virtual:my-module 的解析重定向到该文件:

metro.config.js
const path = require('path'); const fs = require('fs'); const { getDefaultConfig } = require('expo/metro-config'); /** @type {import('expo/metro-config').MetroConfig} */ const config = getDefaultConfig(__dirname); const virtualPath = path.resolve(__dirname, 'node_modules/.cache/virtual/virtual-module.js'); // 在生成目录中创建虚拟模块... fs.mkdirSync(path.dirname(virtualPath), { recursive: true }); fs.writeFileSync(virtualPath, 'export default "Hello World";'); config.resolver.resolveRequest = (context, moduleName, platform) => { if (moduleName === 'virtual:my-module') { return { filePath: virtualPath, type: 'sourceFile', }; } // 确保你调用默认解析器。 return context.resolveRequest(context, moduleName, platform); }; module.exports = config;

这可用于通过自定义导入来模拟 externals。例如,如果你想将 require('expo') 重定向到类似 SystemJS.require('expo') 的自定义内容,你可以创建一个导出 SystemJS.require('expo') 的虚拟模块,并将 expo 的解析重定向到该文件。

自定义转换

转换在 Metro 中有大量缓存。如果你更新了某些内容,请使用 --clear 标志查看更新。例如,npx expo start --clear

Metro 没有非常强大的用于转换文件的插件系统,因此改为使用 babel.config.js 和 caller 对象来自定义转换。

babel.config.js
module.exports = function (api) { // 获取 Expo CLI 正在为其转换的平台。 const platform = api.caller(caller => (caller ? caller.platform : 'ios')); // 检测打包操作是否用于 Hermes 引擎,例如 `'hermes'` | `undefined`。 const engine = api.caller(caller => (caller ? caller.engine : null)); // 是否为服务器环境进行打包,例如 API Routes。 const isServer = api.caller(caller => (caller ? caller.isServer : false)); // 是否为开发或生产环境进行打包。 const isDev = api.caller(caller => caller ? caller.isDev : process.env.BABEL_ENV === 'development' || process.env.NODE_ENV === 'development' ); // 确保配置不会被缓存,否则平台将不会更新。 api.cache(false); // 你也可以提供一种更健壮的 CONFIG 缓存失效方式: // api.cache.invalidate(() => platform); return { presets: ['babel-preset-expo'], plugins: [ // 基于平台添加一个插件... platform === 'web' && 'my-plugin', // 确保过滤掉假值。 ].filter(Boolean), }; };

如果 caller 没有 engineplatformbundler 等信息,那么请确保你使用的是用于 transformer 的 @expo/metro-config/babel-transformer。如果你使用的是自定义 transformer,那么它可能需要扩展 Expo transformer。

如果可能,请始终尝试在 resolver 中实现自定义逻辑,因为缓存会简单得多,也更容易理解。例如,如果你需要重映射一个 import,那么使用 resolver 将其解析到一个静态文件,比解析所有可能的导入方式并用 transformer 重新映射它们更简单也更快。

始终使用 babel-preset-expo 作为默认的 Babel preset,这可确保转换始终与 Expo 运行时兼容。babel-preset-expo 会在内部使用所有 caller 输入,以便针对给定的平台、引擎和环境进行优化。

Node.js 内置模块

当为服务器环境打包时,Expo 的 Metro 配置会根据当前 Node.js 版本自动支持将 Node.js 内置模块(fspathnode:crypto 等)外部化。如果 CLI 正在为浏览器环境打包,那么内置模块会先检查该模块是否已在本地安装,然后回退到一个空的 shim。例如,如果你安装了用于浏览器中的 path,则可以使用它;否则,该模块会自动被跳过。

环境设置

这些环境变量不会在测试环境中定义。

Expo 的 Metro 配置会注入构建设置,这些设置可以通过环境变量在客户端 bundle 中使用。所有变量都会被内联,不能动态使用。例如,process.env["EXPO_BASE_URL"] 将不起作用。

  • process.env.EXPO_BASE_URL 会暴露在 experiments.baseUrl 中定义的基础 URL。这在 Expo Router 中用于在部署时遵守生产环境的基础 URL。

Bundle 拆分

Expo CLI 会在生产环境中根据异步导入自动将 web bundle 拆分为多个 chunk。此功能要求安装 @expo/metro-runtime,并且在入口 bundle 的某处被导入(在 Expo Router 中默认可用)。

异步 bundle 的共享依赖会合并到单个 chunk 中,以减少请求次数。例如,如果你有两个异步 bundle 都导入了 lodash,那么该库会被合并到一个初始 chunk 中。

chunk 拆分的启发式规则无法自定义。例如:

math.js
index.js
math.js
export function add(a, b) { return a + b; }
index.js
import '@expo/metro-runtime'; // 这将被拆分到一个单独的 chunk 中。 import('./math').then(math => { console.log(math.add(1, 2)); });

当你运行 npx expo export -p web 时,bundle 会被拆分为多个文件,入口 bundle 会被添加到主 HTML 文件中。@expo/metro-runtime 会添加用于加载并执行异步 bundle 的运行时代码。

Source map debug ID

如果一个 bundle 导出时带有外部 source map,那么会在文件末尾添加一个 Debug ID 注解,并在 source map 中添加匹配的 debugId,以便将这些文件对应起来。如果未导出 source map,或者使用的是内联 source map,则不会添加该注解。

// <所有源代码> //# debugId=<deterministic chunk hash>

相关联的 *.js.map*.hbc.map source map 会是一个 JSON 文件,其中包含等效的 debugId 属性。debugId 会在 hermes 字节码生成之前注入,以确保在所有情况下都能匹配。

debugId 是 bundle 内容的确定性哈希,但不包含外部 bundle 拆分引用。这与用于创建 chunk 文件名的值相同,只是格式化为 UUID。例如,431b98e2-c997-4975-a3d9-2987710abd44

@expo/metro-config 会在 npx expo exportnpx expo export:embed 期间注入 debugIdnpx expo export:embed 中的任何额外优化步骤,例如 Hermes 字节码生成,都需要手动注入 debugId

Metro require 运行时

你可以通过环境变量 EXPO_USE_METRO_REQUIRE=1 可选地启用自定义 Metro require 实现。该运行时具有以下特性:

  • 使用人类可读的字符串模块 ID,使缺失模块错误更容易追踪。
  • 具有确定性 ID,在多次运行之间以及跨模块都保持一致(开发环境中的 React Server Components 需要此特性)。
  • 移除了对旧版 RAM bundle 的支持。

Magic import 注释

从 SDK 52 起在所有平台可用。

诸如 Workers 和 Node.js 等服务器环境支持在运行时导入任意文件,因此你可能希望保持 import 语法不变,而不是使用 Metro 的 require 系统。你可以在 import() 语句中使用 /* @metro-ignore */ 注释来选择不使用动态导入。

// 手动确保 `./my-module.js` 被包含在相对于模块的正确位置。 const myModule = await import(/* @metro-ignore */ './my-module.js');

Expo CLI 会跳过 ./my-module.js 依赖,并假定开发者已将其手动添加到输出 bundle 中。在内部,这用于导出自定义服务器代码,该代码会根据请求在文件之间动态切换。避免在原生 bundle 中使用此语法,因为在启用 Hermes 的 React Native 中通常无法使用 import()

许多 React 库都提供了 Webpack 的 /* webpackIgnore: true */ 注释来实现类似行为。为了弥合差异,我们也添加了对 Webpack 注释的支持,但建议你在应用中使用 Metro 对应的写法。

ES 模块解析

本节适用于从 SDK 53 起的所有平台。

Metro 会为 ES Module 的 import 和 CommonJS 的 require 使用不同的解析策略。

之前,Metro 采用经典的 Node.js 模块解析策略(与 v12 之前的 Node.js 版本一致),并做了一些额外调整来支持 ES Modules。在这种解析策略下,Metro 会从 node_modules、JS 文件中解析模块,必要时可以省略扩展名(例如 .js),并使用 package.json 中的 mainmodulereact-native 等字段。

现在,随着现代 ES Modules 解析策略的引入,Metro 会先从 node_modules 中解析模块,然后匹配不同的 package.json 字段,例如 exports包暴露的子路径的嵌套映射main

根据包的导入方式,这两种解析策略中的一种会被使用。通常,从 Node 模块中使用 import 导入的文件(而不是 require)会使用 ES Modules 解析策略,并在必要时回退到常规经典 Node.js 解析。未使用 ES Modules 解析,或者通过 CommonJS require 导入的文件,则会使用经典解析策略。

package.json:exports

在执行 ES Modules 解析时,Metro 会查看 package.json:exports 条件映射。这个映射将导入子路径和条件映射到 Node 模块包中的文件。

例如,一个始终暴露 index.js 文件,并且符合 Metro 经典 CommonJS 模块解析的包,可以通过 default 条件指定一个映射。

{ "exports": { "default": "./index.js" } }

不过,同时提供 CommonJS 和 ES Modules 入口点的包,可以通过 importrequire 条件提供映射。

{ "exports": { "import": "./index.mjs", "require": "./index.cjs" } }

默认情况下,Metro 会根据平台以及解析是从 CommonJS require 调用开始,还是从 ES Modules import 语句开始,来匹配不同的条件,并相应地更改条件。

对于原生平台,会添加 react-native 条件;对于 web 导出,会添加 browser 条件;对于服务器导出(例如 API routes 或 React Server functions),会添加 nodereact-serverworkerd 条件。这些条件不会按照它们定义的顺序匹配,而是会按照 package.json:exports 映射中的属性顺序进行匹配。

TypeScript 会独立于 Metro 执行 ES Module 解析,并且在其 compilerOptions.moduleResolution 配置选项设置为 "bundler"(这与 Metro 的行为更接近)或 "node16" / "nodenext" 时,也会遵守 package.json:exports 映射。不过,TypeScript 还会匹配 types 条件。因此,当某个包没有把 types 条件放在其 exports 映射的首位时,类型可能无法正确解析。

由于 exports 映射可能包含子路径,包导入不再一定需要匹配包 modules 文件夹中的某个文件,而可能是一个“重定向”导入。如果在 package.json:exports 中指定了,导入 'package/submodule' 可能会匹配到与 node_modules/package/submodule.js 不同的文件。

{ "exports": { ".": "./index.js", "./submodule": "./submodule/submodule.js" } }

如果你遇到与新的 ES Modules 解析策略不兼容或尚未准备好的包,你可以通过补丁修改其 package.json 文件并添加或修正其 package.json:exports 条件映射来解决问题。不过,也可以通过禁用 unstable_enablePackageExports 选项来阻止 Metro 在解析中使用 package.json:exports 映射。

metro.config.js
const { getDefaultConfig } = require('expo/metro-config'); /** @type {import('expo/metro-config').MetroConfig} */ const config = getDefaultConfig(__dirname); config.resolver.unstable_enablePackageExports = false; module.exports = config;

资产导入

当导入资产时,会创建一个虚拟模块来表示导入该资产所需的数据。

在原生平台上,资产将是一个数字 ID:123,等等,可以通过 require("@react-native/assets-registry/registry").getAssetByID(<NUMBER>) 来查找。在 web 和服务器平台上,资产会根据文件类型而变化。如果文件是图片,那么资产将是 { uri: string, width?: number, height?: number },否则资产将是一个表示该资产远程 URL 的 string。从 SDK 55 起,你可以使用 String(asset) 来获取 web 上任意资产的公开 URL,这不包括无法包含 toString 函数的 React Server Component 环境。

资产可以如下使用:

import { Image } from 'react-native'; import asset from './img.png'; function Demo() { return <Image source={asset} />; }

在 API 路由中,你始终可以假设资产的类型不会是数字:

import asset from './img.png'; export async function GET(req: Request) { const ImageData = await fetch( new URL( // 访问资产 URI。 asset.uri, // 附加到当前请求 URL 的 origin。 req.url ) ).then(res => res.arrayBuffer()); return new Response(ImageData, { headers: { 'Content-Type': 'image/png', }, }); }

Web workers

重要 此功能为 alpha 版本,并可能发生破坏性变更。

new Worker(new URL('./worker', window.location.href));

Expo Metro 提供实验性的 web worker 支持。此功能目前仅支持 web,不能在原生平台上使用,在原生平台上使用会触发错误 "Property 'Worker' doesn't exist"。

Web worker 可用于将工作卸载到 web 上的单独线程,从而让主线程保持响应。这对于计算开销较大的任务很有用,例如图像处理、加密或其他会阻塞主线程的任务。

Worker 可以使用 Blob 内联生成,但有时你可能希望利用 TypeScript 等现代特性,或导入其他模块。

Web worker 依赖于 Expo 的 bundle 拆分支持,这意味着你需要使用 Expo Router,或者安装并导入 @expo/metro-runtime。你也不能在 web worker 中使用环境变量 EXPO_NO_METRO_LAZY=1

请看下面这个将数字翻倍的 worker 示例:

worker.ts
self.onmessage = ({ data }) => { const result = data * 2; // 示例:将数字翻倍 self.postMessage(result); };

这个 worker 文件可以在主应用中作为 Worker 导入:

// worker 的类型为 `Worker` const worker = new Worker(new URL('./worker', window.location.href)); worker.onmessage = ({ data }) => { console.log(`Worker responded: ${data}`); }; worker.postMessage(5);

在幕后,Expo CLI 正在生成如下代码:

const worker = new Worker( new URL('/worker.bundle?platform=web&dev=true&etc', window.location.href) );

生成的 bundle URL 会根据开发/生产环境而变化,以确保 worker 被正确加载并打包。不同于传统的 bundle 拆分,worker 文件需要包含其自身对所有模块的副本,并且不能依赖主 bundle 中的公共模块。

原生 API Worker 传统上在 React Native 中不可用,并且 Expo SDK 也未提供,因此即使这个打包功能在技术上适用于所有平台,它也只对 web 有用。如果你希望同时支持原生平台,从理论上说你可以编写一个原生 Expo 模块来为 Worker API 提供 polyfill。或者,你也可以在 React Native Reanimated 中使用 "worklet" API,将工作卸载到原生端的单独线程。

另外,你也可以通过公共路径导入 Worker:先将一个转换后的 JS 文件放入 public 目录,然后在 worker 导入中使用一个变量引用它:

// 这将避免转换,并直接使用公共路径。 const worker = new Worker('/worker.js'); // 这个变量会破坏转换,导致使用字面路径而不是转换后的路径。 const path = '/worker.js'; const anotherWorker = new Worker(new URL(path, window.location.href));

Worker 构造函数中使用变量不支持打包。若要检查内部 URL,你可以使用内部语法 require.unstable_resolveWorker('./path/to/worker.js') 来获取 URL 片段。

Bare workflow 设置

本指南是版本相关的,在升级/降级 Expo 时需要重新查看。或者,使用 Expo Prebuild 来进行完全自动化的设置。

不使用 Expo Prebuild 的项目必须配置原生文件,以确保始终使用 Expo Metro 配置来打包项目。

这些修改旨在分别将 npx react-native bundlenpx react-native start 替换为 npx expo export:embednpx expo start

metro.config.js

确保 metro.config.js 继承自 expo/metro-config

metro.config.js
const { getDefaultConfig } = require('expo/metro-config'); const config = getDefaultConfig(__dirname); module.exports = config;

android/app/build.gradle

Android 的 app/build.gradle 必须配置为在生产打包时使用 Expo CLI。修改 react 配置对象:

ios/<Project>.xcodeproj/project.pbxproj

在你的 ios/<Project>.xcodeproj/project.pbxproj 文件中,替换以下脚本:

"Start Packager" 脚本

移除 "Start Packager" 脚本。必须在运行应用之前或之后使用 npx expo 启动开发服务器。

"Bundle React Native code and images" 脚本

或者,在 Xcode 项目中,选择 "Bundle React Native code and images" 构建阶段,并添加以下修改:

你可以设置 CLI_PATHBUNDLE_COMMANDENTRY_FILE 环境变量来覆盖这些默认值。

自定义入口文件

默认情况下,React Native 仅支持使用根目录下的 index.js 文件作为入口文件(或类似 index.ios.js 这样的平台特定变体)。Expo 项目允许使用任意入口文件,但这需要额外的裸工作流设置。

开发

可以通过使用 expo-dev-client 包来启用开发模式入口文件。或者,你也可以添加以下配置:

生产

在你的 ios/<Project>.xcodeproj/project.pbxproj 文件中,根据 Metro 将 "Bundle React Native code and images" 脚本替换为设置 $ENTRY_FILE

Android 的 app/build.gradle 必须配置为使用 Metro 模块解析来查找根入口文件。修改 react 配置对象: