React Hook基础入门:useState 和 useEffect 使用指南

发布时间: 2024-01-07 21:02:40 阅读量: 41 订阅数: 46
# 1. React Hook 简介 React Hook 是一种在函数组件中使用状态和副作用的新方法。它于 React 16.8 版本引入,为开发者提供了更简洁、清晰的代码编写方式。本章将介绍 React Hook 的基本概念、使用场景以及与传统类组件的对比。 ### 1.1 React Hook 是什么? React Hook 是一种用于在函数组件中引入状态和副作用的技术。在 React 16.8 之前,为了在函数组件中使用状态和生命周期方法,我们通常需要将函数组件转换为类组件。而 React Hook 的出现,使得我们能够在不编写类的情况下,直接在函数组件中管理状态和处理副作用。 ### 1.2 为什么要使用 React Hook? 使用传统的类组件来管理状态和处理副作用,需要编写更多的模板代码,并且容易导致代码的冗余和混乱。而 React Hook 可以让我们更自由地组织代码,同时也更容易进行测试和重用。 此外,React Hook 还可以帮助我们更好地解决组件之间状态共享的问题。在之前的类组件中,需要使用高阶组件、Render Props 或者 Redux 等技术来实现组件之间的状态共享。而使用 React Hook,我们可以直接在函数组件中使用共享状态,更加简洁明了。 ### 1.3 React Hook 与传统类组件的对比 传统类组件通过继承 `React.Component` 类来定义组件,并且使用 `this.state` 和 `this.setState` 来管理组件内部的状态。而 React Hook 使用 `useState` 来声明和管理状态,使得代码更加简洁。 相比传统类组件的生命周期方法,React Hook 使用 `useEffect` 来处理组件的副作用,如数据获取、订阅和取消订阅等操作。同时,React Hook 的使用也更加灵活,可以根据需要在不同的生命周期阶段执行副作用。 在接下来的章节中,我们将详细介绍如何使用 `useState` 和 `useEffect` 这两个最常用的 React Hook。敬请期待下一章节的内容。 # 2. useState的使用指南 ### 2.1 useState的基本概念 在React的函数式组件中,useState是一个用于声明状态的Hook。它提供了一种在函数组件中使用状态的方式,取代了传统类组件中的this.state和this.setState方法。 ### 2.2 如何在函数式组件中使用useState? 使用useState非常简单,通过调用useState函数,我们可以返回一个包含状态值和改变状态的函数的数组。其中,第一个元素是当前的状态值,第二个元素是改变状态的函数。 在下面的示例中,我们将展示如何使用useState来管理一个计数器的状态: ```javascript import React, { useState } from 'react'; function Counter() { const [count, setCount] = useState(0); return ( <div> <p>当前计数:{count}</p> <button onClick={() => setCount(count + 1)}>增加</button> <button onClick={() => setCount(count - 1)}>减少</button> </div> ); } ``` 代码解析: - 我们首先使用import语句从react模块中引入useState函数。 - 在函数组件内部,我们调用useState(0)来声明一个名为count的状态变量,并将其初始值设置为0。 - useState函数返回一个数组,其中count是用于保存状态值的变量,setCount是用于更新状态的函数。 - 在返回的JSX中,我们可以通过{count}来显示当前的计数值,并通过onClick事件监听器来调用setCount函数更新计数值。 ### 2.3 useState的常见用法和注意事项 除了基本的用法外,useState还有一些常见的用法和一些需要注意的事项。 - 使用对象作为状态: ```javascript import React, { useState } from 'react'; function Form() { const [form, setForm] = useState({ name: '', age: 0 }); function handleInputChange(event) { // 更新name字段 setForm({ ...form, name: event.target.value }); } return ( <form> <input type="text" value={form.name} onChange={handleInputChange} /> </form> ); } ``` 在这个例子中,我们使用了一个包含name和age字段的对象作为状态。通过使用解构语法,我们可以轻松地更新这个对象的属性,而不是使用多个useState。 - 使用函数更新状态: ```javascript import React, { useState } from 'react'; function Counter() { const [count, setCount] = useState(0); function handleIncrement() { setCount((prevCount) => prevCount + 1); } return ( <div> <p>当前计数:{count}</p> <button onClick={handleIncrement}>增加</button> </div> ); } ``` 如果在更新状态时依赖于先前的状态,我们可以传递一个函数给setCount。这个函数接收先前的状态值,并返回新的状态值。 - 注意事项: - 使用多个useState声明状态时,每个useState都是独立的,互不相关。 - 在函数组件中,useState的声明顺序必须保持固定,不能在条件语句或循环中使用。 - useState懒初始化:可以提供一个函数,来延迟计算初始状态的值。 这是关于useState的基本使用指南。在下一章节中,我们将学习另一个重要的Hook - useEffect的使用。 # 3. useEffect的使用指南 在React Hook中,`useEffect`是一个非常重要的 Hook,它可以让我们在函数式组件中执行副作用操作。本章将深入探讨`useEffect`的基本概念、用途和使用方法。 #### 3.1 useEffect的基本概念 `useEffect` 是一个用于执行副作用操作的 Hook。副作用指的是改变 DOM、数据请求、订阅或者其他影响组件外部环境的操作。在函数式组件中,由于没有生命周期方法,因此使用`useEffect`来处理这些操作非常方便。 #### 3.2 useEffect的用途和作用 `useEffect`的作用是让我们在函数式组件中执行副作用操作。这包括但不限于: - 数据订阅与取消订阅 - 执行数据请求 - 执行 DOM 操作 - 设置定时器和清除定时器等 `useEffect`的使用可以帮助我们避免直接在组件渲染过程中执行副作用操作,从而更好地控制副作用的时机和频率。 #### 3.3 如何在函数式组件中使用useEffect? 使用`useEffect`非常简单,只需要在函数式组件内部调用`useEffect`并传入一个回调函数即可。这个回调函数就是我们需要执行的副作用操作。 ```javascript import React, { useEffect } from 'react'; function Example() { useEffect(() => { // 在这里执行副作用操作 console.log('useEffect 被调用了'); }, []); // 传入一个空数组作为第二个参数来指定只在组件挂载和卸载时执行 return ( <div> {/* 组件的 JSX */} </div> ); } ``` 在上面的例子中,我们传入一个空数组作为`useEffect`的第二个参数,这表示这个`useEffect`回调函数只会在组件挂载和卸载时执行。如果不传入第二个参数,`useEffect`会在每次组件渲染完成后都执行一次。 通过这种简单的方式,`useEffect`可以让我们更好地管理副作用操作,避免了传统类组件中生命周期方法带来的复杂性。 以上是关于`useEffect`的使用指南,希望能够帮助你更好地理解和使用这个重要的 React Hook。 # 4. useState和useEffect的结合使用 在前面的章节中,我们已经了解了useState和useEffect的基本使用方法。现在我们将学习如何在React组件中同时使用useState和useEffect,并且介绍一些使用中的注意事项和最佳实践。 ##### 4.1 如何在React组件中同时使用useState和useEffect? 在函数式组件中,我们可以使用useState来定义状态变量,并使用useEffect来处理副作用。让我们来看一个例子,假设我们有一个计时器组件,需要在组件加载后开始计时,并在卸载前停止计时: ```javascript import React, { useState, useEffect } from 'react'; const Timer = () => { const [time, setTime] = useState(0); useEffect(() => { const interval = setInterval(() => { setTime(prevTime => prevTime + 1); }, 1000); return () => { clearInterval(interval); }; }, []); return ( <div> <p>Time: {time}</p> </div> ); }; export default Timer; ``` 在上面的代码中,我们使用useState来定义了一个名为time的状态变量,并将初始值设为0。然后,我们使用useEffect来创建了一个计时器,每秒钟更新一次time的值。在useEffect的回调函数中,我们返回了一个清理函数,它会在组件卸载时被调用,并清除计时器。 我们还要注意到,在useEffect的第二个参数中传入了一个空数组`[]`。这表示该useEffect仅在组件挂载和卸载时执行一次。如果我们需要在time发生变化时重新执行useEffect,可以将time添加到依赖数组中,例如`[time]`。 ##### 4.2 useState和useEffect共同处理状态和副作用的最佳实践 当我们同时使用useState和useEffect时,可以通过合理地组织状态和副作用的代码,来提高代码的可读性和可维护性。这里有几个最佳实践值得我们注意: - 尽量将相关的状态和副作用代码放在一起,增加代码的可读性。 - 合理使用依赖数组,确保副作用仅在特定依赖发生变化时执行。 - 使用清理函数来清除副作用(如定时器、订阅等)。 - 避免滥用副作用,确保副作用的代码具有明确的功能和目的。 至此,我们已经学习了如何在React组件中同时使用useState和useEffect,并介绍了一些使用中的最佳实践。在下一章节中,我们将解答一些常见的问题,并提供相应的解决方案。 # 5. 常见问题及解决方案 ### 5.1 如何处理useState中的复杂状态? 在使用useState的过程中,可能会遇到需要处理复杂状态的情况,例如对象或数组。下面是一些处理该问题的解决方案: #### 方案一:使用对象或数组的解构赋值 可以使用解构赋值语法来获取和更新复杂状态中的某个属性或元素。例如,假设我们有一个包含name和age属性的对象状态: ```jsx const [user, setUser] = useState({ name: '', age: 0 }); // 更新name属性 const handleChangeName = (name) => { setUser({ ...user, name }); } // 更新age属性 const handleChangeAge = (age) => { setUser({ ...user, age }); } ``` 使用解构赋值可以方便地获取和更新对象中的属性。同样的原理也适用于数组中的元素。 #### 方案二:使用useReducer代替useState 当状态变得复杂且包含多个可选操作时,可以考虑使用useReducer来管理状态。useReducer是useState的替代方案,它通过一个reducer函数来管理状态的更新逻辑。下面是一个使用useReducer处理复杂状态的示例: ```jsx const initialState = { count: 0 }; const reducer = (state, action) => { switch (action.type) { case 'increment': return { count: state.count + 1 }; case 'decrement': return { count: state.count - 1 }; default: throw new Error(); } } const Counter = () => { const [state, dispatch] = useReducer(reducer, initialState); // 增加count const handleIncrement = () => { dispatch({ type: 'increment' }); } // 减少count const handleDecrement = () => { dispatch({ type: 'decrement' }); } return ( <div> Count: {state.count} <button onClick={handleIncrement}>+</button> <button onClick={handleDecrement}>-</button> </div> ); } ``` 通过使用useReducer,我们可以将状态更新逻辑集中到一个reducer函数中,使得代码更加清晰和可读。 #### 方案三:使用自定义Hook 如果处理复杂状态的逻辑在多个组件中需要共享,可以考虑使用自定义Hook来封装该逻辑,从而实现状态的复用。 ```jsx const useComplexState = () => { const [user, setUser] = useState({ name: '', age: 0 }); const handleChangeName = (name) => { setUser({ ...user, name }); } const handleChangeAge = (age) => { setUser({ ...user, age }); } return { user, handleChangeName, handleChangeAge, }; } const Profile = () => { const { user, handleChangeName, handleChangeAge } = useComplexState(); return ( <div> <input type="text" value={user.name} onChange={(e) => handleChangeName(e.target.value)} /> <input type="number" value={user.age} onChange={(e) => handleChangeAge(e.target.value)} /> </div> ); } ``` 通过使用自定义Hook,我们可以将复杂状态的逻辑与特定组件解耦,使得代码更加可维护和可重用。 ### 5.2 如何避免useEffect中的无限循环? 在使用useEffect时,可能会遇到无限循环的问题,即useEffect函数的依赖项(第二个参数)会导致useEffect被多次调用,从而陷入循环。下面是一些解决方案: #### 方案一:传递一个空数组作为依赖项 如果useEffect不依赖任何属性或状态,可以将一个空数组作为依赖项传递给useEffect。这样,useEffect只会在组件挂载和卸载时执行一次。 ```jsx useEffect(() => { // 该回调只会在组件挂载和卸载时执行一次 }, []); ``` #### 方案二:明确指定依赖项 如果useEffect依赖特定的属性或状态,需要明确指定这些依赖项。这样,只有在依赖项发生变化时,useEffect才会被调用。 ```jsx const [count, setCount] = useState(0); useEffect(() => { // 该回调只会在count发生变化时执行 console.log('count:', count); }, [count]); ``` #### 方案三:在useEffect内部处理依赖项 如果存在复杂的依赖关系,可以在useEffect内部处理依赖项,以避免出现无限循环。 ```jsx const [count, setCount] = useState(0); const [name, setName] = useState(''); useEffect(() => { if (name === 'foo') { setCount(count + 1); } }, [name]); ``` 在这个示例中,当name为'foo'时,会增加count的值。由于count也是依赖项之一,因此使用了count + 1而不是count++来避免循环调用。 ### 5.3 其他常见使用问题的解决方案 在使用useState和useEffect的过程中,可能会遇到其他一些常见问题。下面是一些解决方案: - 如何使用useEffect处理数据获取和异步操作? - 如何在useEffect中清除副作用? - 如何在useEffect内部定义异步函数? - 如何在useState中处理异步更新? - 如何在React Hook之间共享状态和逻辑? 以上问题的解决方案将在实践经验分享中进行介绍和讨论。在实际项目中,我们通常会遇到更多的场景和问题,需要根据具体情况进行解决。 # 6. 高级技巧与实践经验分享 在这一章中,我们将探讨一些高级技巧和实践经验,以便更好地使用 React Hook 中的 useState 和 useEffect。这些技巧会帮助我们提高代码的性能、复用性,并在实际项目中应用最佳实践。 ### 6.1 如何优化useState和useEffect的性能? #### 6.1.1 使用解构赋值获取特定状态 当使用 useState 声明多个状态时,我们可以使用解构赋值来获取特定的状态变量。这样做有助于提高代码的可读性,并减少冗余代码。例如: ```javascript const [count, setCount] = useState(0); const [name, setName] = useState('John'); ``` #### 6.1.2 使用useCallback和useMemo缓存回调函数和计算结果 在函数组件中,当一个回调函数作为 props 传入子组件时,每次组件重新渲染时,都会创建一个新的回调函数实例。为了避免这种性能损耗,我们可以使用 useCallback 来缓存回调函数。 同样的道理,在计算结果需要被缓存的情况下,我们可以使用 useMemo 来缓存计算结果。例如: ```javascript const MemoizedComponent = useMemo(() => <Component />, []); ``` #### 6.1.3 使用React.memo进行组件的浅比较 使用 React.memo 包裹函数组件能够进行浅比较,只在依赖的 props 改变时重新渲染组件,而不是每次父组件重新渲染时都重新渲染。例如: ```javascript const MemoizedComponent = React.memo(Component); ``` ### 6.2 使用自定义Hook提高代码复用性 自定义 Hook 是一种可以帮助我们提高代码复用性的方式。通过将共享的逻辑抽象为一个自定义 Hook,我们可以在不同的组件中复用这段逻辑。 例如,我们可以创建一个名为 useFetch 的自定义 Hook,用于封装网络请求逻辑。这样我们就可以在任意组件中使用该 Hook,从而复用网络请求逻辑。 ```javascript // useFetch.js import { useState, useEffect } from 'react'; const useFetch = (url) => { const [data, setData] = useState(null); const [loading, setLoading] = useState(true); useEffect(() => { const fetchData = async () => { try { const response = await fetch(url); const jsonData = await response.json(); setData(jsonData); setLoading(false); } catch (error) { console.error(error); } }; fetchData(); }, [url]); return { data, loading }; }; export default useFetch; ``` 然后在组件中使用自定义 Hook: ```javascript import useFetch from './useFetch'; const MyComponent = () => { const { data, loading } = useFetch('https://api.example.com/data'); if (loading) { return <div>Loading...</div>; } return <div>{data}</div>; }; ``` ### 6.3 实战经验分享:在实际项目中使用useState和useEffect的最佳实践 在实际项目中,正确地使用 useState 和 useEffect 是至关重要的。下面是一些实践经验的分享: 1. 尽量将每个 useState 和 useEffect 相关的逻辑放在独立的自定义 Hook 中,以提高代码复用性和可维护性。 2. 仅在有需要时才使用 useEffect,避免过多的副作用影响性能。 3. 在 useEffect 中正确地处理清除副作用的逻辑,防止内存泄漏。 4. 使用解构赋值来获取特定状态,提高代码的可读性。 5. 使用 useCallback 和 useMemo 来缓存回调函数和计算结果,避免不必要的重新渲染。 6. 使用 React.memo 进行组件的浅比较,减少不必要的渲染。 这些是一些使用 useState 和 useEffect 的最佳实践,希望对你在实际项目中使用 React Hook 有所帮助。 到此为止,我们已经完成了关于 React Hook 的基础入门指南。希望本文能帮助你更好地理解和使用 useState 和 useEffect。祝你在 React 开发中取得更多的成就!
corwn 最低0.47元/天 解锁专栏
买1年送3月
点击查看下一篇
profit 百万级 高质量VIP文章无限畅学
profit 千万级 优质资源任意下载
profit C知道 免费提问 ( 生成式Al产品 )

相关推荐

张诚01

知名公司技术专家
09级浙大计算机硕士,曾在多个知名公司担任技术专家和团队领导,有超过10年的前端和移动开发经验,主导过多个大型项目的开发和优化,精通React、Vue等主流前端框架。
专栏简介
本专栏旨在通过实战案例和经验分享,全面讲解React Hook的最佳实践和使用技巧。从React Hook的基础入门开始,逐步深入讲解useState、useEffect、useReducer等核心Hook的使用指南,以及在React应用状态管理中的优化技巧。随后涵盖了性能优化、副作用管理、组件复用性提高、原理解密、DOM操作、网络请求、路由管理、全局状态管理、动画效果实现、单元测试、无障碍功能实现、表单与表单验证逻辑、移动端开发、布局渲染问题解决以及大型数据列表优化等丰富内容。无论是初学者还是有一定经验的开发者,都能从中获取到丰富的知识和经验,并且能够在实际项目中得心应手地运用React Hook来构建高质量的应用。
最低0.47元/天 解锁专栏
买1年送3月
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
C知道 免费提问 ( 生成式Al产品 )

最新推荐

【时间序列分析】:如何在金融数据中提取关键特征以提升预测准确性

![【时间序列分析】:如何在金融数据中提取关键特征以提升预测准确性](https://img-blog.csdnimg.cn/20190110103854677.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjY4ODUxOQ==,size_16,color_FFFFFF,t_70) # 1. 时间序列分析基础 在数据分析和金融预测中,时间序列分析是一种关键的工具。时间序列是按时间顺序排列的数据点,可以反映出某

【线性回归时间序列预测】:掌握步骤与技巧,预测未来不是梦

# 1. 线性回归时间序列预测概述 ## 1.1 预测方法简介 线性回归作为统计学中的一种基础而强大的工具,被广泛应用于时间序列预测。它通过分析变量之间的关系来预测未来的数据点。时间序列预测是指利用历史时间点上的数据来预测未来某个时间点上的数据。 ## 1.2 时间序列预测的重要性 在金融分析、库存管理、经济预测等领域,时间序列预测的准确性对于制定战略和决策具有重要意义。线性回归方法因其简单性和解释性,成为这一领域中一个不可或缺的工具。 ## 1.3 线性回归模型的适用场景 尽管线性回归在处理非线性关系时存在局限,但在许多情况下,线性模型可以提供足够的准确度,并且计算效率高。本章将介绍线

【特征选择工具箱】:R语言中的特征选择库全面解析

![【特征选择工具箱】:R语言中的特征选择库全面解析](https://media.springernature.com/lw1200/springer-static/image/art%3A10.1186%2Fs12859-019-2754-0/MediaObjects/12859_2019_2754_Fig1_HTML.png) # 1. 特征选择在机器学习中的重要性 在机器学习和数据分析的实践中,数据集往往包含大量的特征,而这些特征对于最终模型的性能有着直接的影响。特征选择就是从原始特征中挑选出最有用的特征,以提升模型的预测能力和可解释性,同时减少计算资源的消耗。特征选择不仅能够帮助我

【高维数据降维挑战】:PCA的解决方案与实践策略

![【高维数据降维挑战】:PCA的解决方案与实践策略](https://scikit-learn.org/stable/_images/sphx_glr_plot_scaling_importance_003.png) # 1. 高维数据降维的基本概念 在现代信息技术和大数据飞速发展的背景下,数据维度爆炸成为了一项挑战。高维数据的降维可以理解为将高维空间中的数据点投影到低维空间的过程,旨在简化数据结构,降低计算复杂度,同时尽可能保留原始数据的重要特征。 高维数据往往具有以下特点: - **维度灾难**:当维度数量增加时,数据点在高维空间中的分布变得稀疏,这使得距离和密度等概念变得不再适用

大样本理论在假设检验中的应用:中心极限定理的力量与实践

![大样本理论在假设检验中的应用:中心极限定理的力量与实践](https://images.saymedia-content.com/.image/t_share/MTc0NjQ2Mjc1Mjg5OTE2Nzk0/what-is-percentile-rank-how-is-percentile-different-from-percentage.jpg) # 1. 中心极限定理的理论基础 ## 1.1 概率论的开篇 概率论是数学的一个分支,它研究随机事件及其发生的可能性。中心极限定理是概率论中最重要的定理之一,它描述了在一定条件下,大量独立随机变量之和(或平均值)的分布趋向于正态分布的性

p值在机器学习中的角色:理论与实践的结合

![p值在机器学习中的角色:理论与实践的结合](https://itb.biologie.hu-berlin.de/~bharath/post/2019-09-13-should-p-values-after-model-selection-be-multiple-testing-corrected_files/figure-html/corrected pvalues-1.png) # 1. p值在统计假设检验中的作用 ## 1.1 统计假设检验简介 统计假设检验是数据分析中的核心概念之一,旨在通过观察数据来评估关于总体参数的假设是否成立。在假设检验中,p值扮演着决定性的角色。p值是指在原

数据清洗的概率分布理解:数据背后的分布特性

![数据清洗的概率分布理解:数据背后的分布特性](https://media.springernature.com/lw1200/springer-static/image/art%3A10.1007%2Fs11222-022-10145-8/MediaObjects/11222_2022_10145_Figa_HTML.png) # 1. 数据清洗的概述和重要性 数据清洗是数据预处理的一个关键环节,它直接关系到数据分析和挖掘的准确性和有效性。在大数据时代,数据清洗的地位尤为重要,因为数据量巨大且复杂性高,清洗过程的优劣可以显著影响最终结果的质量。 ## 1.1 数据清洗的目的 数据清洗

【复杂数据的置信区间工具】:计算与解读的实用技巧

# 1. 置信区间的概念和意义 置信区间是统计学中一个核心概念,它代表着在一定置信水平下,参数可能存在的区间范围。它是估计总体参数的一种方式,通过样本来推断总体,从而允许在统计推断中存在一定的不确定性。理解置信区间的概念和意义,可以帮助我们更好地进行数据解释、预测和决策,从而在科研、市场调研、实验分析等多个领域发挥作用。在本章中,我们将深入探讨置信区间的定义、其在现实世界中的重要性以及如何合理地解释置信区间。我们将逐步揭开这个统计学概念的神秘面纱,为后续章节中具体计算方法和实际应用打下坚实的理论基础。 # 2. 置信区间的计算方法 ## 2.1 置信区间的理论基础 ### 2.1.1

正态分布与信号处理:噪声模型的正态分布应用解析

![正态分布](https://img-blog.csdnimg.cn/38b0b6e4230643f0bf3544e0608992ac.png) # 1. 正态分布的基础理论 正态分布,又称为高斯分布,是一种在自然界和社会科学中广泛存在的统计分布。其因数学表达形式简洁且具有重要的统计意义而广受关注。本章节我们将从以下几个方面对正态分布的基础理论进行探讨。 ## 正态分布的数学定义 正态分布可以用参数均值(μ)和标准差(σ)完全描述,其概率密度函数(PDF)表达式为: ```math f(x|\mu,\sigma^2) = \frac{1}{\sqrt{2\pi\sigma^2}} e

【品牌化的可视化效果】:Seaborn样式管理的艺术

![【品牌化的可视化效果】:Seaborn样式管理的艺术](https://aitools.io.vn/wp-content/uploads/2024/01/banner_seaborn.jpg) # 1. Seaborn概述与数据可视化基础 ## 1.1 Seaborn的诞生与重要性 Seaborn是一个基于Python的统计绘图库,它提供了一个高级接口来绘制吸引人的和信息丰富的统计图形。与Matplotlib等绘图库相比,Seaborn在很多方面提供了更为简洁的API,尤其是在绘制具有多个变量的图表时,通过引入额外的主题和调色板功能,大大简化了绘图的过程。Seaborn在数据科学领域得