CloudStack的VO在调用setRemoved方法抛异常的原因
今天在开发中发现一个问题,本来想对一个VO对象的removed值赋值,然后去update一下这条记录,一个最简单的set方法,但是在调用时直接抛异常了。
1: public void setRemoved(Date removed) {
2: this.removed = removed;
3: }
当时很诧异,没有想到这地方会出问题,后来看代码才发现原来cs在这里有拦截器,com.cloud.utils.db.UpdateBuilder#intercept
1: @Override
2: public Object intercept(Object object, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
3: String name = method.getName();
4: if (name.startsWith("set")) {
5: String field = methodToField(name, 3);
6: makeChange(field, args[0]);
7: } else if (name.startsWith("incr")) {
8: makeIncrChange(name, args);
9: } else if (name.startsWith("decr")) {
10: makeDecrChange(name, args);
11: }
12: return methodProxy.invokeSuper(object, args);
13: }
所有set开头的方法都被拦截,转调com.cloud.utils.db.UpdateBuilder#makeChange
1: protected Attribute makeChange(String field, Object value) {
2: Attribute attr = _dao._allAttributes.get(field);
3:
4: assert (attr == null || attr.isUpdatable()) : "Updating an attribute that's not updatable: " + field;
5: if (attr != null) {
6: if (attr.attache == null) {
7: _changes.put(field, new Ternary<Attribute, Boolean, Object>(attr, null, value));
8: } else {
9: if (_collectionChanges == null) {
10: _collectionChanges = new HashMap<Attribute, Object>();
11: }
12: _collectionChanges.put(attr, value);
13: }
14: }
15: return attr;
16: }
在那句assert那里,会判断attr是否可以update。
Attribute是cloudstack的基类GenericDaoBase中保存的对象属性集合,是一个保存了数据表的列名以及列属性的Map对象,这个对象通过Java JPA来进行持久化。
每一个干活的DAO对象都会继承于GenericDaoBase,并且实现自己这类对象的接口,如下图:

刚才说的那个isUpdatable()是个人认为cloudstack中有关数据库权限做的很精巧的地方
1: public final boolean isUpdatable() {
2: return Flag.Updatable.check(flags);
3: }
这里面会调用Flag这个enum对象的check方法,那个flags同样也是在调用Attribute的构造方法时生成的,它根据数据库表的每一列的属性,按照一个逻辑去计算相关的权限,flags的初始值为0,然后针对每一种权限依次去做或运算,最后保存为一个值,遇到需要校验权限的地方,就用下面的check方法,看传进来的权限,比如Flag.Updatable,与flags去与运算,如果与完了的结果与传进来的权限值一样,那么可以继续进行,否则就会由于assert计算表达式的结果为false而退出去。
1: public boolean check(int value) {
2: return (value & place) == place;
3: }
今天的问题就在这,Attribute.flags的值导致assert的表达式结果为false,在初始化时,flags的值明明为135(128+4+2+1),但是在实际计算时却不是这个值,一度很不解。但是为了解决生产环境的问题,不能在这个细节上耽搁太久,于是换了个思路,尝试着调用DAO的remove方法,也能实现针对数据库表的removed字段赋值。具体到为什么flags的值变了,还需要抽时间好好研究一下。
不过值得欣慰的是,如果不是set时抛异常了,自己可能也不会去啃这部分代码来了解。
附1,计算flags时的代码(com.cloud.utils.db.Attribute#setupColumnInfo):
1: protected void setupColumnInfo(Class<?> clazz, AttributeOverride[] overrides, String tableName, boolean isEmbedded, boolean isId) {
2: flags = Flag.Selectable.setTrue(flags);
3: GeneratedValue gv = field.getAnnotation(GeneratedValue.class);
4: if (gv != null) {
5: if (gv.strategy() == GenerationType.IDENTITY) {
6: flags = Flag.DbGenerated.setTrue(flags);
7: } else if (gv.strategy() == GenerationType.SEQUENCE) {
8: assert (false) : "Sequence generation not supported.";
9: flags = Flag.DaoGenerated.setTrue(flags);
10: flags = Flag.Insertable.setTrue(flags);
11: flags = Flag.SequenceGV.setTrue(flags);
12: } else if (gv.strategy() == GenerationType.TABLE) {
13: flags = Flag.DaoGenerated.setTrue(flags);
14: flags = Flag.Insertable.setTrue(flags);
15: flags = Flag.TableGV.setTrue(flags);
16: } else if (gv.strategy() == GenerationType.AUTO) {
17: flags = Flag.DaoGenerated.setTrue(flags);
18: flags = Flag.Insertable.setTrue(flags);
19: flags = Flag.AutoGV.setTrue(flags);
20: }
21: }
22:
23: if (isEmbedded) {
24: flags = Flag.Embedded.setTrue(flags);
25: }
26:
27: if (isId) {
28: flags = Flag.Id.setTrue(flags);
29: } else {
30: Id id = field.getAnnotation(Id.class);
31: if (id != null) {
32: flags = Flag.Id.setTrue(flags);
33: }
34: }
35: column = field.getAnnotation(Column.class);
36: if (gv == null) {
37: if (column == null || (column.insertable() && column.table().length() == 0)) {
38: flags = Flag.Insertable.setTrue(flags);
39: }
40: if (column == null || (column.updatable() && column.table().length() == 0)) {
41: flags = Flag.Updatable.setTrue(flags);
42: }
43: if (column == null || column.nullable()) {
44: flags = Flag.Nullable.setTrue(flags);
45: }
46: Encrypt encrypt = field.getAnnotation(Encrypt.class);
47: if (encrypt != null && encrypt.encrypt()) {
48: flags = Flag.Encrypted.setTrue(flags);
49: }
50: }
51: ElementCollection ec = field.getAnnotation(ElementCollection.class);
52: if (ec != null) {
53: flags = Flag.Insertable.setFalse(flags);
54: flags = Flag.Selectable.setFalse(flags);
55: }
56:
57: Temporal temporal = field.getAnnotation(Temporal.class);
58: if (temporal != null) {
59: if (temporal.value() == TemporalType.DATE) {
60: flags = Flag.Date.setTrue(flags);
61: } else if (temporal.value() == TemporalType.TIME) {
62: flags = Flag.Time.setTrue(flags);
63: } else if (temporal.value() == TemporalType.TIMESTAMP) {
64: flags = Flag.TimeStamp.setTrue(flags);
65: }
66: }
67:
68: if (column != null && column.table().length() > 0) {
69: table = column.table();
70: }
71:
72: columnName = DbUtil.getColumnName(field, overrides);
73: }
附2,数据库表每一列可能的所有权限列表:

CloudStack的VO在调用setRemoved方法抛异常的原因的更多相关文章
- ABP在领域事件中异步调用方法抛异常
在领域事件中调用UserRegistrationManager.RegisterAsync抛异常 Call UserRegistrationManager.RegisterAsync() throw ...
- 执行ArrayList的remove(object)方法抛异常?
简介 或许有很多小伙伴都尝试过如下的代码: ArrayList<Object> list = ...; for (Object object : list) { if (条件成立) { l ...
- Spring AOP不拦截从对象内部调用的方法原因
拦截器的实现原理很简单,就是动态代理,实现AOP机制.当外部调用被拦截bean的拦截方法时,可以选择在拦截之前或者之后等条件执行拦截方法之外的逻辑,比如特殊权限验证,参数修正等操作. 但是最近在项目中 ...
- 抛异常 throw的注意事项
子类覆盖父类只能抛出父类的异常或者子类或者子集注意:如果父类的方法没有抛异常,那么子类覆盖时绝对不能抛. 子类继承父类时,方法抛异常,要么抛父类,要么抛父类下的子类,不能抛父类平级或以上的异常 原因是 ...
- .Net中的AOP系列之《间接调用——拦截方法》
返回<.Net中的AOP>系列学习总目录 本篇目录 方法拦截 PostSharp方法拦截 Castle DynamicProxy方法拦截 现实案例--数据事务 现实案例--线程 .Net线 ...
- C++调用JAVA方法详解
C++调用JAVA方法详解 博客分类: 本文主要参考http://tech.ccidnet.com/art/1081/20050413/237901_1.html 上的文章. C++ ...
- 原!! java直接打印一个对象时,并不是直接调用该类的toString方法 ,而是会先判断是否为null,非null才会调用toString方法
网上看了好多java直接打印一个对象时,直接调用该类的toString方法 . 但是: Object obj=null; System.out.println(obj);//没有报错 System.o ...
- JNI调用native方法出现 java.lang.UnsatisfiedLinkError: XXXclass.XXXmethod()异常的解决办法
昨天拿到JNI的Android工程Demo,然后把demo整合到开发的主线工程上,发现调用JNI方法一直抛同一个异常 java.lang.UnsatisfiedLinkError: XXXclass. ...
- Java 反射 Method的invoke回调调用任意方法
Java 反射 Method的invoke回调调用任意方法 @author ixenos 关键子:Method.Field.invoke方法指针/函数指针.回调函数 invoke回调流程示例 0.由C ...
随机推荐
- Div 自适应屏幕大小
http://blog.csdn.net/wodetiankong516/article/details/7827256 Background 有时, 我们需要将div或者其他的Elemen ...
- python在linux上的GUI无法弹出界面
在进行python写GUI程序的时候,使用Tkinter,发现无法执行程序,报错如下: X connection to localhost:10.0 broken(explicit kill or s ...
- centos6.3 安装配置redis
1.下载安装 1.1 下载包 注:在http://download.redis.io/releases查询需要下载的版本 wget http://download.redis.io/releases/ ...
- 刚刚大学毕业,自己搭网站遇到的问题 一:tomcat中同时部署两个项目的问题
最近直接把两个项目打成war包在tomcat下发布,出现了很多莫名奇妙的问题,就是不能发布成功,只能有一个项目能成功,在网上查了很多方法,以为是两个项目中jar包出现冲突,也按照网上的方法把两个项目中 ...
- VBA
1.ActiveWorkbook是Application对象的一个属性,表示的是一个active Workbook. 2.Application对象可以获得一些顶级的对象,比如ActiveCell,A ...
- APIO2014 爆零总结
真心爆零 不要不服 这次apio给了一种新的赛制 看上去很好? 所有人都可以在线提交 并且实时知道自己的分数 它对每个题目分成若干分数段 每个分数段有若干数据 要获得这个分数段的分数需要通过这个分数段 ...
- hdu5072-Coprime(容斥原理)
题意:给N个互不相同的数,选择出两两互质或者两两不互质的三个数,有多少种选法. 题解:一共有C(N,3)中选择方式,减去不符合要求的,剩下的就是答案. 详见 http://blog.csdn.net/ ...
- tomcat的host配置
本机 etc\hosts 首先了解C:\WINDOWS\system32\drivers\etc\hosts文件配置 127.0.0.1 static1.ezsins.com #adoble ps c ...
- windows XP系统内核文件分析(全)
Windows XP个别 System32 文件 System32 文件夹下个别要移除的文件 我们就要删除另外600 个 system32 文件...我们要一次把它们全都解决掉. 以下是我所删除的 S ...
- URAL 2068 Game of Nuts (博弈)
题意:给定 n 堆石子,每次一个人把它们分成三堆都是奇数的,谁先不能分,谁输. 析:因为每堆都是奇数,那么最后肯定都是要分成1的,那么就把不是1的全加和,然后判断奇偶就OK了. 代码如下: #prag ...