- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个无法在此项目中更改的数据模型。我正在尝试精简和简化下面的示例代码,所以希望这对我要重现的内容仍然有意义。
假设我有两家商店。一家商店拥有“容器”,另一家商店拥有“元素”——每个商店都在整个应用程序中独立使用,用于各种目的。默认情况下,容器仅包含项目 ID。有时,我想对某些页面上使用的数据进行反规范化。我使用派生存储并对每个容器对象进行非规范化,将它们变成 DenormalizedContainers。
假设我有一个用例,我想创建一个新的 Item 对象,然后将其添加到给定的 Container。据我所知,这将导致派生商店更新两次(一次用于更改项目,一次用于更改容器),因此使用该派生商店的任何东西都将更新两次(取决于它的调用方式) .
为什么从 subscribe
到 $
的行为会有所不同?
此外,虽然这对我来说不是什么大问题,但我很好奇是否有任何本地 Svelte-y 方法可以在不更改数据模型的情况下解决这个问题(同时仍然能够使用 subscribe
API?
<script lang="ts">
import { derived, writable } from "svelte/store";
type Container = {
id: string;
itemIds: string[];
};
type Item = {
id: string;
name: string;
};
type DenormalizedContainer = {
id: string;
items: Item[];
};
const containers = writable<Container[]>([]);
const items = writable<Item[]>([]);
const denormalizedContainers = derived(
[containers, items],
([$containers, $items]): DenormalizedContainer[] => {
return $containers.map(({ id, itemIds }) => {
const denormalizedContainer = {
id,
items: itemIds.map((id) =>
$items.find((item) => {
item.id === id;
})
),
};
return denormalizedContainer;
});
}
);
function handleAddContainer() {
const value = Math.random() * 1000;
const newContainer: Container = { id: `new-container-${value}`, itemIds: [] };
containers.set([...$containers, newContainer]);
}
function handleAddItem() {
const value = Math.random() * 1000;
const newItem: Item = { id: `new-id-${value}`, name: `new-name-${value}` };
items.set([...$items, newItem]);
}
function handleAddBoth() {
const value = Math.random() * 1000;
const newItem: Item = { id: `new-id-${value}`, name: `new-name-${value}` };
const newContainer: Container = { id: `new-container-${value}`, itemIds: [newItem.id] };
items.set([...$items, newItem]);
containers.set([...$containers, newContainer]);
}
$: console.log(`$: There are ${$containers.length} containers`);
$: console.log(`$: There are ${$items.length} items`);
$: console.log(`$: There are ${$denormalizedContainers.length} denormalized containers`);
denormalizedContainers.subscribe((newValue) =>
console.log(
`Subscribe: There are ${$denormalizedContainers.length} denormalized containers`
)
);
</script>
<button class="block" on:click={() => handleAddContainer()}>Add container</button>
<button class="block" on:click={() => handleAddItem()}>Add item</button>
<button class="block" on:click={() => handleAddBoth()}>Add container and item</button>
单击每个按钮一次后,这里是日志:
页面加载:
Subscribe: There are 0 denormalized containers
$: There are 0 containers
$: There are 0 items
$: There are 0 denormalized containers
添加容器:
Subscribe: There are 1 denormalized containers
$: There are 1 containers
$: There are 1 denormalized containers
添加项目:
Subscribe: There are 1 denormalized containers
$: There are 1 items
$: There are 1 denormalized containers
同时添加:
Subscribe: There are 1 denormalized containers
Subscribe: There are 2 denormalized containers
$: There are 2 containers
$: There are 2 items
$: There are 2 denormalized containers
通过 $
处理的更新是我一直想要的,但出于某种原因,手动 subscribe
API 显示了两个更新。我在 .svelte
文件之外有几个位置,我需要在这些位置使用 subscribe
API。
与此类似,但没有解释为什么 $
的行为与 subscribe
不同。
Svelte Derived Store atomic / debounced updates
可能是潜在的问题?
https://github.com/sveltejs/svelte/issues/6730
可能在此处的 3.1/3.2 中进行了解释。微任务将 react 性语句捆绑在一起,并在 tick 结束时应用它们。而 subscribe
实时处理更新。注意:将我的 subscribe
代码包装在 tick
中只会使其使用相同的数据运行两次。
最佳答案
如果您查看编译后的输出,您会发现响应式语句是在 update
上触发的,每个事件循环应该只发生一次。
$$self.$$.update = () => {
if ($$self.$$.dirty & /*$containers*/ 256) {
$: console.log(`$: There are ${$containers.length} containers`);
}
if ($$self.$$.dirty & /*$items*/ 128) {
$: console.log(`$: There are ${$items.length} items`);
}
if ($$self.$$.dirty & /*$denormalizedContainers*/ 64) {
$: console.log(`$: There are ${$denormalizedContainers.length} denormalized containers`);
}
};
您可以创建自己的方法,在事件循环中捆绑更新。例如。使用 micro tasks :
function debouncedSubscribe(store, callback) {
let key;
return store.subscribe(value => {
key = {};
const currentKey = key;
queueMicrotask(() => {
if (key == currentKey)
callback(value);
});
});
}
debouncedSubscribe(denormalizedContainers, value =>
console.log(
`Subscribe: There are ${value.length} denormalized containers`
)
);
没有办法再次取消微任务,所以旧的微任务通过一个键作废。确保只执行最后一个任务的其他方法可能更优雅。
通过setTimeout
也可以做到,并且可以取消之前的,但是最小延迟会更大:
function debouncedSubscribe(store, callback) {
let timeout;
return store.subscribe(value => {
clearTimeout(timeout);
timeout = setTimeout(() => callback(value));
});
}
关于javascript - 为什么派生的 Svelte 商店在使用 `$` 和 `subscribe` 时会有不同的行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72805846/
我对 React 真的很陌生,需要问。 我可以有一个 ReactJs Redux store在库中,然后在也有 Redux 商店的应用程序中使用该库? 他们俩都这样做: ..A
我有两个商店版本的 Magento 安装:商店 A 和商店 B。当您转到“mydomain.com”时,我收到以下错误消息: 'There was no Home CMS page configure
我有一个按钮,单击该按钮时,将使用提供的 url 创建一个 JSONstore。然后将商店加载到网格中。如果再次单击该按钮,它会再次添加所有信息(因此会列出两次)。我想要这样,当用户单击按钮时,它会清
已结束。 这个问题是 off-topic .它目前不接受答案。 想要改进这个问题? Update the question所以它是on-topic堆栈溢出。 关闭 9 年前。 Improve this
我在 main.js 中进行 session API 调用,并将响应中的值用作我的根存储的初始值。在 vuex 中,它是这样处理的, DataService.getSession() .then(
我正尝试在 Svelte 商店中维护实时股票报价,以便屏幕上的出价和要价实时更新。我不确定商店的结构如何来保持数据的 react 性和效率。数据来自 websocket,如下所示: {'symbol'
将 Magento 商店从企业版 1.10.1.1 降级到社区版 1.7.0.0 应遵循什么程序? 我做的步骤是: 备份 Magento EE 1.10.1.1 数据库 :) 将此数据库导入到一个名为
我试图在过去 2 天使用内部应用程序共享上传我的应用程序,但无论我做什么,我都无法让它工作。这就是我所做的: 在控制台中,我点击了应用 -> 发布 -> 内部应用共享 我上传了 apk 我将自己添加到
现在的情况: 我有一个实时系统并且运行良好。 我没有测试系统。 我们的实时系统是一个多商店,在一个网站上有多个商店 View 。 问题: 我需要再添加一个 storeview 并在该 livesyst
我正在建立一个拥有零售店和批发商商店的网站。每个产品中的产品都不同,因此不仅仅是针对用户类型调整定价的问题。我需要保护批发部分的密码,以便它只对登录用户可用。我正在使用一个模块来实现这一点,但它只适用
我在应用商店上有几个应用程序,我每天都会检查所有可用的国家/地区,看看是否有人对我的应用程序留下了评论以及它在付费评级最高的位置中处于什么位置。 花时间看 iTunes 变得非常无聊。但我得到的信息非
我正在从 appsettings.json 加载配置文件,其中保存了 api url。 我已经建立了带有效果的ngrx 7商店。我在 app.component.ts onInit 中调度 loadc
我正在尝试为 Angular ngrx 存储架构中的操作编写 reducer : 这是我的 reducer : export const registration = (state: any, {ty
我创建了一个 Windows 商店应用程序,并使用 Visual Studio IDE 将其与商店相关联。在 Visual Studio IDE 中有用于创建应用程序包和上传应用程序包的菜单选项。 我
我有这个代码可以过滤我的商店 onLicenseGridSelect: function(rowmodel, record, index, eOpts) { Ext.getStore('Lic
我已将应用程序发布到 Play 商店。问题是我的应用程序的标题是 XXX YYY,但是当我输入 XXX YYY 时,搜索列表/索引中没有应用程序,但是如果我输入 XXXYYY,我可以找到我的应用程序。
我需要将所有翻译从主存储导出到另一个。 最佳答案 导出数据库中core_translate 表的内容。您可以为此使用 phpmyadmin。 关于php - 如何将内联翻译导出到另一个 Magento
我正在尝试将某些组件 (USERS) 连接到我的商店。我将向您展示每个步骤。 首先我在 index.js 中创建我的商店: // composeWithDevTools helps to follow
我不久前问过这个问题,并大致了解了商店的用途以及为什么使用它的基本解释。 但是,我目前正在为我的公司开发一个网络应用程序,并且遇到了一些计算问题。 首先,我有大约 35 个数据变量。一次只使用其中的几
我是 ngrx 的新手,我只是想了解它,让它工作起来。 我已将 ngrx(8.3 版)添加到我的应用程序中。 我希望有一些东西处于根状态(如果可能的话),然后我的每个功能都有单独的状态。我从根状态开始
我是一名优秀的程序员,十分优秀!