- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的情况有 4 个组件按以下顺序相互嵌套:Products
(页面)、ProductList
、ProductListItem
和 CrossSellForm
.
Products
执行一个 graphql 查询(使用 urql):
const productsQuery = `
query {
products {
id
title
imageSrc
crossSells {
id
type
title
}
}
}
`;
...
const [response] = useQuery({
query: productsQuery,
});
const { data: { products = [] } = {}, fetching, error } = response;
...
<ProductList products={products} />
products
返回一个 Products
数组,其中包含一个字段 crossSells
,该字段返回一个 CrossSells
数组. Products
向下传播到 CrossSellForm
,其中包含一个变异查询,该查询返回一个 CrossSells
数组。
问题是,当我提交 crossSellForm 时,请求成功通过,但 Products
中的 crossSells
没有更新,并且 UI 反射(reflect)了陈旧的数据。只有当 Products
中的初始提取不包含 crossSells
时才会发生这种情况,因此初始响应如下所示:
{
data: {
products: [
{
id: '123123',
title: 'Nice',
imageSrc: 'https://image.com',
crossSells: [],
__typename: "Product"
},
...
]
}
}
}
如果存在现有的crossSell
,则没有问题,ui 会正确更新并且响应如下所示:
{
data: {
products: [
{
id: '123123',
title: 'Nice',
imageSrc: 'https://image.com',
crossSells: [
{
id: 40,
title: 'Nice Stuff',
type: 'byVendor',
__typename: 'CrossSell'
}
],
__typename: "Product"
},
...
]
}
}
}
我在 https://formidable.com/open-source/urql/docs/basics/ 阅读了一些关于 urql 缓存机制的内容。据我了解,它使用文档缓存,因此它根据 __typename
缓存文档。如果查询请求具有相同 __typename
的内容,它将从缓存中提取它。如果 突变
与相同的 __typename
发生,它将使缓存中具有该 __typename
的所有对象无效,因此下次用户获取对象时__typename
它将执行网络请求而不是缓存。
我认为发生的事情是在初始情况下有 products
但没有 crossSells
表单提交成功但 Products
页面不会更新,因为没有引用 CrossSell
的 __typename
对象,但在第二种情况下,它会破坏缓存并再次执行查询,刷新产品和交叉销售,UI 已正确更新。
我真的很享受将 urql hooks 与 React 组件一起使用的体验,并且想继续,但我不确定如何在不使用其他工具的情况下解决这个问题。
我尝试使用以下提示强制重新呈现表单提交:How can I force component to re-render with hooks in React?但它遇到了同样的问题,Products
将再次从缓存中获取,而 crossSells
将返回一个空数组。我考虑过将 urql 的 RequestPolicy 修改为仅网络,以及强制重新渲染,但我认为每次重新获取都会不必要地昂贵。我现在正在尝试的解决方案是将所有状态转移到 redux 中,这是一个单一的事实来源,以便对 crossSells
的任何更新都能正确传播,尽管我确信它会起作用这也意味着我将用标准 redux 样板的钩子(Hook)带来的便利进行交易。
如何在 CrossSellForm
中提交表单时使用 crossSells
优雅地更新 Products
,同时仍然使用 urql 和 hooks?
最佳答案
这里是核心贡献者 👋
正如您已经发现的那样,有一个 Unresolved 问题详细说明了我们简单的默认缓存的固有问题。它是一种文档缓存,不太适合规范化可以提供帮助的更复杂的任务。
当我们有一个空的数据数组时,没有迹象表明需要重新获取特定结果。
您可以尝试缓存和网络,而不是使用仅网络策略,但这并不能解决操作(您的查询)未因突变而无效的根本问题。因此不会触发重新获取。
我非常向您推荐 Graphcache,这是我们的规范化缓存,您也已经发现了它。至少在没有配置的情况下(!),它实际上是一个已经相当智能的直接替代品。 https://github.com/FormidableLabs/urql-exchange-graphcache
它的配置真的只是教它如何自动处理更多任务的插件!如果您需要自定义,我很乐意在这里或通过 Spectrum 帮助您解决问题。但我的建议是,试一试,因为在最好的情况下,所有边缘情况都可以正常工作,无需任何更改 ✨
关于javascript - 在初始查询响应不包含所需类型名的情况下,如何在发生突变时使用 urql 更新 graphql 缓存?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57836075/
我是 Java 新手,这是我的代码, if( a.name == b.name && a.displayname == b.displayname && a.linknam
在下面的场景中,我有一个 bool 值。根据结果,我调用完全相同的函数,唯一的区别是参数的数量。 var myBoolean = ... if (myBoolean) { retrieve
我是一名研究 C++ 的 C 开发人员: 我是否正确理解如果我抛出异常然后堆栈将展开直到找到第一个异常处理程序?是否可以在不展开的情况下在任何 throw 上打开调试器(即不离开声明它的范围或任何更高
在修复庞大代码库中的错误时,我观察到一个奇怪的情况,其中引用的动态类型从原始 Derived 类型更改为 Base 类型!我提供了最少的代码来解释问题: struct Base { // some
我正在尝试用 C# 扩展给定的代码,但由于缺乏编程经验,我有点陷入困境。 使用 Visual Studio 社区,我尝试通过控制台读出 CPU 核心温度。该代码使用开关/外壳来查找传感器的特定名称(即
这可能是一个哲学问题。 假设您正在向页面发出 AJAX 请求(这是使用 Prototype): new Ajax.Request('target.asp', { method:"post", pa
我有以下 HTML 代码,我无法在所有浏览器中正常工作: 我试图在移动到
我对 Swift 很陌生。我如何从 addPin 函数中检索注释并能够在我的 addLocation 操作 (buttonPressed) 中使用它。我正在尝试使用压力触摸在 map 上添加图钉,在两
我设置了一个详细 View ,我是否有几个 Nib 文件根据在 Root View Controller 的表中选择的项目来加载。 我发现,对于 Nibs 的类,永远不会调用 viewDidUnloa
我需要动态访问 json 文件并使用以下代码。在本例中,“bpicsel”和“temp”是变量。最终结果类似于“data[0].extit1” var title="data["+bpicsel+"]
我需要使用第三方 WCF 服务。我已经在我的证书存储中配置了所需的证书,但是在调用 WCF 服务时出现以下异常。 向 https://XXXX.com/AHSharedServices/Custome
在几个 SO 答案(1、2)中,建议如果存在冲突则不应触发 INSERT 触发器,ON CONFLICT DO NOTHING 在触发语句中。也许我理解错了,但在我的实验中似乎并非如此。 这是我的 S
如果进行修改,则会给出org.hibernate.NonUniqueObjectException。在我的 BidderBO 类(class)中 @Override @Transactional(pr
我使用 indexOf() 方法来精细地查找数组中的对象。 直到此刻我查了一些资料,发现代码应该无法正常工作。 我在reducer中尝试了上面的代码,它成功了 let tmp = state.find
假设我有以下表格: CREATE TABLE Game ( GameID INT UNSIGNED NOT NULL, GameType TINYINT UNSIGNED NOT NU
代码: Alamofire.request(URL(string: imageUrl)!).downloadProgress(closure: { (progress) in
我是一名优秀的程序员,十分优秀!