开发者问题收集

React js TypeError:无法读取未定义的属性“map”

2020-09-20
182

使用 map 函数进行渲染时出现错误。它显示:

typeError Cannot read property 'map' of undefined

这是我的所有组件文件:

Menu component.js

import React, {Component} from 'react';
    import { Card, CardImg, CardImgOverlay,
        CardTitle } from 'reactstrap';

    import DishDetail from './DishdetailComponent';

    class Menu extends Component {

        constructor(props) {
            super(props);
        }

        renderDish(dish) {
            if(dish != null){
                return(
                    <DishDetail dish={dish}/>
                );
            }
            else{
                return(
                    <div></div>
                )
            }
        }

        render() {
            const menu = this.props.dishes.map((dish) => {
                return (
                <div className="col-12 col-md-5 m-1">
                    <Card key={dish.id} onClick={() => this.props.onClick(dish.id)}>
                    
                    <CardImg width="100%" src={dish.image} alt={dish.name} />
                    <CardImgOverlay>
                        <CardTitle>{dish.name}</CardTitle>
                    </CardImgOverlay>
                    </Card>
                </div>
                );
            });

            return (
                <div className="container">
                    <div className="row">
                        {menu}
                    </div>            
                </div>
            );
        }
    }

    export default Menu;

DishdetailComponent.js

import React , { Component } from 'react';
    import { Card, CardBody, CardImg, CardText, CardTitle } from 'reactstrap';


    class DishDetail  extends Component{

        constructor(props) {
            super(props);

            this.state = {
                selectedDish: null
            }
        }

        renderDish(dish){
            if(dish != null){
                return(
                    <Card>
                        <CardImg top  width="100%" src={dish.image} alt={dish.name} />
                        <CardBody>
                            <CardTitle>{dish.name}</CardTitle>
                            <CardText>{dish.description}</CardText>
                        </CardBody>
                    </Card>
                );
            }
            else{
                return(
                    <div></div>
                );
            }
        }

        renderComments(comments){
            if(comments != null){
                return(
                    <div className="container">
                        <h4>Comments</h4>
                        {comments.map((comment) => {
                            let date = new Intl.DateTimeFormat('en-US', {
                                year:'numeric',
                                month: 'short',
                                day: '2-digit'
                            }).format(new Date(Date.parse(comment.date)))
                            return(
                                <ul className="list-unstyled" key={comment.id}>
                                    <li className="comment">{comment.comment}</li>
                                    <br/>
                                    <li className="author">-- {comment.author}, {date}</li>
                                </ul>
                            );
                            })}
                    </div>
                )

            }
            else{
                return(
                    <div></div>
                )
            }
        }

        render(){

            const selectedDish = this.props.dish;

            return(
                <div className="container">
                    <div className="row">
                        <div className="col-12 col-md-5 m-1">
                            {this.renderDish(selectedDish)}
                        </div>
                        <div className="col-12 col-md-5 m-1">
                            {this.renderComments(selectedDish.comments)}
                        </div>
                    </div>
                </div>
            )
        }

    }

    export default DishDetail;

MainComponent.js

import React,{ Component } from 'react';
    import { Navbar, NavbarBrand } from 'reactstrap';
    import Menu from './MenuComponents';
    import DishDetail from './DishdetailComponent'
    import { DISHES } from '../shared/shareDishes.js';

    class Main extends Component {

    constructor(props) {
        super(props);

        this.state = {
        dishes: DISHES,
        selectedDish: null 
        }; 
    }

    onDishSelect(dishId) {
        this.setState({ selectedDish: dishId });
    }

    render() {
        return (
        <div>
            <Navbar dark color="primary">
            <div className="container">
                <NavbarBrand href="/">Ristorante confusion</NavbarBrand>
            </div>
            </Navbar>
            <Menu dishes = {this.state.dishes} 
                onClick={(dishId) => this.onDishSelect(dishId)} />
            <DishDetail dish={this.state.dishes.filter((dish) => dish.id === this.state.selectedDish[0])}/>
        </div>
        );
    }
    
    }

    export default Main;

这是 sharedDishes.js 文件

export const DISHES = [
        {
            id: 0,
            name:'Uthappizza',
            image: 'assets/images/uthappizza.png',
            category: 'mains',
            label:'Hot',
            price:'4.99',
            description:'A unique combination of Indian Uthappam (pancake) and Italian pizza, topped with Cerignola olives, ripe vine cherry tomatoes, Vidalia onion, Guntur chillies and Buffalo Paneer.',
            comments: [
                {
                id: 0,
                rating: 5,
                comment: "Imagine all the eatables, living in conFusion!",
                author: "John Lemon",
                date: "2012-10-16T17:57:28.556094Z"
                },
                {
                id: 1,
                rating: 4,
                comment: "Sends anyone to heaven, I wish I could get my mother-in-law to eat it!",
                author: "Paul McVites",
                date: "2014-09-05T17:57:28.556094Z"
                },
                {
                id: 2,
                rating: 3,
                comment: "Eat it, just eat it!",
                author: "Michael Jaikishan",
                date: "2015-02-13T17:57:28.556094Z"
                },
                {
                id: 3,
                rating: 4,
                comment: "Ultimate, Reaching for the stars!",
                author: "Ringo Starry",
                date: "2013-12-02T17:57:28.556094Z"
                },
                {
                id: 4,
                rating: 2,
                comment: "It's your birthday, we're gonna party!",
                author: "25 Cent",
                date: "2011-12-02T17:57:28.556094Z"
                }
            ]                        },
            {
            id: 1,
            name:'Zucchipakoda',
            image: 'assets/images/zucchipakoda.png',
            category: 'appetizer',
            label:'',
            price:'1.99',
            description:'Deep fried Zucchini coated with mildly spiced Chickpea flour batter accompanied with a sweet-tangy tamarind sauce',
            comments: [
                {
                id: 0,
                rating: 5,
                comment: "Imagine all the eatables, living in conFusion!",
                author: "John Lemon",
                date: "2012-10-16T17:57:28.556094Z"
                },
                {
                id: 1,
                rating: 4,
                comment: "Sends anyone to heaven, I wish I could get my mother-in-law to eat it!",
                author: "Paul McVites",
                date: "2014-09-05T17:57:28.556094Z"
                },
                {
                id: 2,
                rating: 3,
                comment: "Eat it, just eat it!",
                author: "Michael Jaikishan",
                date: "2015-02-13T17:57:28.556094Z"
                },
                {
                id: 3,
                rating: 4,
                comment: "Ultimate, Reaching for the stars!",
                author: "Ringo Starry",
                date: "2013-12-02T17:57:28.556094Z"
                },
                {
                id: 4,
                rating: 2,
                comment: "It's your birthday, we're gonna party!",
                author: "25 Cent",
                date: "2011-12-02T17:57:28.556094Z"
                }
            ]
            },
            {
            id: 2,
            name:'Vadonut',
            image: 'assets/images/vadonut.png',
            category: 'appetizer',
            label:'New',
            price:'1.99',
            description:'A quintessential ConFusion experience, is it a vada or is it a donut?',
            comments: [
                {
                id: 0,
                rating: 5,
                comment: "Imagine all the eatables, living in conFusion!",
                author: "John Lemon",
                date: "2012-10-16T17:57:28.556094Z"
                },
                {
                id: 1,
                rating: 4,
                comment: "Sends anyone to heaven, I wish I could get my mother-in-law to eat it!",
                author: "Paul McVites",
                date: "2014-09-05T17:57:28.556094Z"
                },
                {
                id: 2,
                rating: 3,
                comment: "Eat it, just eat it!",
                author: "Michael Jaikishan",
                date: "2015-02-13T17:57:28.556094Z"
                },
                {
                id: 3,
                rating: 4,
                comment: "Ultimate, Reaching for the stars!",
                author: "Ringo Starry",
                date: "2013-12-02T17:57:28.556094Z"
                },
                {
                id: 4,
                rating: 2,
                comment: "It's your birthday, we're gonna party!",
                author: "25 Cent",
                date: "2011-12-02T17:57:28.556094Z"
                }
            ]
            },
            {
            id: 3,
            name:'ElaiCheese Cake',
            image: 'assets/images/elaicheesecake.png',
            category: 'dessert',
            label:'',
            price:'2.99',
            description:'A delectable, semi-sweet New York Style Cheese Cake, with Graham cracker crust and spiced with Indian cardamoms',
            comments: [
                {
                id: 0,
                rating: 5,
                comment: "Imagine all the eatables, living in conFusion!",
                author: "John Lemon",
                date: "2012-10-16T17:57:28.556094Z"
                },
                {
                id: 1,
                rating: 4,
                comment: "Sends anyone to heaven, I wish I could get my mother-in-law to eat it!",
                author: "Paul McVites",
                date: "2014-09-05T17:57:28.556094Z"
                },
                {
                id: 2,
                rating: 3,
                comment: "Eat it, just eat it!",
                author: "Michael Jaikishan",
                date: "2015-02-13T17:57:28.556094Z"
                },
                {
                id: 3,
                rating: 4,
                comment: "Ultimate, Reaching for the stars!",
                author: "Ringo Starry",
                date: "2013-12-02T17:57:28.556094Z"
                },
                {
                id: 4,
                rating: 2,
                comment: "It's your birthday, we're gonna party!",
                author: "25 Cent",
                date: "2011-12-02T17:57:28.556094Z"
                }
            ]
            }
    ]

我使用 id 从另一个文件中的数组对象获取数据并使用 map 函数对其进行迭代。但在渲染函数时它显示 TypeError 我已经检查过但我找不到未定义的属性。如何解决这个问题?

2个回答

确保 dishescommentstruthy 并且是一个 Array :

Array.isArray(this.props.dishes)  && this.props.dishes.map(...)


Array.isArray(comments) && comments.map(...)

祝你好运...

Aakash
2020-09-20

在初始化 Main.js 组件中的状态时,尝试将 selectedDish 值从 null 更改为 0。

this.state = {
        dishes: DISHES,
        selectedDish: 0 
        }; 
    }

此外,将 key={dish.id 从 Card 上移到包装 div

<div className="col-12 col-md-5 m-1" key={dish.id}>

CodeSandbox

在 DishDetail.js 组件中,下一部分是多余的(实际上目前 Menu 和 DishDetail 中的两个构造函数都是多余的)

this.state = {
   selectedDish: null
  };

因为组件正在从 Main.js 组件接收 selectedDish 作为 prop。

希望这能让你开始。

我建议坚持将文件命名为与您的组件相同。例如, DishDetail.js 用于 class DishDetail extends... Menu.js 用于 class Menu extends... ,并在源目录中创建组件文件夹 - src/components/ ,用于这些组件文件。

一切顺利!

dharmabumzee
2020-09-20