React js TypeError:无法读取未定义的属性“map”
使用 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 我已经检查过但我找不到未定义的属性。如何解决这个问题?
确保
dishes
和
comments
是
truthy
并且是一个
Array
:
Array.isArray(this.props.dishes) && this.props.dishes.map(...)
Array.isArray(comments) && comments.map(...)
祝你好运...
在初始化 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}>
在 DishDetail.js 组件中,下一部分是多余的(实际上目前 Menu 和 DishDetail 中的两个构造函数都是多余的)
this.state = {
selectedDish: null
};
因为组件正在从 Main.js 组件接收 selectedDish 作为 prop。
希望这能让你开始。
我建议坚持将文件命名为与您的组件相同。例如, DishDetail.js 用于 class DishDetail extends... , Menu.js 用于 class Menu extends... ,并在源目录中创建组件文件夹 - src/components/ ,用于这些组件文件。
一切顺利!