热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MongoDB和MongoMapper的基本使用方法

MongoDB是基于文档引擎的数据库,这和传统的关系型数据库,例如MySQL很不一样,文档引擎的数据库没有schema的概念。本篇将主要通过创建一个Rails的应用来介绍使用MongoDB和MongoMappergem(译者晓夜注:Rails的插件用来驱动Mongo让Rails更容易使用Mongo)

MongoDB是基于文档引擎的数据库,这和传统的关系型数据库,例如MySQL很不一样,文档引擎的数据库没有schema的概念。本篇将主要通过创建一个Rails的应用来介绍使用MongoDB和MongoMapper gem(译者晓夜注:Rails的插件用来驱动Mongo让Rails更容易使用Mongo)。实际上,很多的Rails程序员都是通过John Nunemaker的一篇博文RailsTips blog开始关注Mongo的。这是一篇相当精彩值得一读的文章,阐述了MongoMapper和MongoDB的七个主要功能并和传统的数据库做了对比,推荐阅读。
值得一提的是博文中的一个功能,也是前文提到的MongoDB没有schema的概念(译者晓夜注:Rails中使用rake db:migrate来控制数据库的结构一致和升级等。原文所说schema-less,是说没有一个schema_info的表来维护用来migrate的数据库版本。)。这样的数据库设计的有点在于,从数据库本身层面解决了schema的问题,不用在Rails中使用migrate操作了。因为在文档引擎的数据库中,每一行都是一个独立的文档类型,可以有自己的一组不同于其他记录的属性。这样的设计在分布式数据库中也会有独到的好处。
安装MongoDB 和 MongoMapper 在创建MongoDB的应用之前,要先安装。MongoDB的不同操作系统版都可以从MongoDB的官网下载界面得到。如果你使用的是苹果的系统,Chris Kampmeier的这篇文章 ,对于安装和配置MongoDB会很有价值,并且附带一份方便的plist文件用来帮助创建LaunchDemon,这样MongoDB就可以开机启动了。值得注意的是该文针对的MongoDB的版本,当前版本为1.2.0 。通过访问http://localhost:28017/可以验证是否成功安装和配置MongoDB。
Testing that MongoDB is running. 使用MongoDB创建Rails应用 在MongoDB正确安装和运行后,我们将创建一个叫做todo的Rails应用来演示如何使用MongoDB:
rails todo
鉴于我们将要使用MongoMapper来驱动MongoDB到Rails。我们需要在/config/environment.rb的配置文件中添加如下:
/config/environment.rb
config.gem "mongo_mapper"
在Rails的initializer文件中MongoDB需要一些额外的配置。在/config/initializers文件目录下创建mongo_config.rb文件,并在该文件中添加如下语句,用来配置MongoMapper将使用的数据库名。
/config/initializers/mongo_config.rb
MongoMapper.database = "todo-#{Rails.env}"
由上可知,通过指定Rails运行环境参数,我们可以在不同的运行环境下创建互不干扰的数据库。然而,如果我们希望移植当前应用到生产环境,我们还需要进行包括验证在内的其他工作,当然,对于我们当前的演示现在的配置已经足够了。然后,运行如下语句,保证MongoMapper的gem已经安装:
sudo rake gems:install
开发相关应用 演示的应用是todo list,实现类似备忘或者待完成任务列表的功能。项目中有一个Project的model和一个Tasks的model,他们有has_many的关系,为了简化开发,突出mongoDB的使用,我们将使用Ryan Bates的Nifty Generators插件来实现。当然,不使用这个插件,我们的项目应用完全可以正常演示。
首先,我们需要通过如下语句创建项目的layout。
script/generate nifty_layout
然后,我们将通过generate和nifty的scaffold创建project的model,这个表只有一个字段叫name,而且,使用mongoDB(译者晓夜注释:没有sechma的概念,也就是不需要一个migrate来控制版本),我们需要这个加上参数--skip-migration来创建如下:
script/generate nifty_scaffold project name:string --skip-migration
当然,上面的脚手架会创建modelcontroller和view,然而,默认创建的是ActiveRecord的基于schema的model,如下:
/app/models/project.rb
class Project < ActiveRecord::Base
  attr_accessible :name
end
以上代码由脚手架生成。
那么,我们需要把ActiveRecord的model,改成MongoMapper的类型,也就是把继承关系从ActiveRecord::Base变成MongoMapper::Document。我们使用key这个方法标明该MongoMapper的字段属性。我们的属性是name,再加上这个字段的类型String,那么定义如下:
/app/models/project.rb
class Project
  include MongoMapper::Document
  key :name, String
end
通过以上的修改,我们就已经拥有了所有添加,更新,删除和列表的操作,这就和我们之前用脚手架创建,用ActiveRecord加上关系型数据库一样,只是我们使用的是MongoMapper和MongoDB。
Using our application to create a new project. 也就是说对于添加,删除,列表和更新的操作,从Rails开发语句来看MongoMapper和ActiveRecord是完全相同的。甚至,MongoMapper还是支持ActiveRecord的验证方式如下:
/app/models/project.rb
validates_presence_of :name
在我们的例子中,不允许name字段为空的需要,在MongoMapper里也可以方便的如下语法表述:
/app/models/project.rb
class Project
  include MongoMapper::Document
  key :name, String, :required => true
end
添加更多的属性 由于MongoDB有schema-less(数据版本记录),我们可以非常容易的添加和更改model的属性,而不需要执行任何migrations的操作。比如,我们需要添加一个priority的属性,我们仅仅需要的是修改Project model如下:
/app/models/project.rb
class Project
  include MongoMapper::Document
  key :name, String, :required => true
  key :priority, Integer
end
如同ActiveRecord支持的一样,model的属性可以直接和form元素中关联。例如,新定义的priority可以显示在下拉菜单中如下:
/app/views/projects/_form.html.erb
<% form_for @project do |f| %>
  <%= f.error_messages %>
  


    <%= f.label :name %>
    <%= f.text_field :name %>
  


  


    <%= f.label :priority %>
    <%= f.select :priority, [1,2,3,4,5] %>
  


  

<%= f.submit "Submit" %>


<% end %>
在显示页面我们也需要做相应的修改以便显示该字段.
/app/views/projects/show.html.erb
<% title "Project" %>


  Name:
  <%=h @project.name %>



  Priority:
  <%=h @project.priority %>



  <%= link_to "Edit", edit_project_path(@project) %> |
  <%= link_to "Destroy", @project, :confirm => 'Are you sure?', :method => :delete %> |
  <%= link_to "View All", projects_path %>


这样,我们在创建新Project的时候,我们将会看到一个提示选择priority的下拉菜单,并且,提交创建以后,会得到一个包括priority显示的Project信息显示页面。
Our new project has a priority. 当我们创建了一个项目,在我们添加priority属性之前,或许你想要知道项目的priority的初始值是什么。我们可以通过访问看到是空白,也就是说,如果一条记录没有属性,那么对于MongoDB的文件类型存储引擎会默认是nil。
The first project has a blank priority value. 处理表之间的关联 在我们前文所述的Todo项目需求描述中,我们还需要定义一个Task model,每个Project会对应多个Task。同样,我们将和创建Project一样,使用脚手架创建这个model。值得注意的是,project_id之前在ActiveRecord都是integer类型,在这里我们使用字符串类型的如下(晓夜:原因见下文):
script/generate nifty_scaffold task project_id:string name:string completed:boolean --skip-migration
如同,修改Project model一样,我们同样修改Task model指定继承MongoMapper如下:
/app/models/Task.rb
class Task
  include MongoMapper::Document
  key :project_id, ObjectId
  key :name, String
  key :completed, Boolean
  belongs_to :project
end
再一次如同Project model,我们将按照MongoMapper的语法修改字段定义。可能我们已经习惯吧poject_id定义成整型,然而,对于MongoDB这里稍微有点区别,我们需要ObjectId类型来存储所有id。
至于,处理不同表之前的关联,我们可以像ActiveRecord一样定义belongs_to,当然,稍微有点不同,在Project中我们需要定义has_many :tasks,在MongoMapper中需要用many代替如下:
/app/models/project.rb
class Project
  include MongoMapper::Document
  key :name, String, :required => true
  key :priority, Integer
  many :tasks
end
现在,我们就可以使用我们用脚手架创建的model controller和view来创建Task了,比较有技巧的一点是,我们开始的时候定义了project_id的属性是String所以,在这里的view界面中就会创建一个对应的文本输入框。然后,我们需要修改这个form表单的,以便我们可以使用下拉菜单选择对应的所属Project。对于,下拉菜单显示project列表,我们完全可以参照ActiveRecord的方式,使用collection_select来实现,如下:
/app/views/tasks/_form.html.erb
<% form_for @task do |f| %>
  <%= f.error_messages %>
  


    <%= f.label :project_id %>
    <%= f.collection_select :project_id, Project.all, :id, :name %>
  


  
也就是,我们可以通过如下方式创建Task并选择所属Project。
Using collection_select to create a select menu as we would with ActiveRecord. 当我们完成创建task后会默认跳到显示task的界面,我们会看到刚刚创建的所属project的id,当然我们更愿意看到Project的name。所以,需要用@task.project.name来代替@task.project_id如下:
/app/views/tasks/show.html.erb
<% title "Task" %>


  Project:
  <%=h @task.project.name %>



这样,我们就可以使用ActiveRecord的模式来显示相关的model的信息了。(晓夜:也就是belong_to has_many的功能)


推荐阅读
  • Hadoop入门与核心组件详解
    本文详细介绍了Hadoop的基础知识及其核心组件,包括HDFS、MapReduce和YARN。通过本文,读者可以全面了解Hadoop的生态系统及应用场景。 ... [详细]
  • Docker的安全基准
    nsitionalENhttp:www.w3.orgTRxhtml1DTDxhtml1-transitional.dtd ... [详细]
  • 探讨如何真正掌握Java EE,包括所需技能、工具和实践经验。资深软件教学总监李刚分享了对毕业生简历中常见问题的看法,并提供了详尽的标准。 ... [详细]
  • 程序员妻子吐槽:丈夫北漂8年终薪3万,存款情况令人意外
    一位程序员的妻子在网上分享了她丈夫在北京工作八年的经历,月薪仅3万元,存款情况却出乎意料。本文探讨了高学历人才在大城市的职场现状及生活压力。 ... [详细]
  • MySQL缓存机制深度解析
    本文详细探讨了MySQL的缓存机制,包括主从复制、读写分离以及缓存同步策略等内容。通过理解这些概念和技术,读者可以更好地优化数据库性能。 ... [详细]
  • 本文探讨了如何在日常工作中通过优化效率和深入研究核心技术,将技术和知识转化为实际收益。文章结合个人经验,分享了提高工作效率、掌握高价值技能以及选择合适工作环境的方法,帮助读者更好地实现技术变现。 ... [详细]
  • 本文探讨了2019年前端技术的发展趋势,包括工具化、配置化和泛前端化等方面,并提供了详细的学习路线和职业规划建议。 ... [详细]
  • Spring Cloud因其强大的功能和灵活性,被誉为开发分布式系统的‘一站式’解决方案。它不仅简化了分布式系统中的常见模式实现,还被广泛应用于企业级生产环境中。本书内容详实,覆盖了从微服务基础到Spring Cloud的高级应用,适合各层次的开发者。 ... [详细]
  • 利用 Jest 和 Supertest 实现接口测试的全面指南
    本文深入探讨了如何使用 Jest 和 Supertest 进行接口测试,通过实际案例详细解析了测试环境的搭建、测试用例的编写以及异步测试的处理方法。 ... [详细]
  • 在 Windows 10 中,F1 至 F12 键默认设置为快捷功能键。本文将介绍几种有效方法来禁用这些快捷键,并恢复其标准功能键的作用。请注意,部分笔记本电脑的快捷键可能无法完全关闭。 ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 计算机网络复习:第五章 网络层控制平面
    本文探讨了网络层的控制平面,包括转发和路由选择的基本原理。转发在数据平面上实现,通过配置路由器中的转发表完成;而路由选择则在控制平面上进行,涉及路由器中路由表的配置与更新。此外,文章还介绍了ICMP协议、两种控制平面的实现方法、路由选择算法及其分类等内容。 ... [详细]
  • This guide provides a comprehensive step-by-step approach to successfully installing the MongoDB PHP driver on XAMPP for macOS, ensuring a smooth and efficient setup process. ... [详细]
  • 深入理解Java中的volatile、内存屏障与CPU指令
    本文详细探讨了Java中volatile关键字的作用机制,以及其与内存屏障和CPU指令之间的关系。通过具体示例和专业解析,帮助读者更好地理解多线程编程中的同步问题。 ... [详细]
  • Mongoose 5.12.10 发布:MongoDB 异步对象模型工具的新特性与修复
    Mongoose 是一款专为异步环境设计的 MongoDB 对象模型工具,支持 Promise 和回调函数。最新版本 Mongoose 5.12.10 带来了多项修复和改进,包括查询选项中的默认值设置、嵌入式判别器填充、以及 TypeScript 定义文件的优化。 ... [详细]
author-avatar
谢海武181_160
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有