跳到主要内容

平台特定代码

在构建跨平台应用时,你可能希望尽可能多地重用代码。但在某些情况下,代码需要有所不同才有意义,例如,你可能希望为 Android 和 iOS 实现单独的视觉组件。

React Native 提供了两种组织代码并按平台分离代码的方法

某些组件可能具有仅在一个平台上有效的属性。所有这些属性都使用 @platform 注释,并在网站上旁边有一个小徽章。

Platform 模块

React Native 提供了一个模块,用于检测应用程序运行所在的平台。你可以使用检测逻辑来实现平台特定的代码。当组件只有小部分是平台特定时,请使用此选项。

tsx
import {Platform, StyleSheet} from 'react-native';

const styles = StyleSheet.create({
height: Platform.OS === 'ios' ? 200 : 100,
});

当在 iOS 上运行时,Platform.OS 将为 ios;当在 Android 上运行时,则为 android

还有一个 Platform.select 方法可用,给定一个对象,其中键可以是 'ios' | 'android' | 'native' | 'default' 之一,它会返回最适合当前运行平台的 value。也就是说,如果你在手机上运行,iosandroid 键将优先。如果未指定这些键,则将使用 native 键,然后是 default 键。

tsx
import {Platform, StyleSheet} from 'react-native';

const styles = StyleSheet.create({
container: {
flex: 1,
...Platform.select({
ios: {
backgroundColor: 'red',
},
android: {
backgroundColor: 'green',
},
default: {
// other platforms, web for example
backgroundColor: 'blue',
},
}),
},
});

这将导致容器在所有平台上都具有 flex: 1,在 iOS 上具有红色背景颜色,在 Android 上具有绿色背景颜色,在其他平台上具有蓝色背景颜色。

由于它接受 any 值,你也可以使用它来返回平台特定的组件,如下所示

tsx
const Component = Platform.select({
ios: () => require('ComponentIOS'),
android: () => require('ComponentAndroid'),
})();

<Component />;
tsx
const Component = Platform.select({
native: () => require('ComponentForNative'),
default: () => require('ComponentForWeb'),
})();

<Component />;

检测 Android 版本
Android

在 Android 上,Platform 模块还可用于检测应用程序运行所在的 Android 平台版本

tsx
import {Platform} from 'react-native';

if (Platform.Version === 25) {
console.log('Running on Nougat!');
}

注意Version 设置为 Android API 版本,而不是 Android OS 版本。要查找映射,请参阅 Android 版本历史记录

检测 iOS 版本
iOS

在 iOS 上,Version-[UIDevice systemVersion] 的结果,这是一个包含当前操作系统版本的字符串。系统版本的示例为“10.3”。例如,要在 iOS 上检测主版本号

tsx
import {Platform} from 'react-native';

const majorVersionIOS = parseInt(Platform.Version, 10);
if (majorVersionIOS <= 9) {
console.log('Work around a change in behavior');
}

平台特定扩展名

当你的平台特定代码更复杂时,你应该考虑将代码拆分到单独的文件中。当文件具有 .ios..android. 扩展名时,React Native 将检测到,并在需要时从其他组件加载相关的平台文件。

例如,假设你的项目中有以下文件

shell
BigButton.ios.js
BigButton.android.js

然后,你可以按如下方式导入组件

tsx
import BigButton from './BigButton';

React Native 将根据运行平台自动选择正确的文件。

原生特定扩展名(即,与 NodeJS 和 Web 共享代码)

当模块需要在 NodeJS/Web 和 React Native 之间共享,但没有 Android/iOS 差异时,你还可以使用 .native.js 扩展名。这对于在 React Native 和 ReactJS 之间共享通用代码的项目特别有用。

例如,假设你的项目中有以下文件

shell
Container.js # picked up by webpack, Rollup or any other Web bundler
Container.native.js # picked up by the React Native bundler for both Android and iOS (Metro)

你仍然可以导入它,而无需 .native 扩展名,如下所示

tsx
import Container from './Container';

专业提示: 配置你的 Web 打包器以忽略 .native.js 扩展名,以避免在生产包中包含未使用的代码,从而减小最终包的大小。