无法在反应嵌套的useState挂钩中读取未定义的属性“状态”

时间:2020-11-01 19:02:27

标签: reactjs react-hooks use-state react-functional-component

我正在尝试将基于此类的类转换为基于功能组件的

这是实际的基于类别的代码

import React, { Component } from 'react';
import L from 'leaflet'
import { Map, TileLayer, Marker, Popup } from 'react-leaflet';
import './App.css';

// Icon properties
var myIcon = L.icon({
  iconUrl: '',
  iconSize: [25, 41],
  iconAnchor: [22, 94],
  popupAnchor: [-10, -90],
});


export default class Fullmap extends Component {
  state = {
    location: {
      lat: 51.097,
      lng: -1.505
  },
    haveUsersLocation: false,
    zoom: 2
  }

  componentDidMount(){
    // Get GeoLocation for setting marker from browser on page load
    navigator.geolocation.getCurrentPosition((position) =>{
      this.setState({
        location:{
          lat: position.coords.latitude,
          lng: position.coords.longitude  
        },
        haveUsersLocation: true,
        zoom: 13  
      });
    }, ()=>{
      // If denied location via browser, get via IP
      fetch('https://ipapi.co/json')
      .then(res => res.json())
      .then(location =>{
        this.setState({
          location:{
            lat: location.latitude,
            lng: location.longitude  
          },
          haveUsersLocation: true,
          zoom: 13  
        });
      })
    });
  }

  render(){
  const position = [this.state.location.lat, this.state.location.lng];
  return (
    
    <div className="App">
      
      <Map className="Map" center={position} zoom={this.state.zoom}>
        <TileLayer
          attribution='&amp;copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors'
          url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
        />
        {this.state.haveUsersLocation ? 
          <Marker
         icon={myIcon}
         position={position}>
          <Popup>
            A pretty CSS3 popup. <br /> Easily customizable.
          </Popup>
        </Marker> : ''
        }
        
      </Map>
      
    </div>
  );
}
}

这一切正常,没有错误,但是现在我将其转换为以下基于功能的代码

import React, { useState, useEffect } from 'react';
import L from 'leaflet'
import { Map, TileLayer, Marker, Popup } from 'react-leaflet';
import './App.css';

// Icon properties
var myIcon = L.icon({
  iconUrl: '',
  iconSize: [25, 41],
  iconAnchor: [22, 94],
  popupAnchor: [-10, -90],
});


export default function MyMap() {

  const [location, setLocation] = useState({
    lat: false,
    lng: null,
    zoom: 2,
    usersLocation: false
  });

  useEffect(() => { 
    // Get GeoLocation for setting marker from browser on page load
    navigator.geolocation.getCurrentPosition((position) =>{
      setLocation([...location,{ 
        lat: position.coords.latitude, 
        lng: position.coords.longitude,
        haveUsersLocation: true,
        zoom: 13
      }]);
    }, ()=>{
      // If denied location via browser, get via IP
      fetch('https://ipapi.co/json')
      .then(res => res.json())
      .then(location =>{
        setLocation([...location,{
          lat: location.coords.latitude, 
          lng: location.coords.longitude,
          haveUsersLocation: true,
          zoom: 13
        }]);
      })
    });
  });
  
  const position = [location.lat, location.lng];
  return (
    
    <div className="App">
      
      <Map className="Map" center={position} zoom={location.zoom}>
        <TileLayer
          attribution='&amp;copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors'
          url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
        />
        {this.state.haveUsersLocation ? 
          <Marker
         icon={myIcon}
         position={position}>
          <Popup>
            A pretty CSS3 popup. <br /> Easily customizable.
          </Popup>
        </Marker> : ''
        }
        
      </Map>
      
    </div>
  );
}

它给我的错误是 TypeError:无法读取第54行的未定义属性“状态”

如果您能帮助我指出我的错误并由于我是个新手,请多解释一下,我将不胜感激。

3 个答案:

答案 0 :(得分:3)

功能组件没有this来访问状态。您可以直接引用状态变量,具体情况为location

展开: 类组件this中的“类”是指存储状态的类实例。在功能组件中,您可以通过useState钩子返回的状态变量来访问状态

答案 1 :(得分:1)

仅类组件有权访问this。如果您想获取haveUsersLocation的当前状态,只需使用location.haveUsersLocation

答案 2 :(得分:1)

代替

{this.state.haveUsersLocation ? 
       <Marker
        ... 
} 

使用

{location.haveUsersLocation ? 
       <Marker
       ... 
}

您已经使用过useState const [location, setLocation] = useState ,因此您可以直接从位置变量获取更新状态。