我正试图从FireBase存储中提取一个下载链接列表,以显示在我的React/NextJS组件中

nfeuvbwi  于 2023-06-22  发布在  React
关注(0)|答案(1)|浏览(72)

当试图拉取数组时,数据总是空的。有人知道我哪里错了吗?我检查了控制台,据我所知有一个对FireBase存储的引用,但没有数据。但我有4个文件在存储应该输出URL链接。这是我的组件

import React, { useState, useEffect } from 'react';

// i'm initializing firebase with firebase.initializeApp(firebaseConfig)
import { projectStorage } from '@/firebaseConfig'; 
 // projectStorage is =         firebase.storage();

const PDFViewer = () => {
  const [pdfUrls, setPdfUrls] = useState();

  useEffect(()=>{
   const storageRef = projectStorage.ref();
   const pdfRef = storageRef.child('myfirebaselocationpath/uploadeddocs/'); 
   const urls = [];
   pdfRef.listAll().then((result) => {
    result.items.forEach((itemRef) => {
    itemRef.getDownloadURL().then((url) => {
      urls.push(url);
    });
  });
  setPdfUrls(urls);
}).catch((error) => {
  console.error('Error retrieving PDF files:', error);
});
},[])
console.log(pdfUrls)
/*  console.log( pdfUrls.map((url) => {
return url
 })
) */
return (
<div>
  <h1>PDF Viewer</h1>
  {pdfUrls.map((url) => (
    <iframe key={url} src={url} width="100%" height="600px" />
  ))}
</div>
 );
};

export default PDFViewer;`

上面的组件是我尝试过的,并期待一个列表的链接Map,但没有得到任何数据显示。pdfUrls总是空的。

t2a7ltrp

t2a7ltrp1#

问题就在这里:

pdfRef.listAll().then((result) => {
  result.items.forEach((itemRef) => {
  itemRef.getDownloadURL().then((url) => {
    urls.push(url);
  });
});
setPdfUrls(urls);

listAllgetDownloadURL都是异步操作,可能需要一些时间才能完成。主代码不会阻塞整个脚本,而是在操作发生时继续运行,然后在数据(文件列表或下载URL)可用时执行回调块。
这意味着您的setPdfUrls(urls);在任何urls.push(url);运行之前运行,这将设置一个空列表。我建议在调试器中检查这个问题,或者添加一些日志记录。
最简单的解决方法是将setPdfUrls(urls);移入内部回调:

pdfRef.listAll().then((result) => {
  result.items.forEach((itemRef) => {
  itemRef.getDownloadURL().then((url) => {
    urls.push(url);
    setPdfUrls(urls);
  });
});

现在它将在确定每个下载URL后调用。这会执行多次更新,因此可能会导致UI中出现一些 Flink ,但数据现在应该会显示出来。
在等待一切完成的同时只调用setPdfUrls(urls);需要更多的工作。最简单的方法可能是将整个useEffect处理程序标记为async,并在其中使用await

useEffect(() => async {
  const storageRef = projectStorage.ref();
  const pdfRef = storageRef.child('myfirebaselocationpath/uploadeddocs/');
  const urls = [];
  const result = await pdfRef.listAll();
  for (const itemRef of result.items) {
    const url = await itemRef.getDownloadURL();
    urls.push(url);
  }
  setPdfUrls(urls);
}, [])

这段代码:

  • 使用await确保异步操作在调用setPdfUrls之前完成。
  • 使用一个for...of循环来确保我们可以在该循环中使用await

相关问题