我有一个人名和他们的生日数据库。生日的格式为mm/dd/yyyy
,如“3/13/1960”。
我想提取一份在特定日期之后出生的人员名单。我称这个日期为“基地”。
您在下面看到的程序,首先创建一个人员数据库(模拟我想要使用的真实数据库),然后提取所需的列表。问题是结果不像我预期的那样:
import datetime as dt
import peewee as pw
db = pw.SqliteDatabase('people1.db')
class Person(pw.Model):
name = pw.CharField()
birthday = pw.DateField(formats=['%m/%d/%Y'])
class Meta:
database = db # This model uses the "people.db" database.
db.create_tables([Person])
bob0 = Person(name='Bob0', birthday='4/13/1940')
bob1 = Person(name='Bob1', birthday='5/13/1950')
bob2 = Person(name='Bob2', birthday='3/13/1960')
bob3 = Person(name='Bob3', birthday='3/13/1970')
bob4 = Person(name='Bob4', birthday='3/13/1980')
bob5 = Person(name='Bob5', birthday='3/13/1990')
base = Person(name="base", birthday='3/13/1960')
bob0.save()
bob1.save()
bob2.save()
bob3.save()
bob4.save()
bob5.save()
base.save()
for item in Person.select().where(Person.birthday > base.birthday):
print item.name , item.birthday
输出:
>>> ================================ RESTART ================================
>>>
Bob0 1940-04-13
Bob1 1950-05-13
Bob3 1970-03-13
Bob4 1980-03-13
Bob5 1990-03-13
>>>
如上所示,base = 3/13/1960
。所以我不应该在输出中有 Bob0 和 Bob1 !我怎么处理呢?
请注意,我不想更改数据库中生日的格式。我也不想获取所有行并在以后检查它们!我只想获取所需的行。
答案 0 :(得分:3)
SQlite将日期时间存储为字符串。正如其他人在评论和其他answers中建议的那样,您应该使用不同的格式来存储日期,以便"日期排序和词汇排序可以解决相同的问题":
import datetime as dt
import peewee as pw
db = pw.SqliteDatabase('people1.db')
class Person(pw.Model):
name = pw.CharField()
birthday = pw.DateField(formats=['%Y-%m-%d'])
class Meta:
database = db # This model uses the "people.db" database.
db.create_tables([Person])
Person.create(name='Bob0', birthday=dt.date(1940, 4, 13))
Person.create(name='Bob1', birthday=dt.date(1950, 5, 13))
Person.create(name='Bob2', birthday=dt.date(1960, 3, 13))
Person.create(name='Bob3', birthday=dt.date(1970, 3, 13))
Person.create(name='Bob4', birthday=dt.date(1980, 3, 13))
Person.create(name='Bob5', birthday=dt.date(1990, 3, 13))
base = Person.create(name="base", birthday=dt.date(1960, 3, 13))
for item in Person.select().where(Person.birthday > base.birthday):
print item.name , item.birthday
这给出了:
Bob3 1970-03-13
Bob4 1980-03-13
Bob5 1990-03-13
<强>更新强>
我没有注意到您的评论,您不想更改数据库。
这是一种提取部分日期的疯狂方法:
SELECT
birthday,
CAST(substr(birthday, 1, instr(birthday, '/') - 1) AS integer),
CAST(substr(substr(birthday, instr(birthday, '/') + 1), 1, instr(substr(birthday, instr(birthday, '/') + 1), '/') - 1) AS integer),
CAST(substr(birthday, instr(birthday, '/') + instr(substr(birthday, instr(birthday, '/') + 1), '/') + 1) AS integer)
FROM person
在我的测试数据中给出:
4/13/1940 4 13 1940
12/13/1950 12 13 1950
3/3/1960 3 3 1960
3/25/1970 3 25 1970
3/13/1980 3 13 1980
3/13/1990 3 13 1990
3/13/1960 3 13 1960
您可以使用这些表达式将它们与给定日期的部分进行比较:
query = """
SELECT *
FROM person
WHERE
(
substr('0000' || CAST(substr(birthday, instr(birthday, '/') + instr(substr(birthday, instr(birthday, '/') + 1), '/') + 1) AS integer), -4, 4) || '-' || -- year
substr('00' || CAST(substr(birthday, 1, instr(birthday, '/') - 1) AS integer), -2, 2) || '-' || -- month
substr('00' || CAST(substr(substr(birthday, instr(birthday, '/') + 1), 1, instr(substr(birthday, instr(birthday, '/') + 1), '/') - 1) AS integer), -2, 2) -- day
) > '1960-03-03'
"""
for item in Person.raw(query):
print item.name, item.birthday
我在这里重建ISO日期并将其用于比较。
答案 1 :(得分:2)
您可以使用sqlite3.Connection.create_function指定自己的sqlite函数,将日期转换为可按字典顺序排序的日期:
import datetime as dt
import peewee as pw
# custom sqlite function to reformat our date string
def _sqlite_reformat_date(unfortunate_date_string):
return dt.datetime \
.strptime(unfortunate_date_string,'%m/%d/%Y') \
.strftime('%Y-%m-%d')
# Subclass pw.SqliteDatabase to add our custom sqlite function
class MySqliteDatabase(pw.SqliteDatabase):
def __init__(self, *args, **kwargs):
super(MySqliteDatabase, self).__init__(*args, **kwargs)
def _add_conn_hooks(self, conn):
conn.create_function('reformat_date', 1, _sqlite_reformat_date)
super(MySqliteDatabase, self)._add_conn_hooks(conn)
db = MySqliteDatabase('people1.db')
# ...
# Your model definition and data inserts from your example above
# ...
rd = pw.fn.reformat_date # Use our custom sqlite function
for item in Person.select().where(rd(Person.birthday) > rd(base.birthday)):
print item.name , item.birthday
虽然这种方法只能“获取所需的行”,但仍会为每一行运行此python函数!它比在python中进行日期比较时获取所有行要好一些,它甚至可能更慢!
然而,_sqlite_reformat_date
函数可以重构得更快,并且很高兴知道将自定义函数添加到sqlite中是多么容易。
答案 2 :(得分:0)
SQL函数的一些链接看起来很有趣但可行,可能更快。
from datetime import datetime
import peewee as pw
db = pw.SqliteDatabase('people1.db')
class Person(pw.Model):
name = pw.CharField()
birthday = pw.DateField(formats=['%m/%d/%Y'])
class Meta:
database = db # This model uses the "people.db" database.
db.create_tables([Person])
bob0 = Person(name='Bob0', birthday='4/13/1940')
bob1 = Person(name='Bob1', birthday='5/13/1950')
bob2 = Person(name='Bob2', birthday='3/13/1960')
bob3 = Person(name='Bob3', birthday='3/13/1970')
bob4 = Person(name='Bob4', birthday='3/13/1980')
bob5 = Person(name='Bob5', birthday='3/13/1990')
bob6 = Person(name='Bob6', birthday='12/1/1990')
base = Person(name="base", birthday='3/13/1960')
bob0.save()
bob1.save()
bob2.save()
bob3.save()
bob4.save()
bob5.save()
bob6.save()
base.save()
month = 'substr(birthday,1,instr(birthday,"/")-1)'
iso_month = 'case when length({month}) = 1 then "0" || {month} else {month} end'.format(month=month)
day = 'trim(trim(birthday,"0123456789"),"/")'
iso_day = 'case when length({day}) = 1 then "0" || {day} else {day} end'.format(day=day)
year = 'substr(ltrim(ltrim(birthday,"0123456789"),"/"),instr(ltrim(ltrim(birthday,"0123456789"),"/"),"/")+1)'
iso_date = 'replace(replace(replace("yyyy-mm-dd","yyyy",{year}),"mm",{iso_month}),"dd",{iso_day})'.format(year=year,iso_month=iso_month,iso_day=iso_day)
iso_base = datetime.strptime(base.birthday,'%m/%d/%Y').date().isoformat()
if __name__ == '__main__':
for item in Person.select().where(pw.SQL(iso_date) > iso_base):
print item.name , item.birthday
#output
#Bob3 1970-03-13
#Bob4 1980-03-13
#Bob5 1990-03-13
#Bob6 1990-12-01
答案 3 :(得分:-1)
SQLite将日期存储为字符串。因此,它们应存储为YYYY-MM-DD,这样可以确保它们正确排序。老实说,没有理由不这样做。
如果您查看sqlite的文档,它甚至无法识别您正在使用的格式的日期:
https://www.sqlite.org/lang_datefunc.html
所以,我的建议是更新你存储日期的方式。
否则,创建一个用户定义的函数,使用strptime
执行正确的操作(假设使用playhouse.sqlite_ext.SqliteExtDatabase
):
@db.func()
def fix_time(s):
return datetime.datetime.strptime(s, '%m/%d/%Y').strftime('%Y-%m-%d')
如果你想坚持常规的旧SqliteDatabase
,你可以调用sqlite3方法connection.create_function
。