一、Should I use a query or a filter to seach records?我应该使用一个查询或者过滤器来搜索记录吗?
这取决于你想要搜索多少reocrds并且它们是否被加载到store中了。
1. Queries
(1) Queries对于搜索数百,书签,或者甚至百万的记录是有用的。你仅仅把搜索选项交给你的服务器,并且它负责把匹配的records列表交还给你。因为来自服务器的响应包含所有匹配的records的ID,这并不重要,如果store没有提前加载它们;它认为它们不在缓存中并且可以如果必要的话可以通过ID请求record。
(2) Queries的缺点是它们不是实时更新的,它们比较慢,并且它们要求你的服务器支持你希望执行的查询类型。
(3) 因为是服务器决定哪些records匹配器这个查询,而不是store,Queries不实时更新。如果你希望更新它们,你必须手动调用reload()方法并且等待服务器响应。如果你在客户端创建了一个条新的record,它将不会显示在结果中直到你存储这条心的record到服务器并且重新加载查询结果。
(4) 因为store必须与你的服务器协商来决定一个查询的结果,它需要一个网络请求。这对用户来说很慢,特别是如果他们是在一个很慢的连接上或者你的服务器响应很慢。当必须咨询服务器时,JS Web应用程序的典型速度可以提高迟钝的知觉。
(5) 最后,执行查询要求store和服务期间协作。默认的,Ember Data将会发送search options,你把它们作为HTTP请求体传递到你的服务器。如果你的服务器不支持这种格式的请求,你要么需要更改你的服务器,要么通过创建自定义的适配器来自定义查询如何被发送。
2. Filter
(1) 过滤器,在另一方面,是在store的缓存中执行一个关于所有records的实时搜索。一旦一个条新的record被加载进store,这个filter将会检查这条record是否匹配,如果匹配,就把它添加进搜索结果的数组中。如果这个数组被展现在一个模板中,模板会自动更新。
(2) 过滤器还考虑到新创建的并且没有被保存进store的records,并且records已经被修改但是没有被保存。如果你希望records显示在搜索结果中一旦它们在客户端被创建或者被修改,你应该使用一个过滤器。
(3) 切记如果store不知道它们,records不会显示在一个过滤器中。你可以通过使用store的push()方法确保一条record在store中。
(4) 还有一个限制,在你遇到性能问题之前你可以在内存中合理的保存和搜索多少records。
3. 最后,记得你可以结果Queries和Filter去利用各自的长处。记得records通过一个query返回到服务器并且缓存进store。你可以使用这个事实来执行一个filter,通过一个query,开始匹配record到store中,并且一个filter function匹配相同的records。
4. 这将卸载到服务器搜索所有可能的records,同时仍然创建一个实时更新列表,它包含了在客户端创建的和修改records。
app/routes/posts/favourited.js
export default Ember.Route.extend({ model() { var store = this.store; // Create a filter for all favorited posts that will be displayed in // the template. Any favorited posts that are already in the store // will be displayed immediately; // Kick off a query to the server for all posts that // the user has favorited. As results from the query are // returned from the server, they will also begin to appear. return store.filter('post', { favorited: true }, function(post) { return post.get('isFavorited'); }); } });
二、How do I inform Ember Data about new records created on the backend?我如何通知Ember Data后台创建的新记录?
1. 当你通过使用Ember Data的store.findRecord方法请求一个record时,Ember将会自动把数据加载进store。对于已经请求过的数据,这允许Ember避免下一次到后台的往返造成的延迟。另外,加载一条record进store将会更新任何RecordArray(例如store.filter或者store.findAll的结果),应该包含这条record。这意味着任何依据RecordArray
的绑定的数据或者计算属性将会自动被同步去包含这条新的或者更新的record的值。
2. 一些应用程序可能希望不通过store.findeRecord请求record添加或者更新record到store。为了完成这个要求,你可以使用DS.Store的push或者pushPayload方法。对于有一个通道(例如 SSE或者 Web Sockets)去通知它后台有新的或者更新的recordsweb的应用程序,这是游泳的。
3. 在Ember Data的store中,push是最简单的方法去加载或者更新records。当使用push的时候,在把record推送进store之前规范化(normalize)JSON对象很重要。
4. push一次只接受一条record。如果你想加载一个records数组到store你可以调用pushMany。
socket.on('message', function (message) { var modelName = message.model; store.push(modelName, store.normalize(modelName, message.data)); });
5. 作为v1.0.0-beta.14的push方法,它接受部分属性去更新存在的records。因此启用更新方法。更新部分属性是有用的,如果你的网络应用程序只接收到一个模型的改变属性的通知。
6. pushPayload对于store#push来说是一个方便的封装,它将反序列化payloads如果模型的序列化器实现一个pushPayload
方法。
注意这个方法和JSONSerializer
一起工作不会有效,因为它没有实现一个pushPayload
方法,这个非常重要。
socket.on('message', function (message) { store.pushPayload(message.model, message.data); });