WCF读书笔记(2)

信道形状(Channel Shape) 用来表述不同的消息交换模式对消息交换双方信道(信息交换的管道)的不同要求,有什么IOutputChannel IReplyChannel IDuplexChannel之类的。。 对于IReplyChannel,服务器返回一个RequestContext类型,作为请求和回复之间的一道桥梁,可以获取也可以返回消息。 会话信道(Session Channel) 从状态保持的角度,信道可以分为两种类型,Datagram Channel和Session Channel,前者不和客户端绑定,后者可以识别客户端。 对于WCF的信道层来说,信道管理器在客户端和服务端扮演不同的角色。服务端的信道管理器用于监听来自客户端的请求,而客户端的信道仅仅是单纯创建用于请求发送和回复接收的信道,因此服务端的消息管理器又称为信道监听器(Channel Listener),客户端的信道管理器则称之为信道工厂(Channel Factory) 绑定元素(Binding Element) 构成一个绑定对象的元素,绑定实现了通信的所有细节,通过创建信道栈实现对消息的交换,系统绑定是指服务于某种类型场景的绑定元素的有序集合。 包括什么BaseHttpBinding之类的。 一个程序集包括元数据,中间语言代码,和资源。程序集已经加载,将一直保存在内存中,直到应用程序域卸载。最好摒弃添加服务引用的服务调用方式,而是直接将包含服务契约的程序集部署到客户端。客户端以直接创建代理的方式进行调用。 WCF可以看成是适配器,是CLR类型和XML两个不同世界的纽带。 依赖倒置原则:即抽象不应该依赖细节,细节应该依赖于抽象;即要针对接口编程,不要对实现编程。高层模块不应该依赖低层模块。两个都应该依赖抽象。 契约关心的是我能做到。而不在于我如何做到。 封送(Marshaling) 解决跨应用程序域对象访问的问题,需要采用一种特别的机制,那就是封送。分为按值封送和按引用封送 按值封送实现了跨应用程序域的数据共享 按引用封送则实现了跨应用程序域的远程调用。 如果一个程序员频繁的使用复制粘贴编程。那就意味着设计需要重构

2012-09-06 · 1 min · bystander

WCF读书笔记(1)

信道的分类:Transport Channel 信道 1.Message Encording Channel 2.Protocol Channel 信道管理器(Channel Manager),信道管理器用于信道栈的创建和生命周期的管理 1.ChannelListener 2.ChannelFactory WCF服务调用的两种典型方式 1.是借助代码生成工具svcUtil.exe导入元数据生成客户端代码和配置,添加服务引用采用的就是这种方式,工具会创建一个继承自Client的服务代理类型。 2.是通过ChannelFactory直接创建服务代理对象进行服务调用。 如果客户端已经进行了终结点的配置 那么通过信道工厂进行调用的代码大致如下: using(Channel<IService> channelFactory=new ChannelFactory<IService>("Service")) { IService ise=channelFactory.CreateChannel(); using(ise as IDisposable) { ise.MethodName(); } } WCF处理的是跨应用程序域,跨机器,跨网络的通信,所以WCF大多数时间进行网络传俗这样的IO操作,IO绑定的操作是采用异步编程(APM【Asynchronous Programming Model】) 消息交换模式MEP(Message Exchange Pattern) MEP定义了参与者进行消息交换的模板,代表一系列的模板,他们定了消息的发送者和接收者,相互进行消息传递的次序,比较典型的三种 1.Datagram 数据包模式,嘴尖的SEND/FORGET模式也叫One-Way模式,基于从一个源到另一个或多个目的地的单向消息传输,并不期待回复 消息报的发送可以分成三个模式,分别是单目的地模式,多投模式,广播模式。 依次接受方更强大。。 2.Request/Reply模式 使用最多的一种模式,请求期待回复。采用同步通信方式,但也可用于异步通信 3.Duplex 双工模式 双方可以互发消息,实现服务器回调客户端。订阅/发布是其中一种典型的实例,TCP可以提供原生的双工通信,WCF通过WSDualHttpBinding实现了基于Http的双工通信,实际上是采用两个HTTP通道实现

2012-09-05 · 1 min · bystander

超时时间已到,但是尚未从池中获取连接

 前段时间数据库的时候,出现这个问题。一般是读到20多万条的时候。会出现这个问题。 找了一下。主要是这两个问题: 一、看所有open的连接是否都close了。 二、如果访问量很大,加上Max Pool Size=512这一句,当然这是要以损失系统性能为代价的! 我查了一下。数据库连接所有的open都关闭了。 max pool size也确实加上了。但是。问题依旧。。 哦,还有的说在数据库连接字串中添加Connect Timeout=500,也就是设置连接超时更长一些。问题依旧。。 后来看到一篇文章中说: DataReader是独占连接 的,每个DataReader都要占用一个连接。当然这个情况是偶尔出现的,所以会很长时间出现一次,因为只有同时有超过连接池最大连接数量的并发操作才 会发生。而且你加大并发数量只能暂时缓解问题。 文中建议用使用dataset来读取。然后我就把读取数据中使用DataReader的地方全部用DataAdapter和DataSet来获取数据库数据,因为DataSet非独占。会将数据保存在内存中,一次连接后释放,问题解决。 记录一下。

2012-09-03 · 1 min · bystander

SqlServer 2008开启远程连接

 对于需要外部访问数据库的操作,需要开启sql server的远程连接。没经验的我等Google之。。大部分操作按照SQL Server 2008 R2如何开启数据库的远程连接来操作 但是。有一些很小的细节需要注意。我的数据库是Sql Server 2008 Express版。这个是VS自带的。为了管理方便,可以安装SQL Server® 2008 Management Studio Express 安装过程不多说。安装完成后,直接打开 服务器名称默认是空的。。“.”好像是不行的,这时候点击右边箭头。更多,本地和远程服务器。在远程服务器里可以找到。点击就可以了。 我按照文章改完。sa还是登不上。。然后又试了一些 如上图右键sa,属性,常规里设置sa密码。不要太简单。状态里的登录选项设为启用。ok。。我碰上的问题就这几个。

2012-08-20 · 1 min · bystander

未能加载文件或程序集“App_Web_xxxx”

 今天在用WCF写服务的时候,服务一直连不上,直接查看svc文件,发现如下错误 未能加载文件或程序集“‘App_Web_****, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",****是一个随机字符串。 解决方法中: 在web.config里配置成这样子: <compilation debug="true" batch="false"> 就好了。中午和下午一直在搞javascript连接WCF的demo,总算是晚上搞定了,,还是老样子,,最大的体会就是很多东西就是看着简单,写起来会有各种各样的问题,比如这次,即使照着微软的官方文档来,也会有错误。动手才是王道,不管做什么。这几天忙完了,写篇文章出来。 最后分享一下微软官方的WCF与Ajax开发实践系列课程,我只能说WCF这东西没有哪一本书比微软官方的技术培训讲的更好了,非常建议学习。 武汉最近下雨了,天气挺凉爽,过几天准备回家吧。。

2012-07-14 · 1 min · bystander

C#中的Array和ArrayList

数组是最基础的数据结构。ArrayList可以看作是Array的复杂版本。下面比较两者的异同 比较: 相同点: Array和ArrayList均实现了相同的接口,因此具有许多相同的操作方法,例如对自身进行枚举,能够以foreach语句遍历。 Array和ArrayList创建的对象均保存在托管堆中。 不同点: Array只能存储同构对象,当然,声明为Object类型的数组除外,因为任何类型都可以隐式转换成Object类型。ArrayList可以存储异构对象,这是因为本质上ArrayList内部维护着一个object[] items类型的字段,在应用ArrayList时,应该考虑装箱和拆箱所带来的性能损失。一般情况下,Array的性能高于ArrayList。 Array可以是一维的,也可以是多维的,而ArrayList只能是一维的。 Array的容量是固定的。一旦声明,不可更改,而ArrayList容量动态增加,当添加元素超过初始容量时,ArrayList会根据需要重新分配。而且可以通过TrimToSize删除空项来压缩体积。其实,除了Array外,其他集合类都是可以动态增加的。 Array的下限可以设置,而ArrayList下限只能是0. Array只有简单的方法来获取或设置元素值,不能随意增加或删除数组元素,而ArrayList提供了更多的方法来操作元素,可以方便的插入或删除指定位置上的元素。一般可以用ArrayList代替Array. tip:List泛型类对应于ArrayList。

2012-07-11 · 1 min · bystander

C#中的字符串驻留

字符串string可以理解为char[],他是一个引用类型。 字符串创建 操作上类似于int,char等类型,直接进行赋值,string str=“bystander”;虽然string 是个类,但是如果你天真的使用 string str=new string("bystander"); 来构造,会导致一个编译错误。因为System.String只提供了数个接受Char*,Char[]类型参数的构造函数。 所以只能像下面这样使用, Char[] cs={'b','y','e'}; String str=new String(cs); 看出来了吧,很麻烦的。所以一般还是使用第一种。 字符串恒定性 恒定性(Immutability),是指字符串一经创建,就不可改变,这是String最为重要的特性之一。具体来说,就是字符串一旦创建,就会在托管堆上分配一块连续的内存空间,我们对其的任何改变都不会影响到原有的String对象,而是重新创建的新的String对象。类似Insert,Substring,ToUpper都只是创建出了新的临时的字符串,会成为下次垃圾回收的目标。 好处: 保证了对String对象的任何操作不会改变源字符串。 恒定性还意味着操作字符串不会出现线程同步问题 恒定性一定程度上,成就了字符串驻留。 字符串驻留 CLR维护一个表来存放字符串,该表叫做拘留表(或驻留池),他包含程序上以编程方式声明或创建的每一个唯一的字符串的引用,因此具有特定值的实例在系统中只有一个。如果将同一个字符串分配给多个变量,那么CLR就会向拘留池检索相同引用,并分配给变量。 通过下面这个例子来说明: class StringInterning { public static void Main() { string strA = "bystander"; string strB = "bystander"; Console.WriteLine(ReferenceEquals(strA,strB)); string strC = "by"; string strD = strC+"stander"; Console.WriteLine(ReferenceEquals(strA,strD)); strD=String.Intern(strD); Console.WriteLine(ReferenceEquals(strA,strD)); } } 猜猜答案是什么。。 正确答案是:True,False,True 为什么不是我们通常认为的那样呢。这就是因为字符串驻留了 缘起 String类型的特性恒定性,对字符串的任何操作都只会创建新的字符串,这会导致性能下降,而String又用的很频繁,为此,CLR使用字符串驻留来解决这一问题。为此,CLR内部维护一个哈希表,来管理其创建的大部分string对象。其中Key为string本身,Value为分配给对应的string的内存地址。 细节 string strA = "bystander"; CLR初始化时,创建一个空的哈希表,当JIT编译方法的时候,会首先在哈希表中查找每一个字符串常量,显然找不到任何"bystander"变量,因此会在托管堆中创建一个新的string对系那个strA,并更新哈希表,Key被赋值为"bystander",Value被赋值为strA的引用.也就是Value内保留了"bystander"在托管堆中的引用地址. string strB = "bystander"; 接着,JIT根据"bystander"查找哈希表,结果找到了,所以JIT不做操作,只把找到的key对应的Value值赋给了strB对象.因此,第一个输出为true,引用相等. string strC = "by"; string strD = strC+"stander"; 同样,JIT向哈希表中添加了Key为"by",Value为托管堆上"by"的地址.返回strC对象.但是注意,strD不同,JIT不检测,因为strD他是动态生成的.这样的字符串不会被添加到哈希表中进行维护,而是在托管堆中直接分配,所以第二个Console输出False. 对于第三个,我们首先看看Intern方法和IsInterned方法,对于动态生成的字符串,因为没有添加到CLR维护的哈希表,所以字符串驻留机制对其失效,但是可以手工开启,来实现高效的比较字符串相等. public static string Intern(string str) public static string IsInterned(string str) 两者的机制都是去哈希表中查找是否存在str字符串,找到的话也都返回对str的引用,不同的是当哈希表中没有str的话,IsInterned返回null,而Intern将把这个字符串添加到哈希表,并返回引用.注意,IsInterned返回非null并不代表两个字符串引用了相同的地址. 所以 strD=String.Intern(strD); Console.WriteLine(ReferenceEquals(strA,strD)); 就很好理解了.

2012-07-08 · 1 min · bystander

C#中XML和二进制的序列化

看书的时候,看到的。然后感觉书上的写的不清楚,于是自己写了一下。还真的有问题。 要进行序列化和反序列化,首先要定义一个可以序列化的类,方法是在类的声明前加上特性 [Serializable] 定义了一个简单的用户类,需要注意的是私有字段是不能序列化的,只有公有字段和公有属性才可以。如下 [Serializable] //特性,可以序列化 public class UserData { public string Name; public int Age; private string sex; public string Sex { set{sex=value;} get{return sex;} } public UserData() { } public UserData(string name, int age, string sex) { Name = name; Age = age; Sex = sex; } } 注意,书上有个例子没有给出默认的构造函数,实际测试时如果没有默认构造函数,是不能执行xml序列化的。注意。 然后就导入需要的命名空间。 需要导入 using System.IO; using System.Runtime.Serialization.Formatters.Binary; using System.Xml.Serialization; 分贝对应文件操作,二进制序列化和xml序列化 为了方便,我封装了四个静态函数,用于实现序列化和反序列化。 //封装二进制序列化方法 public static void BinarySerialize(UserData user) { FileStream fs = new FileStream("user.bin", FileMode.Create); BinaryFormatter formater = new BinaryFormatter(); //执行序列化 formater.Serialize(fs, user); fs.Close(); } //封装二进制反序列化方法 public static UserData BinaryDeserialize() { FileStream fs = new FileStream("user.bin", FileMode.Open, FileAccess.Read, FileShare.Read); BinaryFormatter formater = new BinaryFormatter(); UserData user = formater.Deserialize(fs) as UserData; fs.Close(); return user; } //封装xml序列化方法 public static void XmlSerialize(UserData user) { XmlSerializer serializer = new XmlSerializer(typeof(UserData)); FileStream fs = new FileStream("user.xml", FileMode.Create); serializer.Serialize(fs, user); fs.Close(); } //封装xml反序列化方法 public static UserData XmlDeserialize() { XmlSerializer serializer = new XmlSerializer(typeof(UserData)); FileStream fs = new FileStream("user.xml", FileMode.Open); UserData user = serializer.Deserialize(fs) as UserData; fs.Close(); return user; } 代码比较简单。 最后是一个简单的测试。 ...

2012-07-07 · 1 min · bystander

CSV批量导入解决

 最近的一个事就是那个诡异的CSV导入,后来决定用最笨的方法。就是readline,然后确实是可以正常识别了。思路就是一行行读入,然后构造sql语句,保存为sql文件,然后去批量执行sql文件。其实在codeproject找到了一些类似的工具,都有不如意的地方,但是时间关系还没能自己去定制,这件事完成后,就去完善一下大牛们的工具。 因为是多个csv文件,每个文件对应于我们想建的一个表,实际是大约2000个csv文件,每个文件大约4000行数据,构造成对应的sql语句比较简单,需要注意的就是如果sql server表中的字段和关键字重名,那么需要加[],比如我们有个字段是open,那么实际sql语句中使用的时候,比如 CREATE TABLE ( [open] decimal not null, ) 简单的读写以后我们构造了2000个sql文件,然后我们需要执行这些sql文件。。必然需要写个代码来批量执行了。。为此,我先用C#生成一个批处理文件,然后来执行 void GetBat(string InputPath,string OutFile) { DirectoryInfo di = new DirectoryInfo(InputPath); FileInfo[] fi = di.GetFiles(); // Create an array representing the files in the current directory StringBuilder strtemp = new StringBuilder(); foreach (FileInfo fiTemp in fi) { string temp = "osql -S \"127.0.0.1\" -U \"sa\" -P \"broker\" -d \"test_money\" -i \""; temp += fiTemp.FullName.ToString() + "\"\n"; strtemp.Append(temp); } FileStream fs = new FileStream(OutFile, FileMode.OpenOrCreate); StreamWriter sw = new StreamWriter(fs); sw.WriteLine(strtemp); sw.Close(); fs.Close(); } 简单封装了一下,输入参数是文件夹。里面包含我们的所有sql文件。输出为一个批处理文件。。后来。执行下生成的bat文件就可以了。。 osql -S "127.0.0.1" -U "sa" -P "broker" -d "test_money" -i "D:\test\AMKR.sql" osql -S "127.0.0.1" -U "sa" -P "broker" -d "test_money" -i "D:\test\AMLJ.sql" osql -S "127.0.0.1" -U "sa" -P "broker" -d "test_money" -i "D:\test\AMLN.sql" 600M的sql文件执行起来三个多小时。。因为只算一次,所以就不考虑优化了。。鄙视我们吧。

2012-07-06 · 1 min · bystander

C#调用MatLab实现N阶幻方

 MatLab的计算能力太强大了。最近需要通过C#来调用MatLab来进行一些计算,可是MatLab没用过。安装文件在我硬盘里躺了整整一年。 我们希望的是由外部程序调用MatLab函数。所以。希望可以完全脱离MATLAB环境,实现软件的快速开发。为此需要先介绍一下MCR。 MCR简介 MCR的全称是MATLAB Compiler Runtime,即MATLAB编译器运行时。是一个由MATLAB共享类库构成的执行引擎,他能够使MATLAB文件在没有MATLAB的机器上运行。这一点和.NET Framework相对于.NET程序一样,即为程序的运行提供了底层支持。当发布程序的时候,需要将MCR也打包进来,这样没有MATLAB的机器上也能执行,MCR随MATLAB软件一同发布,可以在MATLAB中输入命令“mcr”或者“mcrinstaller”获取其保存路径: >> mcr The WIN32 MCR Installer, version 7.15, is: D:\Program Files\MATLAB\R2011a\toolbox\compiler\deploy\win32\MCRInstaller.exe MCR installers for other platforms are located in: D:\Program Files\MATLAB\R2011a\toolbox\compiler\deploy\<ARCH> <ARCH> is the value of COMPUTER('arch') on the target machine. Full list of available MCR installers: D:\Program Files\MATLAB\R2011a\toolbox\compiler\deploy\win32\MCRInstaller.exe For more information, read your local MCR Installer help. Or see the online documentation at MathWorks' web site. (Page may load slowly.) 根据上面的运行结果,可知该文件的完整路径是: D:\Program Files\MATLAB\R2011a\toolbox\compiler\deploy\win32\MCRInstaller.exe MWArray API简介 MCR包含了文件MWArray.dll,该文件中的API承担了用户程序和MCR之间数据交换的任务,因此,每一个独立文件都需要包含对该文件的引用,否则程序就不能使用MATLAB中的函数,为使用该文件,需要先使用上一步找到的安装MCR,我安装在D:\Program Files\MATLAB\MATLAB Compiler Runtime目录,然后该dll该文件在安装完MCR后位于: D:\Program Files\MATLAB\MATLAB Compiler Runtime\v715\toolbox\dotnetbuilder\bin\win32\v2.0,名字是MWArray.dll, 另外,该文件中有两个重要的命名空间,MathWorks.MATLAB.NET.Arrays和MathWorks.MATLAB.NET.Utility,Arrays命名空间下的类提供从其他任何兼容CLS(Common Language Specification) 语言访问MATLAB中数组的功能,这些类支持数组格式化、类型的特定索引和错误处理的功能。而Utility命名空间下的类提供了对MWArray类架构和MATLAB公共运行时的托管API的通用支持。 实施 为了将MATLAB中的程序作为组件提供给其他.NET程序使用,需要做两方面的工作: 1.将M文件打包为与.NET兼容的程序集 2.在外部程序中添加对程序集的引用 MATLAB端的工作 以我这个功能为例。打开MatLab,然后File->New->Deployment Project.然后在Type里选择.net Assembly,Ok即可。我选择的工程名是test,注意,这个test就会生成一个名为test.dll的程序集,所以起个好名字,,我没起好。如图1 然后在图2的视图中定位到你选择的工程目录,并依次点击。 添加一个test.m的文件。图3 这个名称随意。但是要注意,这个名字必须和你里面将要写的那个函数名称一致,这个很好理解。好了以后,双击test.m就会打开。输入 function m=test(n) m=magic(n); end 创建一个test方法。该方法在C#调用的时候要用到,接受一个参数,然后输出一个幻方。保存关闭m文件,然后回到上图,双击test.prj也就是工程文件。 来到图4 点击Add Class,就是向test.dll的程序集中添加类。我添加了一个类叫做Func1.这个类用来在C#中创建一个对象,然后调用test(n)这个函数,然后出现Add Files,点击打开浏览框,把刚才写好的test.m选上。就可以了 最后看窗口右上角,,点击那个build图标就ok,等个几分钟,就会在目录里生成test.dll文件了。图5 VS端的代码 现在,打开vs,创建C#工程,添加两个引用,分别是MWArray.dll和test.dll,添加引用的方法就是在解决方案资源管理器中,的引用上,点击,然后添加引用->浏览->找到这两个dll,为了方便,我一般会把这两个dll复制到我的C#工程目录里,然后添加引用,如前面所说,MWArray.dll里有两个命名空间,我们使用using语句都添加上。也包括生成的test.dll using test; using MathWorks.MATLAB.NET.Arrays; using MathWorks.MATLAB.NET.Utility; 然后设计C#界面,很简单的拖一个界面,图6 然后单击事件代码: Func1 f = new Func1(); MWNumericArray na = null; MWArray[] ansArray = null; int i = Convert.ToInt16(textBox1.Text); ansArray = f.test(1, i); na = (MWNumericArray)ansArray[0];//只有一个数组返回 MessageBox.Show(na.ToString()); 看到test有两个参数,第一个参数是指返回结果书,我们只需要一个就好了。所以为1,后面那个是阶数了。运行结果如图7: ...

2012-07-02 · 1 min · bystander