Castle ActiveRecord学习实践(9):使用ActiveRecord的一些技巧

摘要:本文将会介绍使用ActiveRecord中的一些技巧。

 

主要内容

1.由实体类生成数据表

2.运行存在的SQL脚本

3.使用空属类型

4.使用枚举类型的属性

5.使用NHibernate中的日志记录

 

一.由实体类生成数据表

在前面所用到的例子中我们都是先有数据表结构,然后才有实体类,然而这会让很多朋友认为ORM怎么变成了ROM了,其实这只是我们平时的一个开发时的习惯问题,ActiveRecord是支持先有实体类,再由实体类生成数据库表。只不过我们可以在开发中根据项目的实际情况在这两种之间选择。看下面的代码,要生成数据库表结构,在实体类中需要多提供一些信息,是否为空,字段长度等。

None.gif [ActiveRecord( " Blogs " )]
None.gif
None.gif
public   class  Blog : ActiveRecordBase
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
InBlock.gif    
private int _id;
InBlock.gif
InBlock.gif    
private String _name;
InBlock.gif
InBlock.gif    
private String _author;
InBlock.gif
InBlock.gif 
InBlock.gif
InBlock.gif    [PrimaryKey(PrimaryKeyType.Native, 
"blog_id")]
InBlock.gif
InBlock.gif    
public int Id
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
get dot.gifreturn _id; }
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
set dot.gif{ _id = value; }
InBlock.gif
ExpandedSubBlockEnd.gif    }

InBlock.gif
InBlock.gif 
InBlock.gif
InBlock.gif    [Property(
"blog_name", NotNull=true, Length=25)]
InBlock.gif
InBlock.gif    
public String Name
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
get dot.gifreturn _name; }
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
set dot.gif{ _name = value; }
InBlock.gif
ExpandedSubBlockEnd.gif    }

InBlock.gif
InBlock.gif 
InBlock.gif
InBlock.gif    [Property(
"blog_author", NotNull=true, Length=50)]
InBlock.gif
InBlock.gif    
public String Author
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
get dot.gifreturn _author; }
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
set dot.gif{ _author = value; }
InBlock.gif
ExpandedSubBlockEnd.gif    }

InBlock.gif
ExpandedBlockEnd.gif}


要生成数据库表需要调用ActiveRecordStarter.CreateSchema()方法就可以了。

None.gif public   void  Initli()
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
InBlock.gif    XmlConfigurationSource source 
= new XmlConfigurationSource("MyConfig.xml");
InBlock.gif
InBlock.gif    
InBlock.gif
InBlock.gif    ActiveRecordStarter.Initialize( source, 
typeof(Blog),typeof(Post),typeof(Custom));
InBlock.gif
InBlock.gif    ActiveRecordStarter.CreateSchema();
InBlock.gif
ExpandedBlockEnd.gif}


这里需要注意两点:

 

1.生成数据库表时只有当该表不存在时ActiveRecord才会生成,否则表如果存在ActiveRecord不会做任何事情,也不会报任何错误。

2.如果在实体类中没有指定字段的长度和是否为空,则默认生成的字段是允许为空的,且字符类生成后的字段类型为Nvarchar,长度为255

 

二.运行存在的SQL脚本

有时候我们会想在ActiveRecord框架启动时运行一个已经存在的SQL脚本来生成数据库表结构,ActiveRecord同样也提供了这样的功能,通过调用ActiveRecordStarter.CreateSchemaFromFile()来实现。示例代码如下:

None.gif public   void  Initli()
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
InBlock.gif    XmlConfigurationSource source 
= new XmlConfigurationSource("MyConfig.xml");
InBlock.gif
InBlock.gif    
InBlock.gif
InBlock.gif    ActiveRecordStarter.Initialize( source, 
typeof(Blog));
InBlock.gif
InBlock.gif 
InBlock.gif
InBlock.gif    ActiveRecordStarter.CreateSchemaFromFile(
"MySqlScript.sql");
InBlock.gif
ExpandedBlockEnd.gif}

 

三.使用空属类型

在进行数据库操作时,有时候需要进行空值的处理,在ActiveRecord中给我们提供了一组空属类型,可以方便的进行处理,比如可以这样写属性:

None.gif [Property]
None.gif
None.gif
public  NullableDateTime CreatedDate
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
get dot.gifreturn _createdDate; }
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
set dot.gif{ _createdDate = value; }
InBlock.gif
ExpandedBlockEnd.gif}

None.gif
None.gif 
None.gif
None.gif[Property]
None.gif
None.gif
public  NullableInt32 Count
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
get dot.gifreturn _count; }
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
set dot.gif{ _count = value; }
InBlock.gif
ExpandedBlockEnd.gif}


看一下ActiveRecord提供的空属类型与实际类型对照表

CLR Basic Type

Nullable Type

System.Boolean

Nullables.NullableBoolean

System.Byte

Nullables.NullableByte

System.Char

Nullables.NullableChar

System.DateTime

Nullables.NullableDateTime

System.Decimal

Nullables.NullableDecimal

System.Double

Nullables.NullableDouble

System.Guid

Nullables.NullableGuid

System.Int16

Nullables.NullableInt16

System.Int32

Nullables.NullableInt32

System.Int64

Nullables.NullableInt64

System.SByte

Nullables.NullableSByte

System.Single

Nullables.NullableSingle

注意在使用空属类型时需要添加以下引用

None.gif Nullables.dll
None.gif
None.gifNullables.NHibernate.dll

 

四.使用枚举类型属性

ActiveRecord中我们可以定义一个属性的类型为枚举类型,示例代码:

None.gif public   class  Post : ActiveRecordBase
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
InBlock.gif    
//dot.gifdot.gif
InBlock.gif

InBlock.gif 
InBlock.gif
InBlock.gif    
private StatusType _status_type_id;
InBlock.gif
InBlock.gif 
InBlock.gif
InBlock.gif    
public enum StatusType
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif
InBlock.gif        Editing 
= 0,
InBlock.gif
InBlock.gif        Published 
= 1,
InBlock.gif
InBlock.gif        Archived 
= 2
InBlock.gif
ExpandedSubBlockEnd.gif    }

InBlock.gif
InBlock.gif 
InBlock.gif
InBlock.gif    [Property(
"status_id")]
InBlock.gif
InBlock.gif    
public StatusType Status
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif    
dot.gif{
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
get dot.gifreturn _status_type_id; }
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif        
set dot.gif{ _status_type_id = value; }
InBlock.gif
ExpandedSubBlockEnd.gif    }

InBlock.gif
ExpandedBlockEnd.gif}


在使用该实体类的StatusType属性时,可以这样赋值:

None.gif Post.Status  =  Post.StatusType.Archived;

 

五.使用NHibernate的日志记录

用过NHibernate的朋友都知道,NHibernate是用Log4net来记录日志的,在ActiveRecord中也可以通过简单的配置来使用Log4net记录。配置示例:

None.gif <? xml version="1.0" encoding="utf-8"  ?>
None.gif
None.gif
< configuration >
None.gif
None.gif    
< configSections >
None.gif
None.gif        
< section  name ="nhibernate"  type ="System.Configuration.NameValueSectionHandler, System, Version=1.0.5000.0,Culture=neutral, PublicKeyToken=b77a5c561934e089"   />
None.gif
None.gif        
< section  name ="activerecord"  type ="Castle.ActiveRecord.Framework.Config.ActiveRecordSectionHandler, Castle.ActiveRecord"   />
None.gif
None.gif        
< section  name ="log4net"  type ="log4net.Config.Log4NetConfigurationSectionHandler, log4net"   />
None.gif
None.gif    
</ configSections >
None.gif
None.gif    
< activerecord >      
None.gif
None.gif    
</ activerecord >
None.gif
None.gif    
< log4net >
None.gif
None.gif        
<!--  Define some output appenders  -->
None.gif
None.gif        
< appener  name ="trace"  type ="log4net.Appender.TraceAppender, log4net" > Õ
None.gif
None.gif            
< layout  type ="log4net.Layout.PatternLayout,log4net" >
None.gif
None.gif                
< param  name ="ConversionPattern"  value ="%d [%t] %-5p %c [%x] &lt;%P{user}&gt; - %m%n"   />
None.gif
None.gif            
</ layout >
None.gif
None.gif        
</ appener > Õ
None.gif
None.gif        
< appener  name ="console"  type ="log4net.Appender.ConsoleAppender, log4net" > Õ
None.gif
None.gif            
< layout  type ="log4net.Layout.PatternLayout,log4net" >
None.gif
None.gif                
< param  name ="ConversionPattern"  value ="%d [%t] %-5p %c [%x] &lt;%P{user}&gt; - %m%n"   />
None.gif
None.gif            
</ layout >
None.gif
None.gif        
</ appener > Õ
None.gif
None.gif        
< appener  name ="rollingFile"  type ="log4net.Appender.RollingFileAppender,log4net" > Õ
None.gif
None.gif            
< param  name ="File"  value ="log.txt"   />
None.gif
None.gif            
< param  name ="AppendToFile"  value ="true"   />
None.gif
None.gif            
< param  name ="RollingStyle"  value ="Date"   />
None.gif
None.gif            
< param  name ="DatePattern"  value ="yyyy.MM.dd"   />
None.gif
None.gif            
< param  name ="StaticLogFileName"  value ="true"   />
None.gif
None.gif            
< layout  type ="log4net.Layout.PatternLayout,log4net" >
None.gif
None.gif                
< param  name ="ConversionPattern"  value ="%d [%t] %-5p %c [%x] &lt;%X{auth}&gt; - %m%n"   />
None.gif
None.gif            
</ layout >
None.gif
None.gif        
</ appener > Õ
None.gif
None.gif        
< root >
None.gif
None.gif            
<!--  priority value can be set to ALL|INFO|WARN|ERROR  -->
None.gif
None.gif            
< priority  value ="ALL"   />
None.gif
None.gif            
< appener-ref  ref ="rollingFile"   /> Õ
None.gif
None.gif        
</ root >
None.gif
None.gif    
</ log4net >
None.gif
None.gif    
< nhibernate >
None.gif
None.gif        
<!--  with this set to true, SQL statements will output to the console window if it's a console app  -->
None.gif
None.gif        
< add  key ="hibernate.show_sql"  value ="true"   />
None.gif
None.gif    
</ nhibernate >
None.gif
None.gif
</ configuration >


初始化配置在调用ActiveRecordStarter.Initialize()方法之前

None.gif public   void  Initli()
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
InBlock.gif    XmlConfigurationSource source 
= new XmlConfigurationSource("MyConfig.xml");
InBlock.gif
InBlock.gif    log4net.Config.XmlConfigurator.Configure();
InBlock.gif
InBlock.gif    ActiveRecordStarter.Initialize( source, 
typeof(Blog));
InBlock.gif
ExpandedBlockEnd.gif}


六. Hooks

 

有时候我们会在保存,加载,删除等操作时做一些必需的处理,这时可以通过重载以下三个方法来实现:

None.gif BeforeSave(IDictionary state) 
None.gif
None.gifBeforeLoad(IDictionary state) 
None.gif
None.gifBeforeDelete(IDictionary state)


比如说我们想在保存的时候设置Post的创建时间为当前时间,可以这样去写

None.gif protected   override   bool  BeforeSave(IDictionary state)
None.gif
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif
InBlock.gif    state[
"Created"= DateTime.Now;
InBlock.gif
InBlock.gif    
return true;
InBlock.gif
ExpandedBlockEnd.gif}

 

本文就到这里了,同时本篇也是Castle ActiveRecord学习实践系列的最后一篇,过几天我会提供一个完整的ActiveRecord的例子程序。下面有时间我会继续写Castle中的其他部分,包括IOCFacilityAspect#DynamicProxy等。

 

参考资料

文中很多内容来自于Castle的官方网站http://www.castleproject.org

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值