Oracle拾遗系列(一)

Oracle拾遗系列(一)

 

鉴于内容监管困难、赢利模式缺少、成本压力巨大,市面上大大小小的「网盘云盘」要么转型要么关停。为此,我的双十一购物车里已经躺了2个1T的硬盘(就等12点开抢了),Evernote也付费升级成了高级用户,就差没自建云服务了。一些网络服务特别是免费的服务,比如之前的google reader、yahoo mail、百度空间...都是说停就停,让你措手不及。今天翻了下之前发在百度空间里的好多文章,幸好还有「微信」这么优雅的产品,就随手搬一篇老文mark下。


1、关于oracle的add_months

先来看看下面这几个select语句的结果吧:

select add_months(to_date('2010-02-28','yyyy-mm-dd'),1) from dual;
ADD_MONTHS(TO_DATE('2010-02-28
------------------------------
2010-3-31

select add_months(to_date('2012-02-28','yyyy-mm-dd'),1) from dual;
ADD_MONTHS(TO_DATE('2010-02-28
------------------------------
2010-3-31

select add_months(to_date('2010-03-31','yyyy-mm-dd'),1) from dual;
ADD_MONTHS(TO_DATE('2010-03-31
------------------------------
2010-4-30

ADD_MONTHS returns the date d plus n months. The argument n can be any integer. If d is the last day of the month or if the resulting month has fewer days than the day component of d, then the result is the last day of the resulting month. Otherwise, the result has the same day component as d.

哦,原来ADD_MONTHS会先优先判断是否当月最后一天,如果是当月的最后一天,则结果为N月后的最后一天;如果不是,则为N月后的第D天。不过当N月后没有第D天的话(比如4月份没有31日),它也会把它置成最后一天。

2、关于oracle的not in

IN是一个成员条件, 对于给定的一个集合或者子查询,它会比较每一个成员值。 IN功能上相当于 =ANY 的操作,而NOT IN 功能上相当于 !=ALL 的操作。 IN在逻辑上实际上就是对给定的成员集合或者子查询结果集进行逐条的判定。

SELECT name FROM employee A WHERE A.name not in ('gallen','tomoya',NULL)

结果会是什么呢?结果是什么都没有,结果为空集。为什么呢? 其实上面的sql语句等价于:

SELECT name FROM employee A WHERE A.name!='gallen' AND A.name!='tomoya' AND A.name!=NULL

根据NULL的运算特性和真值表,该语句无论前两个判定条件是否为真,其结果一定是NULL或者FALSE。故绝对没有任何记录可以返回。

其实平时的时候并没有上面的语句那么明显,集合往往会是一个子查询,比如:

SELECT name FROM employee A WHERE A.name not in (select name FROM employee A WHERE A.salary>8000)

这时候,你要预先在子查询里把NULL去掉,比如用nvl。

3、Oracle修改字段的类型、精度、范围

可能修改字段的精度和范围比修改字段类型来得更常见,比如我们需要把某个字段的数据类型从原来的number(10)变为number(10,2)型的实际操作。要是没有数据的话直接用以下语句即可

alter table tbl_test modify cityCode number(10,2);

但是有数据的话就不能用上面方法了,为什么呢?因为oracle只允许在类型相同的情况下,增加精度或者放大范围,"column to be modified must be empty to decrease precision or scale" 。

有人会问number(10)变为number(10,2)精度不是变大了吗?其实不然,number()类型的默认精度是38位,你不指定精度,oracle无法辨别具体位数,所以无论是number(10)变为number(10,2)还是number(10,2)变为number(10),在该字段非空的情况下都是不允许的。

alter table tbl_test add cityCode_temp number(5,2)
update tbl_test set cityCode_temp=permile;
alter table tbl_test drop column cityCode;
alter table tbl_test rename column cityCode_temp to cityCode;

这种方法会使列名发生变化,而且字段顺序增加 有可能发生行迁移,对应用程序会产生影响 以下方法是比较好的方法 不用使列名发生变化 也不会发生表迁移,但这个有个缺点是表要更新两次 如果数据量较大的话 产生的undo和redo更多 ,前提也是要停机做 要是不停机的话 ,也可以采用在线重定义方式来做 以下是脚本:

alter table tbl_test add cityCode_temp number;
Add/modify columns
alter table tbl_test modify cityCode null;
update tbl_test set cityCode_temp=cityCode,cityCode=null;
commit;
alter table tbl_test modify cityCode number(10,2);
update tbl_test set cityCode=cityCode_temp,cityCode_temp=null;
commit;
alter table tbl_test drop column cityCode_temp;
alter table tbl_test modify cityCode not null;
select * from tbl_test ;

上述的相关内容就是对Oracle修改字段类型的方法的描述,希望会给你带来一些帮助在此方面。

4、未完待续...


欢迎关注我的个人微信公众号:能叔

能叔

猜你喜欢

转载自tomoya.iteye.com/blog/2347490