首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

RavenDB 文档建模--使用 RavenDB 作为键/值存储

RavenDB 非常适合键/值存储,为了确保快速存取数据库,RavenDB 在设计的时候降低了存储和加载文档的成本,这是 RavenDB 和其他数据库相比最大的有点。 由于数据限制必须是 JSON ,因此使用 RavenDB 作为键/值存储是完全没问题的。使用 RavenDB 缓存信息的常见场景有:存储购物车信息、存储用户会话数据、缓存热点数据等等。在默认情况下,RavenDB 不会对存储以及加载文档增加额的外成本,因此可以使用所有访问模型中最简单的快速数据库。一般来说键/值建模的复杂性在于生成适当的键以及可以对其执行哪些操作。在使用 RavenDB 作为键/值存储的情况下,下面所列的内容是很有用的:

02
  • 您找到你想要的搜索结果了吗?
    是的
    没有找到

    RavenDB文档建模--琐碎的注意事项--缓存

    RavenDB 使用基于 HTTP 的 REST 用于客户端和服务端的通信,也就是说我们在操作文档的时候其实就是使用 WEB 发送 HTTP 请求,那么基于这一点 RavenDB 就可以利用 HTTP 的特性来执行一些东西。 其中最常见的是 RavenDB 客户端 API 使用 HTTP 特性在客户端开启缓存。每个从服务端返回的响应都包含一个 etag 头内容,如果我们只是请求的单个文档,那么这个 etag 头内容就是文档的 etag 标题,如果我们请求的是多个文档的话,这个 etag 头内容就会包含一个计算值(具体计算值将在后面的专题详细讲解)。客户端将会缓存服务器的响应、URL 和 etag 的值,那么当有和缓存 URL 想的请求进入客户端时,我们会将其发送到服务端,同时也告知服务端,客户端存在一个特定 etag 值的请求结果。服务端在收到信息后会检查 etag 和客户端上的 etag 是否一样,如果一样就不返回数据,让客户端继续使用缓存的数据,这样就减少了网络的负载和服务端的压力。 另外,RavenDB 还有一个叫做 Aggressive Caching 的功能,它可以让看客户端 API 注册来自服务端的更改。也就是说,当我们在本地缓存了一些值后,就不需要再向服务端发送请求,让服务端判断是否要给我们返回新数据,通过这个功能如果服务端的数据发生了改变,那么服务端就会通知客户端,这时我们可以去请求服务端来获取新的数据。这个功能对于查询类似 configure 文档或大型文档来说可以大大的节省性能。

    02

    RavenDB 文档建模--建模注意事项

    我们在开始讲解如何在 RavenDB 中建模之前,先来看看注意事项,这些内容与我们将要辨析的模型有着直接的关系。 这里需要注意的第一点是 不要在不同应用之间建立共享数据库。很多设计者会建立共享数据库,用以在不同的应用之间共享相同的数据,虽然这样做能减少数据存储量,以及实现多应用使用相同数据的目的,但是在 RavenDB 中并不推崇这样的做法。这是因为虽然不同的应用看起来有些数据是一样的,我们会强制它们使用相同的方式处理数据,但是在大多数情况下不同的应用程序使用相互不同的方式处理类似的数据,如果使用共享数据的话,一个应用程序共享数据的结构的改变就会造成其他应用跟着一起改变,进而导致数据模型复杂性增加,并且也会增加不同应用开发团队之间沟通的成本和时间。因此每个应用程序应该对立的进行数据建模,并不断的根据需求进行改进。 读到到这里,肯定有人会问了:不同的应用程序直接或多或少的都需要共享数据,那么使用 RavenDB 如何实现这一点呢?我们可以使用 RavenDB 内置的 ETL 功能在不同应用程序服务器之间建立数据/信息流(这个内容将会在后续讲解)。 另一个要注意的是 某些情况下应该数据冗余存储,比如在 Order 文档中存在 Address 文档的链接,但是如果 Address 中的配送地址变了,那么 Order 文档中的历史订单的配送地址也会跟着改变,这样就出现了我上一篇文章说的数据损坏。那么,我们在进行建模的时候,应该考虑我的关注点是当前值(例如 Order 文档中的当前订单配送地址)还是时间点值(例如 Order 文档的历史订单配送地址),如果是时间点值那么我们就需要进行数据冗余存储,例如在 Order 文档中存储配送地址的详细信息。 以上几小段的内容总结下来就是建模文档的核心原则:

    02

    RavenDB文档建模--琐碎的注意事项--缓存查询属性

    缓存查询属性是我们在实际开发中会遇到的,什么是缓存查询属性呢?举个例子来说,在电子商城的订单系统中每个账户都有自己的订单数据,有时用户需要查看自己截止到目前所订单的数量,那么这个账户的订单数量可以被视为 查询属性,因为从众多的订单中统计出某个账户的订单数量是一件会消耗很多资源的命令,因此会将这个订单数量存储在缓存中(例如存储在RavenDB中),在后续查询中我们不需要再次从数据库中查询,只需要在缓存冲查询即可,这就叫做 缓存查询属性。 缓存查询属性的行为开起来很常见也很有意义,但是着是一个不良的行为。为什么这么说呢?首先在大部分领域中这种类似的属性并不是客户必须有的部分(可有可无),也不是客户文档必须包含的部分,其次,为了保证这个属性会在相关内容变更(例如订单删除和新增)时也跟着更改,我们就需要在相关内容发生变化时也去改变它的内容,等于说我们要对数据库多进行N次的操作,然后将更新的数据在存入缓存中,这样就会增大失败的概率,接着,我在进行开发设计前还需要考虑哪些操作会改变查询属性,如果是比较简单的项目还好,那如果是大型项目呢?里面的操作错综复杂,如何保证覆盖所有的操作? 缓存查询属性这个问题其实是一个业务和成本方面的问题,在大多数情况下我们只是想在页面中展示这个值,并且要从关系型数据库中查询出这个值的话可能会很昂贵,因此很多人会将这个值直接放在缓存中。在 RavenDB 中我们可以使用 MapReduce 聚合操作来处理,我们根本就不需要缓存这种属性,也减少了成本,MapReduce的使用因为是一个很大的模块,因此我将放在后面专门开始一个专题来讲解。在解决完缓存查询属性的问题后,下一步我们该考虑如何处理并发的问题和并发问题对建模的影响,这个问题我将放在下一篇文章讲解。

    02

    RavenDB文档建模--琐碎的注意事项--文档的引用处理、包含以及加载

    这篇文章比较简单,在这个专题的一开始,我们探究了对象和文档之间的关系,我们只是专注于构建模型,忽略了跳过我们如何在图表阶段之外处理关系。那么这一小篇文章我们就来简单的说一下这个问题。 我们需要考虑两个单独的操作。在查询和加载文档期间获取相关信息可以使用Include调用来完成,这时一个非常常用的功能,因为他可以减少请求服务端的次数。第二个操作是查询,也就是说当想根据相关文档的属性查询特定文档。例如前面文章所说的幼儿园的例子,查询母亲叫刘妈妈的孩子,由于子文档不再包含父级文档的名称,那么我们将如何搜索它呢?RavenDB 不允许我们使用多连接,但它允许在索引阶段为相关数据编制索引,然后对其进行查询。因此使用这个功能通过母亲的名字查询孩子非常容易。索引功能将在索引专题中进行进一步讲解。我在这里提到它,是因为知道它的存在会影响我们对数据建模的方式,在决定如何对相关数据进行建模时,它可以有很大的帮助。但是最终决策几乎总是归结为我们是想要数据的时间点视图还是当前值。对于第一个选项,我们通常会将值从源复制到其自己的文档中,对于第二个选项,我们可以在索引和查询以及从服务器获取数据时使用。

    05

    扫码

    添加站长 进交流群

    领取专属 10元无门槛券

    手把手带您无忧上云

    扫码加入开发者社群

    相关资讯

    热门标签

    活动推荐

      运营活动

      活动名称
      广告关闭
      领券