开发者问题收集

在我的 React Native 应用中获取 TypeError:null 不是对象

2022-03-20
145

登录后,用户被重定向到个人资料页面,我从 firestore(firebase)获取数据,然后使用它来显示个人资料详细信息,但我收到渲染错误 TypeError:null 不是对象。我检查了我的 firestore 文档是否包含我正在获取的所有数据。

我的个人资料页面源代码 ->

import { StyleSheet, View, Text, ScrollView, Image, TouchableOpacity } from "react-native";
import auth from '@react-native-firebase/auth';
import firestore from '@react-native-firebase/firestore'

function App() {
    const [userData, setUserData] = useState(null);
    const user = auth().currentUser;
    const data = firestore().collection('users').doc(user.uid).get().then(documentSnapshot => {
        setUserData(documentSnapshot.data())
    })

    return (
        <View style={styles.container}>
            <ScrollView style={styles.scrollView} contentContainerStyle={styles.contentContainer}>
                <View style={{alignSelf: 'center', margin: 20}}>
                    <View style={styles.userImg}>
                        <Image source={{uri: 'https://i.imgur.com/Wceq3H2.png'}} style={styles.image} resizeMode='center' />
                    </View>
                </View>
                <View style={styles.userData}>
                    <Text style={styles.userName}>Name: {userData.name}</Text>
                    <Text style={styles.userName}>Age: {userData.age}</Text>
                    <Text style={styles.userName}>Gender: {userData.gender}</Text>
                    <Text style={styles.userEmail}>Email: {userData.email}</Text>
                    <Text style={styles.userName}>Phone Number: {userData.phone}</Text>
                    <Text style={styles.userName}>Blood Group: {userData.blood}</Text>
                    <Text style={styles.userName}>Height: {userData.height}</Text>
                    <Text style={styles.userName}>Weight: {userData.weight}</Text>
                <View style={styles.text}>
                    <Text style={styles.subtext}>Want to Sign Out?</Text>
                    <TouchableOpacity onPress={() => auth().signOut()} style={{marginHorizontal: 5}}>
                        <Text style={{color: 'rgba(81,135,200,1)'}}>Sign out here</Text>
                    </TouchableOpacity>
                </View>
                </View>
            </ScrollView>
        </View>
    )
}

export default App;

const styles = StyleSheet.create({
    container: {
        flex: 1,
        backgroundColor: '#393636'
    },
    scrollView: {
        height: '20%',
        width: '100%',
        padding: 0,
        borderWidth: 0,
        borderRadius: 0,
    },
    contentContainer: {
        paddingBottom: 50
    },
    userImg: {
        width: 200,
        height: 200,
        borderRadius: 100,
        overflow: "hidden",
        paddingTop: 50
    },
    image: {
        flex: 1,
        height: undefined,
        width: undefined
    },
    userData: {
        marginLeft: 20,
        marginRight: 20,
        marginBottom: 20,
        padding: 30,
        borderRadius: 30,
        flex: 1,
        backgroundColor: '#302C2C',
        marginTop: 60
    },
    userName: {
        fontSize: 18,
        color: '#daf1f1',
        margin: 5
    },
    userEmail: {
        fontSize: 18,
        color: '#daf1f1',
        margin: 5
    },
    text: {
        flexDirection: "row",
        marginBottom: 20,
        marginTop: 8
    },
    subtext: {
        marginHorizontal: 5,
        color: '#a6a6a6'
    }
})```
2个回答

我建议您在屏幕挂载时使用 React 中的钩子 useEffect 来获取数据并将其添加到状态以显示在屏幕上。另一个提示是,使用可选链运算符来避免当对象上不存在属性并访问该属性时出现错误。在您的例子中, userData 被初始化为 null 。代码将如下所示:

import React, { useState, useEffect } from 'react'
import { StyleSheet, View, Text, ScrollView, Image, TouchableOpacity } from "react-native";
import auth from '@react-native-firebase/auth';
import firestore from '@react-native-firebase/firestore';

function App() {
  const [userData, setUserData] = useState(null);
  const user = auth().currentUser;

  const fetchDataFromFirestore = () => {
    firestore().collection('users').doc(user.uid).get().then(documentSnapshot => {
      setUserData(documentSnapshot.data())
    })
  }

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

  return (
    <View style={styles.container}>
      <ScrollView style={styles.scrollView} contentContainerStyle={styles.contentContainer}>
        <View style={{alignSelf: 'center', margin: 20}}>
          <View style={styles.userImg}>
            <Image source={{uri: 'https://i.imgur.com/Wceq3H2.png'}} style={styles.image} resizeMode='center' />
          </View>
        </View>
        <View style={styles.userData}>
          <Text style={styles.userName}>Name: {userData?.name}</Text>
          <Text style={styles.userName}>Age: {userData?.age}</Text>
          <Text style={styles.userName}>Gender: {userData?.gender}</Text>
          <Text style={styles.userEmail}>Email: {userData?.email}</Text>
          <Text style={styles.userName}>Phone Number: {userData?.phone}</Text>
          <Text style={styles.userName}>Blood Group: {userData?.blood}</Text>
          <Text style={styles.userName}>Height: {userData?.height}</Text>
          <Text style={styles.userName}>Weight: {userData?.weight}</Text>
          <View style={styles.text}>
            <Text style={styles.subtext}>Want to Sign Out?</Text>
            <TouchableOpacity onPress={() => auth().signOut()} style={{marginHorizontal: 5}}>
              <Text style={{color: 'rgba(81,135,200,1)'}}>Sign out here</Text>
            </TouchableOpacity>
          </View>
        </View>
      </ScrollView>
    </View>
  )
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#393636'
  },
  scrollView: {
    height: '20%',
    width: '100%',
    padding: 0,
    borderWidth: 0,
    borderRadius: 0,
  },
  contentContainer: {
    paddingBottom: 50
  },
  userImg: {
    width: 200,
    height: 200,
    borderRadius: 100,
    overflow: "hidden",
    paddingTop: 50
  },
  image: {
    flex: 1,
    height: undefined,
    width: undefined
  },
  userData: {
    marginLeft: 20,
    marginRight: 20,
    marginBottom: 20,
    padding: 30,
    borderRadius: 30,
    flex: 1,
    backgroundColor: '#302C2C',
    marginTop: 60
  },
  userName: {
    fontSize: 18,
    color: '#daf1f1',
    margin: 5
  },
  userEmail: {
    fontSize: 18,
    color: '#daf1f1',
    margin: 5
  },
  text: {
    flexDirection: "row",
    marginBottom: 20,
    marginTop: 8
  },
  subtext: {
    marginHorizontal: 5,
    color: '#a6a6a6'
  }
})

export default App;
marcelofreires
2022-03-21

试试这个

const unmounted = useRef(false);
const getData = () => {
    if (!unmounted.current) {
      const data = firestore().collection('users').doc(user.uid).get().then(documentSnapshot => {
          setUserData(documentSnapshot.data())
          setUserData(data)
        }
      }
      useEffect(() => {
        getData()
        return () => {
          unmounted.current = true;
        };
      }, [])

问题是你没有使用 useEffect..

Ahmed Khalil
2022-03-21