【问题标题】:React Native cannot update State Hook with data from AsyncStorageReact Native 无法使用来自 AsyncStorage 的数据更新 State Hook
【发布时间】:2020-06-02 09:26:37
【问题描述】:

您好,我目前正在学习 React Native 中的 AsyncStorage。 所以我写了一些这样的状态

const [isLoading,setIsLoading] = useState(true)
const [id, setId] = useState(null)

我将它设置为 null 因为我想从 AsyncStorage 获取 id,所以我编写了一个简单的函数来获取和设置值,如下所示:

async function getAsyncStorageId(){
  const userId = await AsyncStorage.getItem('id')
  console.log('user id from AsyncStorage is: ',userId)
  setId(userId)
  console.log('state hooks id : ',id)
}

我把它放入 useEffect Hooks

 useEffect(()=>{
  getAsyncStorageId()
 },[])

所以我测试了我的代码,id 状态不会更新它保持为空,但是 console.log(userId) 有一个 id 值。

user id from AsyncStorage is:  6
state hooks id :  null

我不知道为什么我不能使用异步存储中的值设置状态

======================== *edit : 这是我用来在 AsyncStorage 中设置 id 的代码。 我将此代码放在我的登录屏幕中,因此当用户成功登录时,我会将 id、用户名、令牌和角色存储在异步存储中

const login = async () => {
try {
  let response = await axios.post(`${url}/users/login`, {
    email: props.email,
    password: props.password,
  });
  let loginData = response.data;
  console.log(loginData,'sukses login')
  setAsyncStorage('token',loginData.token)
  loginData.name === null ? setAsyncStorage('name', 'pengguna') : setAsyncStorage('name',loginData.name)
  setAsyncStorage('role', loginData.role)
  setAsyncStorage('id', loginData.id)

  props.navigation.navigate("Artikel",{
    id:loginData.id,
    token:loginData.token,
    name:loginData.name
  })

} catch (error) {
  console.log('error login',props.email,props.password);
  alert(error.response.data.message)
}

};

export const setAsyncStorage = async (key, value) => {
  try {
    await AsyncStorage.setItem(`${key}`, `${value}`);
  } catch (error) {
    console.log(error);
  }
};

===============更新2==========================

所以我尝试将登录函数中的 id 值更改为这样的字符串

const login = async () => {
try {
  let response = await axios.post(`${url}/users/login`, {
    email: props.email,
    password: props.password,
  });
  let loginData = response.data;
  console.log(loginData,'sukses login')
  setAsyncStorage('token',loginData.token)
  loginData.name === null ? setAsyncStorage('name', 'pengguna') : setAsyncStorage('name',loginData.name)
  setAsyncStorage('role', loginData.role)
  setAsyncStorage('id', loginData.id.toString())

  props.navigation.navigate("Artikel",{
    id:loginData.id,
    token:loginData.token,
    name:loginData.name
  })

} catch (error) {
  console.log('error login',props.email,props.password);
  alert(error.response.data.message)
}

};

============================= 这是我的个人资料屏幕页面中的完整代码,所以我尝试使用 setTimeout 到 console.log(id) 并运行函数来获取用户 ID,但仍然没有运气

import React, { useState, useEffect } from 'react'
import { KeyboardAvoidingView, Text, View, AsyncStorage } from 'react-native'

import ProfileForms from './ProfileForms/ProfileForms'
import axios from 'axios'
import styles from "../styles";
import {url} from '../config/variables'
import { ScrollView } from 'react-native-gesture-handler';

export default function ProfileScreen (){
  const [isLoading,setIsLoading] = useState(true)
  const [id, setId] = useState(null)
  const [name, setName] = useState(null)
 
  async function getAsyncStorageId(){
    const userId = await AsyncStorage.getItem('id')
    console.log('user id from AsyncStorage is: ',userId)
    await setId(userId)
    console.log('state hooks id : ',id)
  }

  async function getUserData(){
    console.log('getting userr data')
    try {
      let result = await axios.get(`${url}/users/${id}`)
      console.log(result.data)
      setName(result.data.name)
      setIsLoading(false)
    } catch (error) {
      console.log(error)
    }
  }

  function test(){
    getAsyncStorageId()
    setTimeout(() => {
      console.log(id)
      getUserData()
    }, 5000);
  }

  useEffect(()=>{
    test()
  },[])

  return( 
    <ScrollView>
    <KeyboardAvoidingView behavior={Platform.OS == "ios"? "padding" : "height"} style={styles.container}>
      {
        !isLoading &&
        <ProfileForms
          id={id}
          name={name}
          setName={setName}
        />
      }
      {
        isLoading &&
        <View>
          <Text>Loading</Text>
        </View>
      }
    </KeyboardAvoidingView>
    </ScrollView>
  )
}

===============再次更新=================

所以我创建了一个变量,并用这个变量替换了我的 id 状态,它可以工作.. 目前我认为我会坚持使用这个

let desperateId = null



async function getAsyncStorageId(){
  const userId = await AsyncStorage.getItem('id')
  console.log('user id from AsyncStorage is: ',userId)
  desperateId=userId
  await setId(userId)
  console.log('state hooks id : ',id)
}

【问题讨论】:

  • 你能告诉我你在AsyncStorage中设置id的代码吗?
  • setId 是异步的,所以在调用setId 之后id 被记录为空是正常的。只需在setId(userId) 之前添加await,通常它应该可以工作
  • @MahdiN 我试过了,但值仍然是“null”
  • @MahdiN 但是当我编辑我的代码(任何地方)时,应用程序会自动刷新并以某种方式改变了值,但是当我重建并直接转到 profilescreen 时它不会改变
  • 我认为这是因为快速刷新,当您更改代码时,模拟器会用新代码刷新。我确定问题出在异步setId 上,在useEffect 中调用getAsyncStorageId 后尝试控制台日志ID 并将等待添加到getAsyncStorageId

标签: reactjs react-native react-native-android asyncstorage


【解决方案1】:

我认为您只能将string 类型存储在AsyncStorage 中。尝试像这样设置值。

AsyncStorage.setItem('id', '5')

你的情况是这样的

setAsyncStorage('id', JSON.stringify(loginData.id))

更新:

const [id, setId] = useState('')

【讨论】:

  • 我的 setAsyncStorage 自动将数字改为字符串
  • 问题是我不能用异步存储的值改变我的状态钩子
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2021-02-05
  • 1970-01-01
  • 2016-05-16
  • 1970-01-01
  • 2019-03-27
  • 2020-08-23
  • 1970-01-01
相关资源
最近更新 更多