您现在的位置是:网站首页> 编程资料编程资料

rails常用数据库查询操作、方法浅析_ruby专题_

2023-05-26 365人已围观

简介 rails常用数据库查询操作、方法浅析_ruby专题_

1、获取数据

获取第一条、最后一条记录

复制代码 代码如下:

Model.first
Model.first(options)
Model.find(:first, options)
Model.last
Model.last(options)
Model.find(:last, options)

通过id获取记录

复制代码 代码如下:

Model.find(1, 10, options)
Model.find([1, 10], options)
.find all

复制代码 代码如下:

Model.all(options)

对一组数据进行相同操作

复制代码 代码如下:

User.all.each do |user|
    NewsLetter.weekly_deliver(user)
end

如果表记录数比较大,这种方式比较耗资源,因为它会一次载入整个表的数据。改用以下这种方式,它每次只载入1000行,然后逐步yield完整个表

复制代码 代码如下:

User.find_each do |user|
    NewsLetter.weekly_deliver(user)
end

自定义方式,find_each接受和find同样的options

复制代码 代码如下:

User.find_each(:batch_size => 5000, :start => 2000) do |user|
    NewsLetter.weekly_deliver(user)
end

find_in_batches,和find_each相似,但它yield时传递的是model对象数组,而不是单个model对象

复制代码 代码如下:

Invoice.find_in_batches(:include => :invoice_lines) do |invoices|
    export.add_invoices(invoices)
end

2、查询条件

通过替换?来传递条件值,可避免SQL注入

复制代码 代码如下:

Client.first(:conditions => ["orders_count = ?", params[:orders])

symbol占位条件

复制代码 代码如下:

Client.all(:conditions => ["created_at >= :start_date AND created_at <= :end_date", {:start_date => params[:start_date], :end_date => params[:end_date] }])

范围条件 in(集合)

复制代码 代码如下:

Client.all(:conditions => ["created_at IN (?)", (params[:start_date].to_date)..(params[:end_date].to_date])

生成sql

复制代码 代码如下:

SELECT * FROM users WHERE (created_at IN ('2007-12-31','2008-01-01','2008-01-02','2008-01-03','2008-01-04','2008-01-05', '2008-01-06','2008-01-07','2008-01-08'))

如果要生成日期时间,再加上.to_time
复制代码 代码如下:
params[:start_date].to_date.to_time,生成2007-12-01 00:00:00格式

有上数据库会在以上条件中报错,如Mysql会报查询语句过长的错误,此时可以改成created_at > ? AND created_at < ?的形式

Hash条件

复制代码 代码如下:

Client.all(:conditions => {:locked => true })

带范围条件

复制代码 代码如下:

Client.all(:conditons => {:created => (Time.now.midnight - 1.day)..Time.now.midnight})

生成sql

复制代码 代码如下:

SELECT * FROM clients WHERE (clients.created_at BETWEEN '2008-12-21 00:00:00' AND '2008-12-22 00:00:00')

集合条件

复制代码 代码如下:

Client.all(:conditons => {:orders_count => [1,3,5])

生成sql

复制代码 代码如下:

SELECT * FROM clients WHERE (clients.orders_count IN (1,3,5))

3、查询选项

排序

复制代码 代码如下:

#单个排序
Client.all(:order => "created_at ASC")
#多个排序
Client.all(:order => "orders_count ASC, created_at DESC")

返回指定字段

复制代码 代码如下:

Client.all(:select => "viewable_by, locked")
#使用函数
Client.all(:select => "DISTINCT(name)")

限定和偏移Limit and Offset

复制代码 代码如下:

Client.all(:limit => 5)
#生成
SELECT * FROM clients LIMIT 5
Client.all(:limit => 5, :offset => 5)
#生成
SELECT * FROM clients LIMIT 5, 5

Group分组

复制代码 代码如下:

Order.all(:group => "date(created_at)", :order => "created_at")

生成sql
复制代码 代码如下:
SELECT * FROM orders GROUP BY date(created_at)

Having
复制代码 代码如下:

Order.all(:group => "date(created_at)", :having => ["created_at > ?", 1.month.ago)

生成sql
复制代码 代码如下:
SELECT * FROM orders GROUP BY date(created_at) HAVING created_at > '2009-01-15'

只读

复制代码 代码如下:

client = Client.first(:readonly => true)
client.locked = false
client.save
#对只读对象进行保存将会触发ActiveRecord::ReadOnlyRecord异常

更新时锁定记录

乐观锁Optimistic Locking

为使用乐观锁,须在表里建一个lock_version的字段,每次更新记录时,ActiveRecord自动递增lock_version的值,

复制代码 代码如下:

c1 = Client.find(1) c2 = Client.find(1) c1.name = "Michael" c1.save c2.name = "should fail" c2.save # Raises a ActiveRecord::StaleObjectError

备注:You must ensure that your database schema defaults the lock_version column to 0.

This behavior can be turned off by setting ActiveRecord::Base.lock_optimistically = false.

指定乐观锁字段名

复制代码 代码如下:

class Client < ActiveRecord::Base set_locking_column :lock_client_column end

悲观锁Pessimistic Locking
悲观锁定由数据库直接提供

复制代码 代码如下:

Item.transaction do
    i = Item.first(:lock => true)
    i.name = 'Jones'
    i.save
end

Mysql执行返回
复制代码 代码如下:
SQL (0.2ms) BEGIN Item Load (0.3ms) SELECT * FROM `items` LIMIT 1 FOR UPDATE Item Update (0.4ms) UPDATE `items` SET `updated_at` = '2009-02-07 18:05:56', `name` = 'Jones' WHERE `id` = 1 SQL (0.8ms) COMMIT

为特定数据库加入原始的lock声明
为Mysql的锁定声明为共享模式,即锁定时仍然可读

复制代码 代码如下:
Item.transaction do  i = Item.find(1, :lock => "LOCK IN SHARE MODE")  i.increment!(:views) end

4、关联表

复制代码 代码如下:

Client.all(:joins => "LEFT OUTER JOIN address ON addresses.client_id = clients.id')

生成sql
复制代码 代码如下:
SELECT clients.* FROM clients LEFT OUTER JOIN addresses ON addresses.client_id = clients.id

使用Array、Hash、Named Associations关联表
有如下model

复制代码 代码如下:

class Category < ActiveRecord::Base
    has_many :posts

-六神源码网