在JS类中处理异步数据
目标:我想建立一个解决新闻接口并在视图中呈现数据的类。 我的类(myNews
)有几种方法,一种方法是fetchnews()
。在这种方法中,我从API获取数据。来自API的数据应在类Varaible(News
)中分配。在另一种方法中(exhingews()
),我想访问数据以迭代它。
我的问题:我是JS的初学者,我想知道如何查询数据。从我的理解来看,流是这样的: 构造函数调用fetchnews Fetch News首先得到承诺,等到解决诺言得到解决 然后将数据分配给会员 但是JS Async正在运行JS,此时不再等到承诺得到解决,并且继续使用空数组。这正是我的问题。
问题我需要做什么或更改以正确获取数据。这样我就可以构建一个HTML列表。
我的JS代码
class MyNews {
news = [];
apiUrl = "https://jsonplaceholder.typicode.com/posts";
constructor() {
this.news;
this.fetchNews();
}
fetchNews() {
fetch(this.apiUrl)
.then(async (data) => {
this.news = await data.json();
console.log("A:",this.news.length); // i have results
})
.catch((err) => {
console.log(err)
});
// ...
}
showNews() {
console.log("B:",this.news.length); // empty array
return this.news;
}
}
n = new MyNews;
console.log("C:", n.showNews().length );
Goal: I want to build a class that addresses a news interface and renders the data in the view.
My class (MyNews
) has several methods and one method is fetchNews()
. In this method I get the data from the API. The data from the api should be assigned in a class varaible (news
). In another method (showNews()
) I want to access the data to iterate over it.
My problem: I am beginner in js and I wonder how to query the data. From my understanding the flow is like this:
constructor calls fetchNews
fetch news first gets a promise and waits until the promise is resolved
then the data is assigned to the member
but the JS async is running js doesn't wait at this point until the promise is resolved and it continues working with the empty array. This is exactly my problem.
Question What do I need to do or change to fetch the data correctly. So that I can build a HTML list for example.
My JS Code
class MyNews {
news = [];
apiUrl = "https://jsonplaceholder.typicode.com/posts";
constructor() {
this.news;
this.fetchNews();
}
fetchNews() {
fetch(this.apiUrl)
.then(async (data) => {
this.news = await data.json();
console.log("A:",this.news.length); // i have results
})
.catch((err) => {
console.log(err)
});
// ...
}
showNews() {
console.log("B:",this.news.length); // empty array
return this.news;
}
}
n = new MyNews;
console.log("C:", n.showNews().length );
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
data:image/s3,"s3://crabby-images/d5906/d59060df4059a6cc364216c4d63ceec29ef7fe66" alt="扫码二维码加入Web技术交流群"
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(5)
几个问题:
首先是
fetchnews
是一个异步过程。在您的示例中,显示的
将始终为0,因为fetchnews
在调用eashingews
时尚未获取任何内容。我们可以通过将这些方法的调用在班级外移动来减轻其中的一些。您有
fetch
和async/等待
的奇数混合物 - 最好坚持一个或另一个。A couple of issues:
The first is that
fetchNews
is an async process. In your exampleshowNews
will always be 0 becausefetchNews
hasn't fetched anything whenshowNews
is called. We can mitigate some of this by moving the calling of those methods outside of the class.You have an odd mix of
fetch
andasync/await
- it's best to stick with one or the other.问题在于
fetch()
是一个异步函数。这就是实际发生的事情:完成的构造函数。但是,这并不能完全解决您的问题,因为构造函数不能成为异步功能。
我建议使用此解决方案:
将
fetchnews
的呼叫移至myNews类上的init
方法,并将其变为异步。您可以在
顺便说一句:
this.news;
似乎不必要。The problem is that
fetch()
is an async function. This is what actually happens:First thing that comes to mind is using the await keyword or using promises directly. This does not fully solve your problem though, because constructor can not be an async function.
I recommend this solution:
Move the call of
fetchNews
to aninit
method on the MyNews class and make it async.You can find more on async constructors in this stackoverflow thread.
btw: the
this.news;
in the constructor seems unnecessary.我建议将
getandshownews()
函数添加到您的课程中。然后,我们将确保从
fetchnews()
提取呼叫中返回承诺,因此我们可以在另一个异步函数中等待等待。在
getandShownews()
函数中,我们将等待fetchnews()结果(承诺),这将确保填充新闻项目。然后,我们将通过循环浏览每个项目来展示新闻。
您可以在班级以外的功能中执行此操作,但是在班上这样做似乎很有意义。
I'd suggest adding a
getAndShowNews()
function to your class.We'll then ensure to return the promise from the
fetchNews()
fetch call, so we canawait
it in another async function.In the
getAndShowNews()
function we'll await the fetchNews() result (a promise), this will ensure the news items are populated.We'll then show the news by looping through each item.
You could do this in a function outside of the class, but it seems to make sense to do it in the class.
构造函数的调用
fetchnews
不等待它。它只是为稍后将要解决的东西创造了诺言(例如,手柄)。您的显示的
需要在fetchnews
之后链接。另外,我会用async
/等待
为了清楚起见,我还将从
fetchnews
返回新闻,而不是保留为类成员The constructor's call to
fetchNews
does NOT waits for it. It just creates a Promise (e.g a handle) to something that'll be resolved later. YourshowNews
needs to be chained after thefetchNews
. Also, I would write it withasync
/await
for clarityI would also return the news from
fetchNews
instead of keep as a class member对你来说还可以吗?
Is that OK for you ?