Lazy Loading ialah corak reka bentuk di mana sumber dimuatkan hanya apabila ia diperlukan. Ini berfaedah untuk menambah baik masa muat awal aplikasi React Native, mengurangkan penggunaan memori dan meningkatkan prestasi keseluruhan.
Skrin Pemuatan Malas (Pemisahan Kod):
Dalam React Native, pemuatan malas biasanya digunakan untuk komponen, terutamanya apabila anda mempunyai skrin berbeza yang mungkin tidak sering dikunjungi pengguna. Dengan malas memuatkan skrin ini, anda mengurangkan saiz berkas awal.
Lazy Loading dengan React.lazy() dan Suspense:
React memperkenalkan fungsi React.lazy() untuk membolehkan pemuatan malas komponen. Untuk menggunakan pemuatan malas, Suspense digunakan sebagai sandaran sehingga komponen dimuatkan.
Dalam penggunaan biasa, semua sumber, komponen, perpustakaan dan data dimuatkan terlebih dahulu apabila apl dimulakan. Pendekatan ini berfungsi dengan baik untuk aplikasi kecil tetapi boleh menjadi tidak cekap dan intensif sumber apabila apl itu berkembang, menjejaskan prestasi dan masa pemuatan.
import React from 'react'; import HomeScreen from './screens/HomeScreen'; import ProfileScreen from './screens/ProfileScreen'; const App = () => { return ( <> <HomeScreen /> <ProfileScreen /> </> ); }; export default App;
Penjelasan:
Dengan pemuatan malas, komponen, perpustakaan atau data dimuatkan hanya apabila diperlukan. Ini meningkatkan prestasi dengan mengurangkan masa muat awal dan penggunaan memori kerana hanya sumber yang diperlukan dimuatkan atas permintaan.
import React, { Suspense, lazy } from 'react'; import { ActivityIndicator } from 'react-native'; const HomeScreen = lazy(() => import('./screens/HomeScreen')); const ProfileScreen = lazy(() => import('./screens/ProfileScreen')); const App = () => { return ( <Suspense fallback={<ActivityIndicator size="large" color="#0000ff" />}> <HomeScreen /> <ProfileScreen /> </Suspense> ); }; export default App;
Penjelasan:
Feature | Normal Usage | Lazy Loading |
---|---|---|
Loading Strategy | Everything is loaded upfront when the app starts. | Components, resources, or data are loaded only when needed. |
Initial Load Time | Higher, as all resources are loaded at once. | Lower, as only essential components are loaded upfront. |
Memory Usage | Higher, as all components and resources are loaded into memory. | Lower, as only necessary components are loaded into memory. |
User Experience | Slower startup but smoother transitions once loaded. | Faster startup but slight delay when loading resources. |
Best for | Small applications with limited components. | Large applications where not all components are used initially. |
Implementation | Simpler, as everything is bundled at once. | Requires managing dynamic imports and possibly loading states. |
Lazy loading ensures that screens or components are only mounted when they are accessed (when the user navigates to them), thus improving performance, especially in apps with multiple screens.
import React, { Suspense, lazy } from 'react'; import { createStackNavigator } from '@react-navigation/stack'; import { NavigationContainer } from '@react-navigation/native'; import { ActivityIndicator } from 'react-native'; // Lazy load screens const HomeScreen = lazy(() => import('./screens/HomeScreen')); const ProfileScreen = lazy(() => import('./screens/ProfileScreen')); const Stack = createStackNavigator(); const App = () => { return ( <NavigationContainer> <Stack.Navigator> <Stack.Screen name="Home" component={() => ( <Suspense fallback={<ActivityIndicator size="large" color="#0000ff" />}> <HomeScreen /> </Suspense> )} /> <Stack.Screen name="Profile" component={() => ( <Suspense fallback={<ActivityIndicator size="large" color="#0000ff" />}> <ProfileScreen /> </Suspense> )} /> </Stack.Navigator> </NavigationContainer> ); }; export default App;
Explanation:
In React Native, lazy loading can be achieved using libraries like react-native-fast-image or manually handling image loading by tracking visibility with tools like IntersectionObserver.
react-native-fast-image is a performant image component that provides built-in lazy loading.
npm install react-native-fast-image
import React from 'react'; import { View, ScrollView, Text } from 'react-native'; import FastImage from 'react-native-fast-image'; const LazyLoadingImages = () => { return ( <ScrollView> <Text>Scroll down to load images</Text> <FastImage style={{ width: 200, height: 200 }} source={{ uri: 'https://example.com/my-image1.jpg', priority: FastImage.priority.normal, }} resizeMode={FastImage.resizeMode.contain} /> <FastImage style={{ width: 200, height: 200 }} source={{ uri: 'https://example.com/my-image2.jpg', priority: FastImage.priority.normal, }} resizeMode={FastImage.resizeMode.contain} /> </ScrollView> ); }; export default LazyLoadingImages;
Explanation:
In cases where you don't want to use a third-party library, you can implement lazy loading by tracking when an image enters the viewport using tools like IntersectionObserver (web) or a custom scroll listener in React Native.
import React, { useState, useEffect } from 'react'; import { View, Image, ScrollView } from 'react-native'; const LazyImage = ({ src, style }) => { const [isVisible, setIsVisible] = useState(false); const onScroll = (event) => { // Implement logic to determine if image is visible based on scroll position const { y } = event.nativeEvent.contentOffset; if (y > 100) { // Example: load image when scrolled past 100px setIsVisible(true); } }; return ( <ScrollView onScroll={onScroll} scrollEventThrottle={16}> <View> {isVisible ? ( <Image source={{ uri: src }} style={style} /> ) : ( <View style={style} /> )} </View> </ScrollView> ); }; const App = () => { return ( <LazyImage src="https://example.com/my-image.jpg" style={{ width: 200, height: 200 }} /> ); }; export default App;
Explanation:
When using Redux, you may want to lazy load certain reducers only when necessary, such as for specific screens or features.
Start by setting up a standard Redux store, but instead of adding all reducers upfront, create an injection method.
import { configureStore, combineReducers } from '@reduxjs/toolkit'; const staticReducers = { // Add reducers that are needed from the start }; export const createReducer = (asyncReducers = {}) => { return combineReducers({ ...staticReducers, ...asyncReducers, }); }; const store = configureStore({ reducer: createReducer(), }); // Store injected reducers here store.asyncReducers = {}; export default store;
In the above code:
Create a helper function to inject new reducers dynamically into the store.
// Helper function to inject a new reducer dynamically export function injectReducer(key, asyncReducer) { if (!store.asyncReducers[key]) { store.asyncReducers[key] = asyncReducer; store.replaceReducer(createReducer(store.asyncReducers)); } }
The injectReducer function checks if a reducer has already been added. If not, it injects it into the store and replaces the current root reducer.
Imagine you have a new page or feature that needs its own reducer. You can inject the reducer dynamically when this page is loaded.
import { lazy, Suspense, useEffect } from 'react'; import { useDispatch } from 'react-redux'; import { injectReducer } from './store'; import featureReducer from './features/featureSlice'; // The reducer for this feature const FeatureComponent = lazy(() => import('./components/FeatureComponent')); const FeaturePage = () => { const dispatch = useDispatch(); useEffect(() => { injectReducer('feature', featureReducer); // Dynamically load the reducer }, [dispatch]); return ( <Suspense fallback={<div>Loading...</div>}> <FeatureComponent /> </Suspense> ); }; export default FeaturePage;
Here:
The reducer for the feature is written as usual, using Redux Toolkit.
import { createSlice } from '@reduxjs/toolkit'; const featureSlice = createSlice({ name: 'feature', initialState: { data: [] }, reducers: { setData: (state, action) => { state.data = action.payload; }, }, }); export const { setData } = featureSlice.actions; export default featureSlice.reducer;
You might want to remove a reducer when it's no longer needed, for example, when navigating away from a page.
Here’s how you can remove a reducer:
export function removeReducer(key) { if (store.asyncReducers[key]) { delete store.asyncReducers[key]; store.replaceReducer(createReducer(store.asyncReducers)); } }
You can call this function when a feature or page is unmounted to remove its reducer from the store.
If you are using heavy third-party libraries, lazy loading them can help optimize performance.
import React, { useState } from 'react'; const HeavyComponent = React.lazy(() => import('heavy-library')); // React.lazy(() => import('moment')) const App = () => { const [showComponent, setShowComponent] = useState(false); return ( <View> <Button title="Load Heavy Component" onPress={() => setShowComponent(true)} /> {showComponent && ( <Suspense fallback={<Text>Loading...</Text>}> <HeavyComponent /> </Suspense> )} </View> ); };
Example: Lazy Loading Data:
import React, { useState, useEffect } from 'react'; import { FlatList, ActivityIndicator, Text } from 'react-native'; const LazyLoadData = () => { const [data, setData] = useState([]); const [loading, setLoading] = useState(true); useEffect(() => { fetch('https://api.example.com/data') .then(response => response.json()) .then(json => { setData(json); setLoading(false); }); }, []); if (loading) { return <ActivityIndicator />; } return ( <FlatList data={data} renderItem={({ item }) => <Text>{item.name}</Text>} keyExtractor={item => item.id} /> ); }; export default LazyLoadData;
Explanation:
Lazy loading helps improve the performance, memory usage, and user experience of your React Native app, making it more efficient and responsive for users.
Atas ialah kandungan terperinci Kuasai konsep pemuatan malas dalam React Native/ReactJS. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!