将角消防员图书馆放置 - II
怎么入门文章编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《将角消防员图书馆放置 - II》,涉及到,有需要的可以收藏一下
>以前,我们创建了自己的firestore getters来返回适当的观察力,从而将诺言转变为可冷观察的诺言。今天,让我们继续使用其他命令,以正确地映射我们的数据。
>映射数据
>现在我们不依赖rxfire返回映射的文档id,我们将创建自己的转换器。
“ firestore有一个内置的converter withconverter to conconter tor map to and firestore。不,谢谢。我们将从这里拿走。getdoc()返回具有id的文档napshot,以及上述数据()。 getdocs()返回带有docs数组的querysnapshot,除其他外,这是返回的documentsnapshot对象。我们的类别模型现在看起来像这样:
// category.model
// mapping new instance
export interface icategory {
name: string | null;
id: string | null;
key?: string;
}
// static class or export functions, it's a designers choice
export class category {
public static newinstance(data: querydocumentsnapshot): icategory {
if (data === null) {
return null;
}
const d = data.data();
return {
id: data.id,
name: d['name'],
key: d['key']
};
}
// receives querysnapshot and maps out the `data() ` and id
public static newinstances(data: querysnapshot): icategory[] {
// read docs array
return data.docs.map(n => category.newinstance(n));
}
}
因此,现在的服务调用看起来像这样:
// category.service
getcategories(params: iwhere[] = []): observable<icategory[]> {
// ...
return from(getdocs(_query)).pipe(
map((res: querysnapshot) => {
// here map
return category.newinstances(res);
})
);
}
getcategory(id: string): observable<icategory> {
return from(getdoc(doc(this.db, 'categories', id))).pipe(
map((res: documentsnapshot) => {
// map
return category.newinstance(res);
}),
);
}
>让我们看看什么adddoc()返回。>
创建文档
>我创建了一个带有名称和键的快速表单,添加类别,以及一个服务呼叫以创建类别:
// components/categories.list
addnew() {
// read form:
const newcat: partial<icategory> = this.fg.value;
this.categoryservice.createcategory(newcat).subscribe({
next: (res) => {
// the category list should be updated here
console.log(res);
}
});
}
>该服务有望返回可观察到的,以下内容可能无法将其删除,返回的对象是一个文档重新率,仅是对具有id创建的文档的引用。
// category.service
createcategory(category: partial<icategory>): observable<any> {
return from(adddoc(collection(this.db, 'categories'), category)).pipe(
map(res => {
// i have nothing but id!
return {...category, id: res.id};
})
);
}
>可能看起来不错,但是我不希望在不浏览映射器的情况下直接使用id,解决方案可以像单独的映射器一样简单,或者我们可以稍微适应newinstance // category.model
// adjust newinstance to accept partial category
public static newinstance(data: documentsnapshot | partial<icategory>, id?: string): icategory {
if (data === null) {
return null;
}
const d = data instanceof documentsnapshot ? data.data() : data;
return {
id: data.id || id, // if data.id doesnt exist (as in adddoc)
name: d['name'],
key: d['key']
};
}
然后,在服务呼叫中,我们通过返回的id
// category.service
createcategory(category: partial<icategory>): observable<icategory> {
return from(adddoc(collection(this.db, 'categories'), category)).pipe(
map(res => {
// update to pass category and res.id
return category.newinstance(category, res.id);
}),
);
}
firestore语法的奇怪案例
>以下方法是如此相同,如果您不区分它们,您可能会在50岁之前失去头发。因此,我决定列出它们,然后忘记它们。
doc(collection(this.db,'cantories'))返回新生成的id >
setdoc(_document_reference,{... newcategory},选项)设置文档数据,并在文档参考中保存提供的id
adddoc(collection(this.db,'cantories'),{... newcategory});添加带有自动生成id的文档(这是doc(collection ...)然后setdoc())- >的快捷方式
- updatedoc(doc(this.db,'categories','staust_id'),partial_category)这会更新现有文档(doc(db ...)的快捷键(db ...),然后使用现有id)进行setdoc()
- >从前两个开始激起了混乱,它们看起来是如此相似,但是它们是不同的
-
// find document reference, or create it with new id const categoryref = doc(this.db, 'categories', '_somedocumentreferenceid'); // create a document reference with auto generated id const categoryref = doc(collection(this.db, 'categories'));
现在,返回的参考文档可用于创建实际文档,或对其进行更新。因此,预计将通过setdoc - 成功
- >如果文档不存在wit合并设置为true,它将创建一个带有部分数据的新文档。不好。小心。
// then use the returned reference to setdoc, to add a new document
setdoc(categoryref, {name: 'bubbles', key: 'bubbles'});
// pass partial document with options: merge to partially update an existing document
setdoc(existingcategoryref, {name: 'bubble'}, {merge: true});
>
// add a new documnet with a new generated id
adddoc(collection(this.db, 'categories'), {name: 'bubbles', key: 'bubbles'});
// update and existing document, this will throw an error if non existent
updatedoc(existingcategoryref, {name: 'bubbles'})
出于插图目的,这是创建新类别的另一种方法
// an alternative way to create
createcategory(category: partial<icategory>): observable<icategory> {
// new auto generated id
const ref = doc(collection(this.db, 'categories'));
return from(setdoc(ref, category)).pipe(
map(_ => {
// response is void, id is in original ref
return category.newinstance(category, ref.id);
})
);
}
更新文档
>在上面的构建中,我为类别详细信息创建了一个快速表单以允许更新。我仍然会通过所有字段,但是我们总是可以通过部分字段。
>
// category.sevice
// id must be passed in category
updatecategory(category: partial<icategory>): observable<icategory> {
return from(updatedoc(doc(this.db, 'categories', category.id), category)).pipe(
map(_ => {
// response is void
// why do we do this? because you never know in the future what other changes you might need to make before returning
return category.newinstance(category);
})
);
}
这是概念的证明,有一些方法可以拧紧松动的末端,例如强迫id通过,检查不存在的文档,返回不同的数据形状等。>
使用它,收集表格,并捕获id:
// categories/list.component
update(category: icategory) {
// gather field values
const cat: partial<icategory> = this.ufg.value;
// make sure the id is passed
this.categoryservice.updatecategory({...cat, id: category.id}).subscribe({
next: (res) => {
console.log(res); // this has the same category after update
}
});
}
删除文档
这是直截了当的,我们可能会选择退还布尔值以获得成功。
>
// category.sevice
deletecategory(id: string): observable<boolean> {
return from(deletedoc(doc(this.db, 'categories', id))).pipe(
// response is void, but we might want to differentiate failures later
map(_ => true)
);
}
我们只是通过传递id
来称其为
// category/list.component
delete(category: icategory) {
this.categoryservice.deletecategory(category.id).subscribe({
next: (res) => {
console.log('success!', res);
}
});
}
>没有直截了当的批量删除方式。让我们继续前进。
负责任地查询
>我们创建了一个iwher模型,以允许将任何查询传递到我们的类别列表。但是,我们应该控制在模型中查询的字段,并保护我们的组件免受现场变化的影响。我们还应该控制可以询问的内容,以及如何始终领先于火箱上的任何隐藏价格。
// where.model
// the param fields object (yes its plural)
export interface ifieldoptions {
label?: string; // exmaple
key?: string;
// other example fields
month?: date;
recent?: boolean;
maxprice?: number;
}
// map fields to where conditions of the proper firestore keys
export const maplistoptions = (options?: ifieldoptions): iwhere[] => {
let c: any[] = [];
if (!options) return [];
if (options.label) {
// mapping label to name
c.push({ fieldpath: 'name', opstr: '==', value: options.label });
}
if(options.key) {
c.push({ fieldpath: 'key', opstr: '==', value: options.key });
}
// maxprice is a less than or equal operator
if (options.maxprice) {
c.push({ fieldpath: 'price', opstr: '<=', value: options.maxprice });
}
if (options.month) {
// to implement, push only data of the same month
c.push(...mapmonth(options.month));
}
if (options.recent) {
const lastweek = new date();
lastweek.setdate(lastweek.getdate() - 7);
// things that happened since last week:
c.push({ fieldpath: 'date', opstr: '>=', value: lastweek });
}
return c;
}
我们服务的更改就像这样>
// category.sevice
// accept options of specific fields
getcategories(params?: ifieldoptions): observable<icategory[]> {
// translate fields to where:
const _where: any[] = (mapfieldoptions(fieldoptions))
.map(n => where(n.fieldpath, n.opstr, n.value));
// pass the where to query
const _query = query(collection(this.db, this._url), ..._where);
// ... getdocs
}
这是如何以最高价格获取一组文档的示例:
// example usage
something$ = someservice.getlist({maxprice: 344});
这更有意义。我添加了一个日期示例,请注意,即使firestore日期为type tymestamp,在javascript日期上的操作正常。该示例的想法是说明组件不应该真正知道实现的详细信息,因此获得“最新文档”,例如应该像这样的>
// example
something$ = someservice.getlist({recent: true});
另一个示例是获取日期,firestore直接处理javascript日期:// where.model
const mapmonth = (month: date): iwhere[] => {
const from = new date(month);
const to = new date(month);
from.setdate(1);
from.sethours(0, 0, 0, 0);
to.setmonth(to.getmonth() + 1);
to.sethours(0, 0, 0, 0); // the edge of next month, is last day this month
let c: iwhere[] = [
{ fieldpath: 'date', opstr: '>=', value: from },
{ fieldpath: 'date', opstr: '<=', value: to }
];
return c;
};
时间戳
firestore参考时间戳。
>
// somemodel
// use timestamp class to map from and to dates
const jsdate = new date();
// adddoc with
const newdata = {
fbdate: timestamp.fromdate(jsdate)
}
// documentsnapshot returns data() with date field, the type is timestamp
// use todate to convert to js date
return data['fbdate'].todate();
firestore lite
firebase为firestore提供了简单crud的lite版本,到目前为止,我们仅使用了lite版本中的那些。我更改了所有
导入{...}来自'@angular/fire/firestore';
导入{...}来自'@angular/fire/firestore/lite';
>构建块的确很小。我个人项目中减少的主要部分从502 kb下降到352 kb。
拦截
>使用httpclient,我们使用了截距函数来添加加载效果,那么我们如何使用解决方案来做到这一点?我们可以将所有呼叫调查到一个照顾它的http服务。我们的新服务应该与此有关:
// http.service
@injectable({ providedin: 'root' })
export class httpservice {
public run(fn: observable<any>): observable<any> {
// show loader here
showloader();
// return function as is
return fn
.pipe(
// hide loader
finalize(() => hideloader()),
// debug and catcherrors as well here
);
}
}
然后在服务中
// category.service
private httpservice = inject(httpservice);
getcategories(params?: iwhere[]): observable<icategory[]> {
// ...
// wrap in run
return this.httpservice.run(from(getdocs(q)).pipe(
map((res: querysnapshot) => {
return category.newinstances(res);
})
));
}
>上面的一个设计问题是类型检查的丢失,因为内部函数返回可观察到的
修复
// http.service
// ...
public run<t>(fn: observable<any>): observable<t> {
// ...
}
然后这样称呼:>
// category.service
return this.httpService.run<ICategory[]>(from(getDocs(q)).pipe(
map((res: QuerySnapshot) => {
return Category.NewInstances(res);
})
));
>查看我们先前使用状态管理创建的加载效果。我们需要的只是注入州服务,然后呼叫显示和隐藏。这使用了良好的ol'rxjs,我调查了信号,看不到它如何替换rxj,而不会造成大混乱。可能是一个罚款星期二。 就是这样。在50岁之前不要丢发头发。
你上火了吗?继续说话。还没有结束。
资源
stackblitz project
>在angular
中使用rxj创建负载效果
- 将angular fire firestore库使用-ii- sekrab车库
- 角燃箱炸弹群。以前,我们创建了自己的firestore获取器,以返回适当的观察力,从而将诺言转变为可感冒的诺言。今天,让我们继续使用其他命令,以正确地映射我们的数据。
garage.sekrab.com
了解VUE ND VUE 3之间的差异
- 上一篇
- 了解VUE ND VUE 3之间的差异
- 下一篇
- Golang环境深处潜水:从零到英雄
-
- 文章 · 前端 | 10秒前 | CSS表格样式
- CSS表格虚线边框怎么设置
- 408浏览 收藏
-
- 文章 · 前端 | 3分钟前 | html
- IDEA运行HTML的详细步骤教程
- 237浏览 收藏
-
- 文章 · 前端 | 3分钟前 |
- CSSGrid固定列宽技巧分享
- 364浏览 收藏
-
- 文章 · 前端 | 7分钟前 |
- CSS伪元素使用教程:::before和::after实战详解
- 378浏览 收藏
-
- 文章 · 前端 | 8分钟前 |
- JavaScript模块加载与规范详解
- 406浏览 收藏
-
- 文章 · 前端 | 11分钟前 |
- 弹性盒子和浮动布局有什么不同
- 135浏览 收藏
-
- 文章 · 前端 | 12分钟前 |
- 动态容器高度平滑过渡技巧分享
- 171浏览 收藏
-
- 文章 · 前端 | 12分钟前 |
- CSSGrid固定与自适应列宽技巧
- 471浏览 收藏
-
- 文章 · 前端 | 15分钟前 |
- CSP安全集成Stripe方法详解
- 340浏览 收藏
-
- 文章 · 前端 | 23分钟前 |
- CSSfloat按钮组排列技巧
- 167浏览 收藏
-
- 文章 · 前端 | 27分钟前 |
- CSS高度动画跳跃?用transition-height和max-height解决
- 472浏览 收藏
-
- 文章 · 前端 | 31分钟前 |
- optgroup标签使用方法及示例详解
- 287浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3229次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3442次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3472次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4583次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3851次使用
-
- JavaScript函数定义及示例详解
- 2025-05-11 502浏览
-
- 优化用户界面体验的秘密武器:CSS开发项目经验大揭秘
- 2023-11-03 501浏览
-
- 使用微信小程序实现图片轮播特效
- 2023-11-21 501浏览
-
- 解析sessionStorage的存储能力与限制
- 2024-01-11 501浏览
-
- 探索冒泡活动对于团队合作的推动力
- 2024-01-13 501浏览

