SQLite时间处理
Posted wxc7009
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了SQLite时间处理相关的知识,希望对你有一定的参考价值。
sqlite数据库处理时间问题 和 日期时间函数
首先,sqlite数据库在时间处理上和sqlserver还有oracle不同,下面根据自己做过的实例总结一下. 创建了一个Log数据表:
LogID SourceID OperatorID LogType LogLevel LogTime LogContent
1 aaa.aspx 0 2 1 2011-08-18 16:44:32.000 aaaa
2 bbb.aspx 1 2 2 2011-08-18 16:38:32.000 bbbb
3 ccc.aspx 2 3 3 2011-09-02
cccc
4 ddd.aspx 3 1 4 2011-08-15
dddd
5 eee.aspx 4 1 3 2011-08-18
eee
普通的sqlserver的查询语句如下:select * from Log where
LogTime=\'2011-09-02\'可以查询出"\'2011-09-02"的数据。
而在sqlite数据库中,写上面的语句,查询不到任何数据;说明sqlserver于sqlite对于时间处理上是不同的。那么如何写一条以时间为查询条件的sql语句才能在sqlite数据库中把想要的结果查询出来呢?请看::::
select * from Log where datetime(LogTime)=datetime(\'2011-08-18 16:38:32.000\')
sqlite数据查询语句,必须对时间字段和传入的时间参数做转换.即加上datetime()转换. 执行了上面的语句,就可以得到要查找到结果。如果按照sqlserver的写法,那么是不会得到查询结果的。另外,一些错误的查询语句如下:
1: select * from Log where datetime(LogTime)=datetime(\'2011-08-18\'),这条语句只能查询到一条数据,就是对应LogID=5的那条,而不 会得到LogID=1和LogID=2的数据。可见,sqlite对于时间是非常严格的。精确度非常高。
2:select * from
Log where datetime(LogTime)=datetime(\'2011-8-18\'),这条语句是查询不到任何结果,因为sqlite的时间要求是 yyyy-MM-dd或者yyyy-MM-dd hh:mm:ss的。当月数为10以下,那么必须写成0x的形式(如:05),所以在做开发的时候一定要对入库的时间做相应处理。
3: select *
from Log where datetime(LogTime)=datetime(\'2011-08-9\'),同样的,这条语句也查询不到任何结果,应该把9改成09,就可以得到查询结果了。
SQLite 日期时间函数
SQLite并没有datatime字段类型,但是可以在字符串类型字段中存储时间,并提供了一些比较实用的日期时间操作函数
strftime(日期时间格式, 日期时间字符串, 修正符, 修正符, ……)
strftime( 日期时间格式, 日期时间字符串 ) 也就等价于AAuto中的:
time( 日期时间字符串,日期时间格式 ) ,sqlite与AAuto 使用的格式化语法也一样。
参考:http://www.aau.cn/doc/reference/libraries/kernel/time/time.html
strftime() 函数返回一个经过格式化的日期时间,
它可以用下面的符号对日期和时间进行格式化:
%d 一月中的第几天 01-31
%f 小数形式的秒,SS.SSSS
%H 小时 00-24
%j 一年中的第几天 01-366
%J Julian Day Numbers
%m 月份 01-12
%M 分钟 00-59
%s 从 1970-01-01日开始计算的秒数
%S 秒 00-59
%w 星期,0-6,0是星期天
%W 一年中的第几周 00-53
%Y 年份 0000-9999
%% % 百分号
date,time,datetime,julianday 函数
date(日期时间字符串, 修正符, 修正符, ……) 等价于 strftime(“%Y-%m-%d”,…)
time(日期时间字符串, 修正符, 修正符, ……) 等价于 strftime(“%H:%M:%S”,…)
datetime(日期时间字符串, 修正符, 修正符, ……) 等价于
strftime(“%Y-%m-%d %H:%M:%S”,…)
julianday(日期时间字符串, 修正符, 修正符, ……) 等价于
strftime(“%J”,…)
日期时间字符串
可以用以下几种格式:
格式有严格的要求
2008-06-15 03:35:28 日期只能用\'-\'分隔,时间只能用\':\' 分隔,不足二位数的必须补零
- YYYY-MM-DD
- YYYY-MM-DD HH:MM
- YYYY-MM-DD HH:MM:SS
- YYYY-MM-DD HH:MM:SS.SSS
- YYYY-MM-DDTHH:MM
- YYYY-MM-DDTHH:MM:SS
- YYYY-MM-DDTHH:MM:SS.SSS
- HH:MM
- HH:MM:SS
- HH:MM:SS.SSS
- now
- DDDD.DDDD
在第五种到第七种格式(ISO8601)中的“T”是一个分割日期和时间的字符;
第八种到第十种格式只代表2000-01-01日的时间,
第十一种格式的’now’表示返回一个当前的日期和时间,使用格林威治时间(UTC);
第十二种格式表示一个 Julian Day Numbers。
修正符: 日期和时间可以使用下面的修正符来更改日期或时间: - NNN days
- NNN hours
- NNN minutes
- NNN.NNNN seconds
- NNN months
- NNN years
- start of month
- start of year
- start of week
- start of day
- weekday N
- unixepoch
- localtime
- utc 前六个修正符就是简单的增加指定数值的时间和日期;第七到第十个修正符表示返回当前日期的开始;第十一个修正符表示返回下一个星期是N的日期和时间;第十二个修正符表示返回从1970-01-01开始算起的秒数;第十三个修正符表示返回本地时间。
下面举一些例子: - 计算机当前时间 SELECT date(‘now’)
- 计算机当前月份的最后一天 SELECT date(‘now’,’start of month’,’+1 month’,’-1 day’)
- 计算UNIX 时间戳1092941466表示的日期和时间 SELECT datetime(‘1092941466’,’unixepoch’)
- 计算 UNIX 时间戳1092941466 表示的本地日期和时间 SELECT datetime(‘1092941466’,’unixepoch’,’localtime’)
- 计算机当前UNIX 时间戳 SELECT strftime(‘%s’,’now’)
- 两个日期之间相差多少天 SELECT jolianday(‘now’)-jolianday(‘1981-12-23’)
- 两个日期时间之间相差多少秒 SELECT julianday(\'now\')*86400 - julianday(\'2004-01-01 02:34:56\')*86400
- 计算今年十月份第一个星期二的日期 SELECT date(\'now\',\'start of year\',\'+9 months\',\'weekday 2\');
- 取大于现在时间的数据 select * from 表 where 日期字段>datetime(\'now\',\'localtime\')
- 比较日期指定部份,举一反三,同样使用strftime格式式日期来对日、周、年比较 select * from 表 where strftime(\'%m\',日期字段)=strftime(\'%m\',\'now\')
- 大于指定时间的第一条 select title,pubtime from article where pubtime>\'2008-06-15 03:35:28\' order by pubtime asc Limit 1 Offset 0
- 小于指定时间的第一条
select title,pubtime from article where pubtime<\'2008-06-15 03:35:28\' order
by pubtime desc Limit 1 Offset 0
简单示例:
SELECT
datetime(CHANGE_DATE,\'localtime\'),
strftime(\'%Y-%m-%d\',CHANGE_DATE,\'localtime\'),
datetime(\'now\',\'localtime\'),
strftime(\'%Y-%m-%d\',\'now\',\'localtime\'),
DATE(\'now\',\'localtime\'),
time(\'now\',\'Localtime\'),
time(\'2010-11-27 01:12:21\',\'Localtime\',\'-8 hour\') as Time
FROM SALARY_HISTORY ;
SELECT * FROM SALARY_HISTORY WHERE date(CHANGE_DATE,\'Localtime\')=Date(\'now\',\'Localtime\')
一 时间的保存
经过GOOGLE发现大多数的解决方法为datetime.ToString("s")
来解决的,经过测试此方法虽然解决的问题,但还不够完美。
因为这样格式化出来的时间在用工具SQLite Developer 查看时显示的时间看起来很怪,不直观。而且如果在SQLite Developer
手动修改了时间,在程序中会报错,因为这个时候保存的时间格式发现了改变。经过测试发现datetime.ToString("yyyy-MM-dd
hh:mm:ss")可以很好的解决这个问题。
二 时间的查询
如果你用SQLite作开发,一定少不了时间的查询,一定会让你动不少脑精。因为和别的数据库不一样,就如要查询2009.3.20
11:00:00领取工资的有多少人的SQL怎么写呢,你一定会写成:
select count(*) from T where statue=\'1\' and [date]=\'2009-03-20
11:00:00\'
仔细查看会发现有问题,因为没有结果,实际表中是有结果的,这是为什么,其实我也不没有搞清楚。这个问题还是在国外的一个
论坛发现解决方法的。只要改一下上面的语句就可以了
select count(*) from T where statue=\'1\' and
datetime([date])=datetime(\'2009-03-20 11:00:00\')
or
select count(*) from T where statue=\'1\' and datetime([date])=\'2009-03-20
11:00:00\'
记住2009-03-20不能写成为2009-3-20.
以上方法经过测目前没有发现问题,当然我也是初次使用SQLite来开发一个小项目,也许还有问题没有发现出来,请各位指教!
Sqlite 查询时间格式处理
Windows 下没有strptime,sqlite查询时间戳字段后得出的是char*,要转成time_t很不爽。临时解决方案如下:
1、sql语句需要格式化时间戳字段:
string strSql = "select strftime(\'%%Y%%m%%d%%H%%M%%S\',mytime) as mytime
from mytable“;
2、写一个C函数进行转换:
long TimeStamp2Time_t(char* pszTimeStamp /*"YYYYMMDDhhmmss"*/)
{
int ttt[6][4]={{0,5,4,1900}
,{4,4,2,1}
,{6,3,2,0}
,{8,2,2,0}
,{10,1,2,0}
,{12,0,2,0}};
time_t t=time(NULL); struct tm* mytm = localtime(&t);
for(int i=0;i<6;i++)
{
char tmp[5]="";
memcpy(tmp,((char*)pszTimeStamp)+ttt[i][0],ttt[i][2]);
*(((int*)mytm)+ttt[i][1])=
atoi(tmp)-ttt[i][3];
}
return mktime(mytm);
}
SQLite支持5个日期和时间函数如下:
S.N. |
函数 |
例子 |
1 |
date(timestring, modifiers...) |
This returns the date in this format: YYYY-MM-DD |
2 |
time(timestring, modifiers...) |
This returns the time as HH:MM:SS |
3 |
datetime(timestring, modifiers...) |
This returns YYYY-MM-DD HH:MM:SS |
4 |
julianday(timestring, modifiers...) |
This returns the number of days since noon in Greenwich on November 24, 4714 B.C. |
5 |
strftime(timestring, modifiers...) |
This returns the date formatted according to the format string specified as the first argument formatted as per formatters explained below. |
上述五个日期和时间函数时间字符串作为参数。后跟零个或多个修饰符的时间字符串。 strftime()函数还需要一个格式字符串作为其第一个参数。下面的部分将给予您详细的时间字符串和改性剂的不同类型。
5.1 时间字符串:
一时间字符串可以在任何采用以下格式:
S.N. |
时间字符串 |
例子 |
1 |
YYYY-MM-DD |
2010-12-30 |
2 |
YYYY-MM-DD HH:MM |
2010-12-30 12:10 |
3 |
YYYY-MM-DD HH:MM:SS.SSS |
2010-12-30 12:10:04.100 |
4 |
MM-DD-YYYY HH:MM |
30-12-2010 12:10 |
5 |
HH:MM |
12:10 |
6 |
YYYY-MM-DDTHH:MM |
2010-12-30 12:10 |
7 |
HH:MM:SS |
12:10:01 |
8 |
YYYYMMDD HHMMSS |
20101230 121001 |
9 |
now |
2013-05-07 |
可以使用“T”作为一个文字字符分隔日期和时间。
5.2 修饰符
随后的时间字符串可以由零个或多个的修饰符将改变日期和/或任何上述五大功能返回时间。修饰符应用于从左侧到右侧和下面的修饰符可在SQLite使用:
- NNN days
- NNN hours
- NNN minutes
- NNN.NNNN seconds
- NNN months
- NNN years
- start of month
- start of year
- start of day
- weekday N
- unixepoch
- localtime
- utc
5.3 格式化:
SQLite 提供了非常方便的函数strftime() 来格式化任何日期和时间。可以使用以下替换格式化的日期和时间:
替代 |
描述 |
%d |
Day of month, 01-31 |
%f |
Fractional seconds, SS.SSS |
%H |
Hour, 00-23 |
%j |
Day of year, 001-366 |
%J |
Julian day number, DDDD.DDDD |
%m |
Month, 00-12 |
%M |
Minute, 00-59 |
%s |
Seconds since 1970-01-01 |
%S |
Seconds, 00-59 |
%w |
Day of week, 0-6 (0 is Sunday) |
%W |
Week of year, 01-53 |
%Y |
Year, YYYY |
%% |
% symbol |
5.4 例子
让我们尝试不同的例子,现在使用SQLite的提示的。以下计算当前的日期:
sqlite> SELECT date(\'now\');
2013-05-07
以下计算当前月份的最后一天:
sqlite> SELECT date(\'now\',\'start of month\',\'+1 month\',\'-1 day\');
2013-05-31
以下计算给定的日期和时间的UNIX时间戳1092941466:
sqlite> SELECT datetime(1092941466, \'unixepoch\');
2004-08-19 18:51:06
以下计算UNIX时间戳1092941466抵消本地时区的日期和时间:
sqlite> SELECT datetime(1092941466, \'unixepoch\', \'localtime\');
2004-08-19 11:51:06
以下计算当前的UNIX时间戳:
sqlite> SELECT datetime(1092941466, \'unixepoch\', \'localtime\');
1367926057
以下计算的美国“独立宣言”签署以来的天数:
sqlite> SELECT julianday(\'now\') - julianday(\'1776-07-04\');
86504.4775830326
以下一个特别的时刻在2004年以来的秒数计算:
sqlite> SELECT strftime(\'%s\',\'now\') - strftime(\'%s\',\'2004-01-01 02:34:56\');
295001572
以下计算日期为当年10月的第一个星期二:
sqlite> SELECT date(\'now\',\'start of year\',\'+9 months\',\'weekday 2\');
2013-10-01
以下计算时间自UNIX纪元秒(类似strftime(\'%s\',\'now\') ,除了包括小数部分):
sqlite> SELECT (julianday(\'now\') - 2440587.5)*86400.0;
1367926077.12598
UTC与本地时间值之间进行转换,格式化日期时间,使用UTC或localtime修改如下:
sqlite> SELECT time(\'12:00\', \'localtime\');
05:00:00
sqlite> SELECT time(\'12:00\', \'utc\');
19:00:00
以上是关于SQLite时间处理的主要内容,如果未能解决你的问题,请参考以下文章
在片段中显示适当的 SQLite 数据(相同的条目出现在不同的选择条目中)
SQLite 片段函数实现不会在 TextView 中将文本格式化为 HTML
当我从用户获取数据并将其保存到 SQLite 数据库中时,我应该怎么做才能使列表视图在片段中工作
如果我想从另一个片段中添加书签,为啥我的书签单词没有保存到 sqlite 数据库?
java.lang.NullPointerException: Attempt to invoke virtual method ‘int android.database.sqlite异常(代码片段