React Hooks 是 React 16.8 的新增特性,它可以让你在函数组件中使用 state 以及其他的 React 特性。以下是八种常用的 React Hooks,以及它们的概念、示例、横向对比和难点解析。

  1. useState



import React, { useState } from 'react';
 
function Example() {
  const [count, setCount] = useState(0);
  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>Click me</button>
    </div>
  );
}
  1. useEffect



import React, { useState, useEffect } from 'react';
 
function Example() {
  const [count, setCount] = useState(0);
 
  // 类似于类中的componentDidMount和componentDidUpdate:
  useEffect(() => {
    // 更新文档的标题
    document.title = `You clicked ${count} times`;
  });
 
  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>Click me</button>
    </div>
  );
}
  1. useContext



import React, { useContext } from 'react';
import { ThemeContext } from './ThemeContext';
 
function Button() {
  const theme = useContext(ThemeContext);
  return (
    <button style={{ background: theme.background, color: theme.foreground }}>
      I am styled by theme context!
    </button>
  );
}
  1. useReducer



import React, { useReducer } from 'react';
 
function Example() {
  const [state, dispatch] = useReducer(reducer, initialState);
 
  function handleIncrementClick() {
    dispatch({ type: 'increment' });
  }
 
  return (
    <div>
      <p>Count: {state.count}</p>
      <button onClick={handleIncrementClick}>Increment</button>
    </div>
  );
}
 
function reducer(state, action) {
  switch (action.type) {
    case 'increment':
      return { count: state.count + 1 };
    default:
      throw new Error();
  }
}
  1. useCallback



import React, { useCallback } from 'react';
 
function Example() {
  const memoizedCallback = useCallback(() => {
    doSomething(a, b);
  }, [a, b]);
 
  return <SomeComponent onSomething={memoizedCallback} />;
}
  1. useMemo



import React, { useMemo } from 'react';
 
function Example() {
  const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);
 
  return <div>{memoizedValue}</div>;
}
  1. useRef



import React, { useRef } from 'react';
 
function Example() {
  const ref = useRef(null);
 
  function handleClick() {
    ref.current.focus();
  }
 
  return (



import React from 'react';
import { BrowserRouter as Router, Route, Link } from 'react-router-dom';
 
// 定义一个组件来展示路由的内容
const Home = () => <h2>Home</h2>;
const About = () => <h2>About</h2>;
const Users = () => <h2>Users</h2>;
 
// 使用Router定义路由
const App = () => (
  <Router>
    <div>
      <nav>
        <ul>
          <li>
            <Link to="/">Home</Link>
          </li>
          <li>
            <Link to="/about/">About</Link>
          </li>
          <li>
            <Link to="/users/">Users</Link>
          </li>
        </ul>
      </nav>
 
      {/* 路由匹配 */}
      <Route path="/" exact component={Home} />
      <Route path="/about/" component={About} />
      <Route path="/users/" component={Users} />
    </div>
  </Router>
);
 
export default App;

这个代码实例展示了如何在React应用中使用react-router-dom来定义和使用路由。它定义了三个组件(Home, About, Users),并通过Link组件提供导航链接。Route组件根据URL的变化来渲染对应的组件。这是一个基本的React路由使用示例,适合初学者学习和理解。

React Navigation 7 是一个强大且灵活的React Native导航库,它提供了丰富的API和组件来构建跨平台的导航界面。

以下是如何安装和使用React Navigation 7的基本步骤:

  1. 安装React Navigation 7:



npm install @react-navigation/native
  1. 安装额外的依赖项,例如react-native-screensreact-native-safe-area-context



npm install react-native-screens react-native-safe-area-context
  1. 链接原生模块(对于React Native <= 0.59):



react-native link
  1. 在代码中使用React Navigation 7创建一个基本的栈导航器:



import * as React from 'react';
import { View, Text } from 'react-native';
import { NavigationContainer } from '@react-navigation/native';
import { createStackNavigator } from '@react-navigation/stack';
 
// 定义导航器
const Stack = createStackNavigator();
 
// 定义屏幕
function HomeScreen() {
  return (
    <View style={{ flex: 1, alignItems: 'center', justifyContent: 'center' }}>
      <Text>Home Screen</Text>
    </View>
  );
}
 
function DetailsScreen() {
  return (
    <View style={{ flex: 1, alignItems: 'center', justifyContent: 'center' }}>
      <Text>Details Screen</Text>
    </View>
  );
}
 
// 使用导航器
const App = () => (
  <NavigationContainer>
    <Stack.Navigator>
      <Stack.Screen name="Home" component={HomeScreen} />
      <Stack.Screen name="Details" component={DetailsScreen} />
    </Stack.Navigator>
  </NavigationContainer>
);
 
export default App;

这个例子展示了如何创建一个包含两个屏幕的栈导航器,其中第一个屏幕是主页,第二个是详情页。NavigationContainer是React Navigation 7提供的顶级容器,用于管理应用内的路由状态。createStackNavigator用于创建一个栈导航器,并使用Stack.Screen组件来定义导航器中的每个屏幕。

React Native Drawer 是一个为React Native应用提供侧边栏导航功能的库。以下是如何使用该库的基本步骤:

  1. 安装库:



npm install --save react-native-drawer
  1. 安装依赖项(如果你使用的是react-native init创建的项目):



react-native link react-native-drawer
  1. 在你的代码中引入并使用Drawer组件:



import React, { Component } from 'react';
import { View, Text, Image } from 'react-native';
import Drawer from 'react-native-drawer';
import { Scene, Router, Actions } from 'react-native-router-flux';
 
class MyApp extends Component {
  render() {
    return (
      <Drawer
        type="displace"
        content={<MyDrawerContent/>}
        tapToClose
        openDrawerOffset={0.2} // 20% of window width
        panCloseMask={0.2} // 20% opacity of the content view
      >
        <Router>
          <Scene key="root">
            <Scene key="home" component={Home} title="Home" />
            <Scene key="profile" component={Profile} title="Profile" />
          </Scene>
        </Router>
      </Drawer>
    );
  }
}
 
function MyDrawerContent() {
  return (
    <View style={{flex: 1, backgroundColor: '#fff'}}>
      <Image source={require('./images/my_image.png')} />
      <Text>Hello, this is the Drawer content!</Text>
      <TouchableOpacity onPress={() => Actions.home()}>
        <Text>Go to Home</Text>
      </TouchableOpacity>
    </View>
  );
}
 
export default MyApp;

在这个例子中,我们创建了一个名为MyApp的组件,它使用Drawer组件作为其根组件,并在侧边栏内容中定义了自定义内容(MyDrawerContent)。我们使用react-native-router-flux来定义应用的路由,并在侧边栏内容中提供一个按钮,用于导航回主页。这只是一个简单的例子,实际应用可能需要更复杂的导航逻辑和组件。

关闭React Native项目中的黄色警告通常涉及到修改组件的props校验。你可以使用prop-types库来确保传入组件的props是有效的。

首先,确保安装了prop-types




npm install prop-types

然后,在你的组件中导入prop-types并定义你想要校验的props:




import PropTypes from 'prop-types';
 
class MyComponent extends React.Component {
  // ...组件的其余部分
}
 
MyComponent.propTypes = {
  // 指定prop名和类型
  exampleProp: PropTypes.string.isRequired,
};
 
// 如果你想为props设置默认值,可以这样做:
MyComponent.defaultProps = {
  exampleProp: '默认值',
};

propTypes中,你可以指定exampleProp是一个字符串(PropTypes.string)并且它是必需的(.isRequired)。如果你还想为prop设置默认值,可以在defaultProps中设置。

这样,当你传递给MyComponentexampleProp不符合propTypes定义的规则时,开发者工具控制台将不再显示黄色警告。




import React from 'react';
import { BrowserRouter, Routes, Route } from 'react-router-dom';
 
function Home() {
  return <h2>Home</h2>;
}
 
function About() {
  return <h2>About</h2>;
}
 
function Users() {
  return <h2>Users</h2>;
}
 
function App() {
  return (
    <BrowserRouter>
      <Routes>
        <Route path="/" element={<Home />} />
        <Route path="/about" element={<About />} />
        <Route path="/users/*" element={<Users />} />
      </Routes>
    </BrowserRouter>
  );
}
 
export default App;

这个例子展示了如何在React Router v6中定义路由。它使用BrowserRouter作为路由的顶层容器,并通过RoutesRoute组件配置具体的路由。每个Route定义了一个路径和与之关联的React组件。当用户访问对应的URL时,相应的组件会被渲染。这种方式简洁明了,易于理解和使用。

在React中使用Ant Design的Input组件时,如果你想让输入框在进入页面时自动聚焦,你可以通过使用useEffect钩子在组件挂载后设置聚焦。

以下是一个简单的例子:




import React, { useEffect } from 'react';
import { Input } from 'antd';
 
const AutoFocusInput = () => {
  useEffect(() => {
    // 使用refs获取输入框DOM元素,并调用focus方法
    const inputRef = React.createRef();
    inputRef.current.focus();
  }, []); // 确保useEffect只在组件挂载时执行一次
 
  return (
    <Input ref={inputRef} placeholder="自动聚焦的输入框" />
  );
};
 
export default AutoFocusInput;

在这个例子中,我们使用React.createRef()创建了一个ref,并将其传递给Input组件的ref属性。useEffect钩子用于设置聚焦,它的依赖数组为空数组,意味着它只会在组件挂载时执行一次。




import React from 'react';
import { View, Text, Image } from 'react-native';
import { TabNavigator } from 'react-native-tab-navigator';
 
class MyHomeScreen extends React.Component {
  render() {
    return (
      <View style={{ flex: 1, justifyContent: 'center', alignItems: 'center' }}>
        <Text>Home Screen</Text>
      </View>
    );
  }
}
 
class MyNotificationsScreen extends React.Component {
  render() {
    return (
      <View style={{ flex: 1, justifyContent: 'center', alignItems: 'center' }}>
        <Text>Notifications Screen</Text>
      </View>
    );
  }
}
 
class MyMessagesScreen extends React.Component {
  render() {
    return (
      <View style={{ flex: 1, justifyContent: 'center', alignItems: 'center' }}>
        <Text>Messages Screen</Text>
      </View>
    );
  }
}
 
export default class App extends React.Component {
  render() {
    return (
      <TabNavigator>
        <TabNavigator.Item
          title="Home"
          renderIcon={() => <Image source={require('./img/home.png')} style={{ width: 25, height: 25 }} />}
          renderSelectedIcon={() => <Image source={require('./img/home.png')} style={{ width: 25, height: 25 }} />}
          badgeText="1"
          selected={true}
          onPress={() => this.props.navigation.navigate('Home')}>
          <MyHomeScreen />
        </TabNavigator.Item>
        <TabNavigator.Item
          title="Notifications"
          renderIcon={() => <Image source={require('./img/notifications.png')} style={{ width: 25, height: 25 }} />}
          renderSelectedIcon={() => <Image source={require('./img/notifications.png')} style={{ width: 25, height: 25 }} />}
          onPress={() => this.props.navigation.navigate('Notifications')}>
          <MyNotificationsScreen />
        </TabNavigator.Item>
        <TabNavigator.Item
          title="Messages"
          renderIcon={() => <Image source={require('./img/messages.png')} style={{ width: 25, height: 25 }} />}
          renderSelectedIcon={() => <Image source={require('./img/messages.png')} style={{ width: 25, height: 25 }} />}
          onPress={() => this.props.navigation.navigate('Messages')}>
          <MyMessagesScreen />
        </TabNavigator.Item>
      </TabNavigator>
    );
  }
}

这个代码实例展示了如何在React Native应用中使用\`react-native-ta

在React中,可以使用useRef钩子来创建一个ref对象,并将其传递给子组件。父组件可以通过这个ref对象访问子组件的实例。

以下是一个简单的例子:




import React, { useRef, useEffect } from 'react';
 
// 子组件
function ChildComponent(props, ref) {
  useImperativeMethods(ref, () => ({
    getChildData: () => '子组件的数据'
  }));
  // ...
}
 
// 父组件
function ParentComponent() {
  const childRef = useRef();
 
  useEffect(() => {
    if (childRef.current) {
      // 使用ref访问子组件的方法
      const childData = childRef.current.getChildData();
      console.log(childData);
    }
  }, [childRef]);
 
  return (
    <div>
      <ChildComponent ref={childRef} />
    </div>
  );
}

在这个例子中,ChildComponent 是一个子组件,它使用了 useImperativeMethods 钩子(这里假设一个自定义的 useImperativeMethods 钩子,实际上应该是 useImperativeHandle)来暴露一个方法 getChildDataParentComponent 是父组件,它创建了一个ref对象 childRef 并将其传递给子组件。在子组件渲染之后,父组件通过 childRef.current 访问子组件实例,并调用子组件暴露的方法。

由于这个问题涉及到的内容较多,我将提供一个概览性的代码实例,展示如何在React Native项目中注册原生模块。




import android.os.Bundle; // 导入Android OS相关的类
import com.facebook.react.ReactActivity; // 导入React相关的ReactActivity类
import com.facebook.react.bridge.ReactMethod; // 导入注解,标记方法供JS调用
 
// 定义一个React原生模块
public class MyNativeModule extends ReactContextBaseJavaModule {
 
    MyNativeModule(ReactApplicationContext context) {
        super(context);
    }
 
    // 返回模块名,用于JS端注册
    @Override
    public String getName() {
        return "MyNativeModule";
    }
 
    // 标记方法,允许JS调用
    @ReactMethod
    public void sampleMethod(String stringArgument, int numberArgument, Callback callback) {
        // 实现方法逻辑
        callback.invoke("回调结果", true);
    }
}
 
// 主活动类
public class MainActivity extends ReactActivity {
 
    // 重写方法,注册模块
    @Override
    protected List<ReactPackage> getPackages() {
        return Arrays.<ReactPackage>asList(
            new MainReactPackage(),
            new ReactPackage() {
                @Override
                public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
                    List<NativeModule> modules = new ArrayList<>();
                    modules.add(new MyNativeModule(reactContext)); // 注册自定义模块
                    return modules;
                }
 
                @Override
                public List<Class<? extends JavaScriptModule>> createJSModules() {
                    return Collections.emptyList();
                }
 
                @Override
                public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
                    return Collections.emptyList();
                }
            }
        );
    }
}

这个代码实例展示了如何创建一个自定义的原生模块,并在React Native的Android项目中注册和暴露它。MyNativeModule类继承自ReactContextBaseJavaModule,并实现了一个简单的方法sampleMethod。在MainActivity中,我们通过创建一个新的ReactPackage实例并在其中注册了这个模块。这样,原生模块就可以在JavaScript中通过NativeModules访问了。