Silverlight via WCF RIA Service 1:快速入门及测试框架
时间:2011-06-10 来源:陆敏技
本文目的是通过Silverlight ria service完成一次数据的读取过程,并且在此基础上建立打架测试项目。
Ria service借助于WCF和ADO.NET Entity Framework构建分布式开发框架。使用它可以快速构建自己的开发模式。
1:基础结构
首先,创建SL APP,如下:
然后,选择创建web:
可以,也可以不勾选enable wcf ria service,如果勾选了,在SL APP中会多几个DLL的引用,其它没有任何差别。如图:
2:创建一个示例数据库
这是一个小而轻型的数据库,不要小看它,它是《SQL COOKBOOK》的示例数据库。如下:
CREATE TABLE emp ( EMPNO INT NOT NULL , ENAME VARCHAR(32) DEFAULT NULL , JOB VARCHAR(32) DEFAULT NULL , MGR VARCHAR(32) DEFAULT NULL , HIREDATE DATETIME DEFAULT NULL , SAL int DEFAULT NULL , COMM VARCHAR(16) DEFAULT NULL , DEPTNO VARCHAR(8) DEFAULT NULL , PRIMARY KEY ( EMPNO ) ) -- -- Dumping data for table emp -- INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7369 , 'SMITH' , 'CLERK' , '7902' , '1980-12-17' , '800' , NULL , '20' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7499 , 'ALLEN' , 'SALESMAN' , '7698' , '1981-02-20' , '1600' , '300' , '30' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7521 , 'WARD' , 'SALESMAN' , '7698' , '1981-02-22' , '1250' , '500' , '30' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7566 , 'JONES' , 'MANAGER' , '7839' , '1981-04-02' , '2975' , NULL , '20' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7654 , 'MARTIN' , 'SALESMAN' , '7698' , '1981-09-28' , '1250' , '1400' , '30' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7698 , 'BLAKE' , 'MANAGER' , '7839' , '1981-05-01' , '2850' , NULL , '30' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7782 , 'CLARK' , 'MANAGER' , '7839' , '1981-06-09' , '2450' , NULL , '10' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7788 , 'SCOTT' , 'ANALYST' , '7566' , '1982-12-09' , '3000' , NULL , '20' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7839 , 'KING' , 'PRESIDENT' , NULL , '1981-11-17' , '5000' , NULL , '10' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7844 , 'TURNER' , 'SALESMAN' , '7698' , '1981-09-08' , '1500' , '0' , '30' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7876 , 'ADAMS' , 'CLERK' , '7788' , '1983-01-12' , '1100' , NULL , '20' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7900 , 'JAMES' , 'CLERK' , '7698' , '1981-12-03' , '950' , NULL , '30' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7902 , 'FORD' , 'ANALYST' , '7566' , '1981-12-03' , '3000' , NULL , '20' ) INSERT INTO emp ( EMPNO , ENAME , JOB , MGR , HIREDATE , SAL , COMM , DEPTNO ) VALUES ( 7934 , 'MILLER' , 'CLERK' , '7782' , '1982-01-23' , '1300' , NULL , '10' ) -- -- table dept -- CREATE TABLE dept ( DEPTNO INT NOT NULL , DNAME VARCHAR(32) DEFAULT NULL , LOC VARCHAR(32) DEFAULT NULL , PRIMARY KEY ( DEPTNO ) ) INSERT INTO dept ( DEPTNO, DNAME, LOC ) VALUES ( '10', 'ACCOUNTING', 'NEW YORK' ) INSERT INTO dept ( DEPTNO, DNAME, LOC ) VALUES ( '20', 'RESEARCH', 'DALLAS' ) INSERT INTO dept ( DEPTNO, DNAME, LOC ) VALUES ( '30', 'SALES', 'CHICAGO' ) INSERT INTO dept ( DEPTNO, DNAME, LOC ) VALUES ( '40', 'OPERATIONS', 'BOSTON' ) -- -- table t1 -- CREATE TABLE T1 ( ID INT NOT NULL , PRIMARY KEY ( ID ) ) INSERT INTO t1 ( ID ) VALUES ( '1' ) -- -- table t10 -- CREATE TABLE T10 ( ID INT NOT NULL , PRIMARY KEY ( ID ) ) INSERT INTO t10 ( ID ) VALUES ( '1' ) INSERT INTO t10 ( ID ) VALUES ( '2' ) INSERT INTO t10 ( ID ) VALUES ( '3' ) INSERT INTO t10 ( ID ) VALUES ( '4' ) INSERT INTO t10 ( ID ) VALUES ( '5' ) INSERT INTO t10 ( ID ) VALUES ( '6' ) INSERT INTO t10 ( ID ) VALUES ( '7' ) INSERT INTO t10 ( ID ) VALUES ( '8' ) INSERT INTO t10 ( ID ) VALUES ( '9' ) INSERT INTO t10 ( ID ) VALUES ( '10' ) CREATE TABLE emp_bonus ( empno INT , received DATETIME , TYPE INT ) go INSERT INTO emp_bonus VALUES ( 7934, 2005 - 5 - 17, 1 ) INSERT INTO emp_bonus VALUES ( 7934, 2005 - 2 - 15, 2 ) INSERT INTO emp_bonus VALUES ( 7839, 2005 - 2 - 15, 3 ) INSERT INTO emp_bonus VALUES ( 7782, 2005 - 2 - 15, 1 )
3:RIA之ADO.NET ENTITY DATA MODEL
为web创建ADO.NET ENTITY DATA MODEL,如图:
注意,习惯命名规则:Model数据库名.edmx。在下一步中,我们选择“Generate from database”,如下:
在下一步中,我们选择刚才创建的数据库:
在下一步中,我们全选需要创建的数据对象,如红框所示:
点击Finish后,VS为我们生成了一个以edmx为后缀的文件。OK,到此暂停,我们先来看看这个edmx文件是干什么用的,为我们完成了什么工作。
3.1:什么是EDM
ENTITY DATA MODEL,简写为EDM,中文为实体数据模型。它由三个概念组成。概念模型由概念架构定义语言文件 (.csdl)来定义,映射由映射规范语言文件 (.msl),存储模型(又称逻辑模型)由存储架构定义语言文件 (.ssdl)来定义。这三者合在一起就是EDM模式。EDM模式在项目中的表现形式就是扩展名为.edmx的文件。
使用EDM,意味这我们使用的是ADO.NET Entity Framework来操作数据库,即:我们唯一能做的是操作EDM,EDM会将这个操作请求发往数据库。
Entity Framework实现了一套类似于ADO.NET2.0中连接类(它们使用方式相同,均基于Provider模式)的被称作EntityClient的类用来操作EDM。ADO.NET2.0的连接类是向数据库发送SQL命令操作表或视图,而EntityClient是向EDM发送EntitySQL操作Entity。EntityClient在EntityFramework中的作用是相当重要的,所有发往EDM的操作都是经过EntityClient,包括使用LINQ to Entity进行的操作。目前可用的操作EDM的方式如下:
4:针对web EMD的测试
理解了EMD在开发中所处的作用,我们就可以针对EDM来写个测试项目。为了简便期间,我们直接在EMD中的mysampleEntities类型的构造方法中直接创建测试(严格意义来说,这不是个单元测试,这仅是测试)。
针对这个操作,会在测试项目中生成一个mysampleEntitiesTest的类型,同时,在这个类型中会生成一个mysampleEntitiesConstructorTest的方法,如下:
由于我们仅仅测试EDM,而跟WEB本身没有关系,所有我们注释掉了红框中的内容。同时我们写入真正的测试代码,如下:
这里的一个小细节是,连接字符串我们通过编码的方式传入到测试方法。EDM的连接字符串和ADO.NET的连接字符串有很大不同,在这里,可公开一下我们的生成EDM连接字符串的方法:
public string GetConnectionString() { string providerName = "System.Data.SqlClient"; string serverName = "192.168.0.96"; string databaseName = "mysample"; SqlConnectionStringBuilder sqlBuilder = new SqlConnectionStringBuilder(); sqlBuilder.DataSource = serverName; sqlBuilder.InitialCatalog = databaseName; sqlBuilder.IntegratedSecurity = false; sqlBuilder.UserID = "sa"; sqlBuilder.Password = "sasa"; string providerString = sqlBuilder.ToString(); EntityConnectionStringBuilder entityBuilder = new EntityConnectionStringBuilder(); entityBuilder.Provider = providerName; entityBuilder.ProviderConnectionString = providerString; entityBuilder.Provider = providerName; entityBuilder.Metadata = "res://*/Modelmysample.csdl|res://*/Modelmysample.ssdl|res://*/Modelmysample.msl"; return entityBuilder.ToString(); }
调试我们的测试代码,最终我们会获取到数据库中的14条记录。
通过了以上的阐述,我们了解了EDM的作用,以及如何测试EDM。接下来,我们需要知道如何在SL APP中调用WEB中的EDM。
5:Domain Service Class
在SL APP中调用EDM是通过WEB的Domain Service Class来实现的。简单的说来,它是RIA SERVICE框架中的一个重要内容。
在WEB选择添加新项,选择Domain Service Class,命名,下一步会出现如下界面:
我们可以做出如上勾选。确定后,编译整个解决方法,会发现目前的解决方案结构会变成如下形式:
注意,第一个红框部分,是VS自动为我们在SL APP生成的。它是第二个红框在客户端的对应版本。我们需要将它包含到项目中去。它包含了所有服务端版本类中定义的方法、实体等,而可在客户端直接调用。
6:Ria Service数据流转
好了,经过以上的描述我们知道了:
EDM:直接操作数据库;
Domain Service Class:调用EDM,并把数据接口通过WCF的形式开放给客户端;
SilverlightApplication4.Web.g.cs:调用Domain Service Class,完成对数据的读取,并最终呈现给UI;
7:最终展示
在SL APP中的UI调用呈现数据。前台:
后台,直接使用SilverlightApplication4.Web.g.cs中的DomainServiceMySample加载数据:
public MainPage() { InitializeComponent(); DomainServiceMySample context = new DomainServiceMySample(); context.Load(context.GetEmpQuery()); lb1.ItemsSource = context.emps; }
最后界面显示:
本示例源码下载:SilverlightApplication420110610.rar
参考:http://www.cnblogs.com/lsxqw2004/archive/2009/05/31/1495240.html