稳扎稳打Silverlight(22) - 2.0通信之调用WCF服务, 对传输信息做加密

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介:
[索引页]
[源码下载]


稳扎稳打Silverlight(22) - 2.0通信之调用WCF服务, 对传输信息做加密


作者: webabcd


介绍
Silverlight 2.0 调用 WCF 服务,对客户端与服务端传输的消息做加密    
    在 Visual Studio 2008 中使用"添加服务引用"会自动生成代理类。只支持BasicHttpBinding


在线DEMO
http://webabcd.blog.51cto.com/1787395/342779


示例
clientaccesspolicy.xml
<? xml  version ="1.0"  encoding ="utf-8"  ?> 
< access-policy > 
         < cross-domain-access > 
                 < policy > 
                         < allow-from  http-request-headers ="*" > 
                                 < domain  uri ="*"  /> 
                         </ allow-from > 
                         < grant-to > 
                                 < resource  path ="/"  include-subpaths ="true"  /> 
                         </ grant-to > 
                 </ policy > 
         </ cross-domain-access > 
</ access-policy > 
<!--  
System.Net 命名空间 和 System.Net.Sockets 命名空间的跨域调用,需要在目标域的根目录下配置策略文件 
Image 控件 和 MediaElement 控件所访问的跨域地址,不受策略文件的限制 
HTTP 调用 仅支持 GET 和 POST ,只有 200(确定) 和 404(未找到) 状态代码可用 
同域:同一子域、协议和端口。不符合以上任一条件则为跨域 
Silverlight 与 HTTP/HTTPS 的所有通信均为异步 

关于策略文件详见文档 
-->
 
 
1、调用 WCF 服务
WCFService.cs(WCF 服务)
InBlock.gif using System; 
InBlock.gif using System.Linq; 
InBlock.gif using System.Runtime.Serialization; 
InBlock.gif using System.ServiceModel; 
InBlock.gif using System.ServiceModel.Activation; 
InBlock.gif using System.Collections.Generic; 
InBlock.gif using System.Text; 
InBlock.gif 
InBlock.gif using System.Security.Cryptography; 
InBlock.gif using System.IO; 
InBlock.gif 
/// <summary> 
/// 提供 WCF 服务的类 
/// </summary> 
InBlock.gif[ServiceContract] 
InBlock.gif[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)] 
InBlock.gif public  class WCFService 
InBlock.gif
InBlock.gif         /// <summary> 
InBlock.gif         /// 返回指定的 User 对象(用于演示 Silverlight 调用 WCF 服务) 
InBlock.gif         /// </summary> 
InBlock.gif         /// <param name="name">名字</param> 
InBlock.gif         /// <returns></returns> 
InBlock.gif        [OperationContract] 
InBlock.gif         public User GetUser( string name) 
InBlock.gif        { 
InBlock.gif                 return  new User { Name = name, DayOfBirth =  new DateTime(1980, 2, 14) }; 
InBlock.gif        } 
InBlock.gif}
 
WCF.xaml
<UserControl x:Class="Silverlight20.Communication.WCF" 
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"    
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 
        <StackPanel HorizontalAlignment="Left" Margin="5"> 
         
                <TextBlock x:Name="lblMsg" /> 
         
        </StackPanel> 
</UserControl>
 
WCF.xaml.cs
InBlock.gif using System; 
InBlock.gif using System.Collections.Generic; 
InBlock.gif using System.Linq; 
InBlock.gif using System.Net; 
InBlock.gif using System.Windows; 
InBlock.gif using System.Windows.Controls; 
InBlock.gif using System.Windows.Documents; 
InBlock.gif using System.Windows.Input; 
InBlock.gif using System.Windows.Media; 
InBlock.gif using System.Windows.Media.Animation; 
InBlock.gif using System.Windows.Shapes; 
InBlock.gif 
InBlock.gif using Silverlight20.WCFServiceReference; 
InBlock.gif using System.Threading; 
InBlock.gif using System.ServiceModel; 
InBlock.gif 
InBlock.gif namespace Silverlight20.Communication 
InBlock.gif
InBlock.gif         public partial  class WCF : UserControl 
InBlock.gif        { 
InBlock.gif                SynchronizationContext _syncContext; 
InBlock.gif 
InBlock.gif                 /// <summary> 
InBlock.gif                 /// 演示 Silverlight 调用 WCF 服务 
InBlock.gif                 /// </summary> 
InBlock.gif                 public WCF() 
InBlock.gif                { 
InBlock.gif                        InitializeComponent(); 
InBlock.gif 
InBlock.gif                         // 代理的配置信息在配置文件中,UI线程上的异步调用 
InBlock.gif                        Demo(); 
InBlock.gif 
InBlock.gif                         // 代理的配置信息在程序中指定,UI线程上的异步调用 
InBlock.gif                        Demo2(); 
InBlock.gif 
InBlock.gif                         // 后台线程(非UI线程)上的异步调用) 
InBlock.gif                        Demo3(); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                 void Demo() 
InBlock.gif                { 
InBlock.gif                         /*                            
InBlock.gif                         * 服务名Client - 系统自动生成的代理类 
InBlock.gif                         *         方法名Completed - 调用指定的方法完成后所触发的事件 
InBlock.gif                         *         方法名Async(参数1, 参数2 , object 用户标识) - 异步调用指定的方法 
InBlock.gif                         *         Abort() - 取消调用 
InBlock.gif                         */
 
InBlock.gif 
InBlock.gif                        WCFServiceClient client =  new WCFServiceClient(); 
InBlock.gif                        client.GetUserCompleted +=  new EventHandler<GetUserCompletedEventArgs>(client_GetUserCompleted); 
InBlock.gif                        client.GetUserAsync( "webabcd"); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                 void Demo2() 
InBlock.gif                { 
InBlock.gif                         /* 
InBlock.gif                         * 服务名Client - 其构造函数可以动态地指定代理的配置信息(Silverlight 2.0 调用 WCF 只支持 BasicHttpBinding) 
InBlock.gif                         */
 
InBlock.gif 
InBlock.gif                        WCFServiceClient client =  new WCFServiceClient( new BasicHttpBinding(),  new EndpointAddress( "http://localhost:3036/WCFService.svc")); 
InBlock.gif                        client.GetUserCompleted += new EventHandler<GetUserCompletedEventArgs>(client_GetUserCompleted); 
InBlock.gif                        client.GetUserAsync("webabcd2"); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                void client_GetUserCompleted(object sender, GetUserCompletedEventArgs e) 
InBlock.gif                { 
InBlock.gif                        /* 
InBlock.gif                         * 方法名CompletedEventArgs.Error - 该异步操作期间是否发生了错误 
InBlock.gif                         * 方法名CompletedEventArgs.Result - 异步操作返回的结果。本例为 User 类型 
InBlock.gif                         * 方法名CompletedEventArgs.UserState - 用户标识 
InBlock.gif                         */
 
InBlock.gif 
InBlock.gif                        if (e.Error != null
InBlock.gif                        { 
InBlock.gif                                lblMsg.Text += e.Error.ToString() + "\r\n"
InBlock.gif                                return
InBlock.gif                        } 
InBlock.gif 
InBlock.gif                        if (e.Cancelled != true
InBlock.gif                        { 
InBlock.gif                                OutputResult(e.Result); 
InBlock.gif                        } 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                void Demo3() 
InBlock.gif                { 
InBlock.gif                        // UI 线程 
InBlock.gif                        _syncContext = SynchronizationContext.Current; 
InBlock.gif 
InBlock.gif                        /* 
InBlock.gif                         * ChannelFactory<T>.CreateChannel() - 创建 T 类型的信道 
InBlock.gif                         * 服务名.Begin方法名() - 后台线程上异步调用指定方法(最后一个参数为 代理对象) 
InBlock.gif                         */
 
InBlock.gif 
InBlock.gif                        WCFService client = new ChannelFactory<WCFService>(new BasicHttpBinding(), new EndpointAddress("http://localhost:3036/WCFService.svc")).CreateChannel(); 
InBlock.gif                        client.BeginGetUser("webabcd3"new AsyncCallback(ResponseCallback), client); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                private void ResponseCallback(IAsyncResult result) 
InBlock.gif                { 
InBlock.gif                        WCFService client = result.AsyncState as WCFService; 
InBlock.gif 
InBlock.gif                        // 服务名.End方法名() - 获取在后台线程(非UI线程)上异步调用的结果 
InBlock.gif                        User user = client.EndGetUser(result); 
InBlock.gif 
InBlock.gif                        // 调用 UI 线程 
InBlock.gif                        _syncContext.Post(GetResponse, user); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                private void GetResponse(object state) 
InBlock.gif                { 
InBlock.gif                        OutputResult(state as User); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif 
InBlock.gif                /// <summary> 
InBlock.gif                /// 输出异步调用 WCF 服务的方法后返回的结果 
InBlock.gif                /// </summary> 
InBlock.gif                /// <param name="user"></param> 
InBlock.gif                void OutputResult(User user) 
InBlock.gif                { 
InBlock.gif                        lblMsg.Text += string.Format("姓名:{0};生日:{1}\r\n"
InBlock.gif                                user.Name, 
InBlock.gif                                user.DayOfBirth.ToString("yyyy-MM-dd")); 
InBlock.gif                } 
InBlock.gif        } 
InBlock.gif}
 
 
2、对客户端与服务端传输的消息做加密
WCFService.cs(WCF 服务)
InBlock.gif using System; 
InBlock.gif using System.Linq; 
InBlock.gif using System.Runtime.Serialization; 
InBlock.gif using System.ServiceModel; 
InBlock.gif using System.ServiceModel.Activation; 
InBlock.gif using System.Collections.Generic; 
InBlock.gif using System.Text; 
InBlock.gif 
InBlock.gif using System.Security.Cryptography; 
InBlock.gif using System.IO; 
InBlock.gif 
/// <summary> 
/// 提供 WCF 服务的类 
/// </summary> 
InBlock.gif[ServiceContract] 
InBlock.gif[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)] 
InBlock.gif public  class WCFService 
InBlock.gif
InBlock.gif         /// <summary> 
InBlock.gif         /// 返回指定的 User 对象(用于演示传输信息的加密/解密) 
InBlock.gif         /// </summary> 
InBlock.gif         /// <param name="name"></param> 
InBlock.gif         /// <returns></returns> 
InBlock.gif        [OperationContract] 
InBlock.gif         public User GetUserByCryptography( string name) 
InBlock.gif        { 
InBlock.gif                 return  new User { Name = Decrypt(name), DayOfBirth =  new DateTime(1980, 2, 14) }; 
InBlock.gif        } 
InBlock.gif 
InBlock.gif         /// <summary> 
InBlock.gif         /// 解密数据 
InBlock.gif         /// </summary> 
InBlock.gif         /// <param name="input">加密后的字符串</param> 
InBlock.gif         /// <returns>加密前的字符串</returns> 
InBlock.gif         public  string Decrypt( string input) 
InBlock.gif        { 
InBlock.gif                 // 盐值(与加密时设置的值一致) 
InBlock.gif                 string saltValue =  "saltValue"
InBlock.gif                 // 密码值(与加密时设置的值一致) 
InBlock.gif                 string pwdValue =  "pwdValue"
InBlock.gif 
InBlock.gif                 byte[] encryptBytes = Convert.FromBase64String(input); 
InBlock.gif                 byte[] salt = Encoding.UTF8.GetBytes(saltValue); 
InBlock.gif 
InBlock.gif                AesManaged aes =  new AesManaged(); 
InBlock.gif 
InBlock.gif                Rfc2898DeriveBytes rfc =  new Rfc2898DeriveBytes(pwdValue, salt); 
InBlock.gif 
InBlock.gif                aes.BlockSize = aes.LegalBlockSizes[0].MaxSize; 
InBlock.gif                aes.KeySize = aes.LegalKeySizes[0].MaxSize; 
InBlock.gif                aes.Key = rfc.GetBytes(aes.KeySize / 8); 
InBlock.gif                aes.IV = rfc.GetBytes(aes.BlockSize / 8); 
InBlock.gif 
InBlock.gif                 // 用当前的 Key 属性和初始化向量 IV 创建对称解密器对象 
InBlock.gif                ICryptoTransform decryptTransform = aes.CreateDecryptor(); 
InBlock.gif 
InBlock.gif                 // 解密后的输出流 
InBlock.gif                MemoryStream decryptStream =  new MemoryStream(); 
InBlock.gif 
InBlock.gif                 // 将解密后的目标流(decryptStream)与解密转换(decryptTransform)相连接 
InBlock.gif                CryptoStream decryptor =  new CryptoStream(decryptStream, decryptTransform, CryptoStreamMode.Write); 
InBlock.gif 
InBlock.gif                 // 将一个字节序列写入当前 CryptoStream (完成解密的过程) 
InBlock.gif                decryptor.Write(encryptBytes, 0, encryptBytes.Length); 
InBlock.gif                decryptor.Close(); 
InBlock.gif 
InBlock.gif                 // 将解密后所得到的流转换为字符串 
InBlock.gif                 byte[] decryptBytes = decryptStream.ToArray(); 
InBlock.gif                 string decryptedString = UTF8Encoding.UTF8.GetString(decryptBytes, 0, decryptBytes.Length); 
InBlock.gif 
InBlock.gif                 return decryptedString; 
InBlock.gif        } 
InBlock.gif
 
Cryptography.xaml
<UserControl x:Class="Silverlight20.Communication.Cryptography" 
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"    
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> 
        <StackPanel HorizontalAlignment="Left" Margin="5"> 
         
                <TextBlock x:Name="lblMsg" /> 
         
        </StackPanel> 
</UserControl>
 
Cryptography.xaml.cs
InBlock.gif using System; 
InBlock.gif using System.Collections.Generic; 
InBlock.gif using System.Linq; 
InBlock.gif using System.Net; 
InBlock.gif using System.Windows; 
InBlock.gif using System.Windows.Controls; 
InBlock.gif using System.Windows.Documents; 
InBlock.gif using System.Windows.Input; 
InBlock.gif using System.Windows.Media; 
InBlock.gif using System.Windows.Media.Animation; 
InBlock.gif using System.Windows.Shapes; 
InBlock.gif 
InBlock.gif using Silverlight20.WCFServiceReference; 
InBlock.gif using System.Text; 
InBlock.gif using System.Security.Cryptography; 
InBlock.gif using System.IO; 
InBlock.gif 
InBlock.gif namespace Silverlight20.Communication 
InBlock.gif
InBlock.gif         public partial  class Cryptography : UserControl 
InBlock.gif        { 
InBlock.gif                 public Cryptography() 
InBlock.gif                { 
InBlock.gif                        InitializeComponent(); 
InBlock.gif 
InBlock.gif                        Demo(); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                 void Demo() 
InBlock.gif                { 
InBlock.gif                        WCFServiceClient client =  new WCFServiceClient(); 
InBlock.gif 
InBlock.gif                        client.GetUserByCryptographyCompleted+= new EventHandler<GetUserByCryptographyCompletedEventArgs>(client_GetUserByCryptographyCompleted); 
InBlock.gif                        client.GetUserByCryptographyAsync(Encrypt( "webabcd")); 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                 void client_GetUserByCryptographyCompleted( object sender, GetUserByCryptographyCompletedEventArgs e) 
InBlock.gif                { 
InBlock.gif                         if (e.Error !=  null
InBlock.gif                        { 
InBlock.gif                                lblMsg.Text += e.Error.ToString() +  "\r\n"
InBlock.gif                                 return
InBlock.gif                        } 
InBlock.gif 
InBlock.gif                         if (e.Cancelled !=  true
InBlock.gif                        { 
InBlock.gif                                lblMsg.Text +=  string.Format( "姓名:{0};生日:{1}\r\n"
InBlock.gif                                        e.Result.Name, 
InBlock.gif                                        e.Result.DayOfBirth.ToString( "yyyy-MM-dd")); 
InBlock.gif                        } 
InBlock.gif                } 
InBlock.gif 
InBlock.gif                 /// <summary> 
InBlock.gif                 /// 加密数据 
InBlock.gif                 /// </summary> 
InBlock.gif                 /// <param name="input">加密前的字符串</param> 
InBlock.gif                 /// <returns>加密后的字符串</returns> 
InBlock.gif                 private  string Encrypt( string input) 
InBlock.gif                { 
InBlock.gif                         // 盐值 
InBlock.gif                         string saltValue =  "saltValue"
InBlock.gif                         // 密码值 
InBlock.gif                         string pwdValue =  "pwdValue"
InBlock.gif 
InBlock.gif                         byte[] data = UTF8Encoding.UTF8.GetBytes(input); 
InBlock.gif                         byte[] salt = UTF8Encoding.UTF8.GetBytes(saltValue); 
InBlock.gif 
InBlock.gif                         // AesManaged - 高级加密标准(AES) 对称算法的管理类 
InBlock.gif                        AesManaged aes =  new AesManaged(); 
InBlock.gif 
InBlock.gif                         // Rfc2898DeriveBytes - 通过使用基于 HMACSHA1 的伪随机数生成器,实现基于密码的密钥派生功能 (PBKDF2 - 一种基于密码的密钥派生函数) 
InBlock.gif                         // 通过 密码 和 salt 派生密钥 
InBlock.gif                        Rfc2898DeriveBytes rfc =  new Rfc2898DeriveBytes(pwdValue, salt); 
InBlock.gif 
InBlock.gif                         /* 
InBlock.gif                         * AesManaged.BlockSize - 加密操作的块大小(单位:bit) 
InBlock.gif                         * AesManaged.LegalBlockSizes - 对称算法支持的块大小(单位:bit) 
InBlock.gif                         * AesManaged.KeySize - 对称算法的密钥大小(单位:bit) 
InBlock.gif                         * AesManaged.LegalKeySizes - 对称算法支持的密钥大小(单位:bit) 
InBlock.gif                         * AesManaged.Key - 对称算法的密钥 
InBlock.gif                         * AesManaged.IV - 对称算法的密钥大小 
InBlock.gif                         * Rfc2898DeriveBytes.GetBytes(int 需要生成的伪随机密钥字节数) - 生成密钥 
InBlock.gif                         */
 
InBlock.gif 
InBlock.gif                        aes.BlockSize = aes.LegalBlockSizes[0].MaxSize; 
InBlock.gif                        aes.KeySize = aes.LegalKeySizes[0].MaxSize; 
InBlock.gif                        aes.Key = rfc.GetBytes(aes.KeySize / 8); 
InBlock.gif                        aes.IV = rfc.GetBytes(aes.BlockSize / 8); 
InBlock.gif 
InBlock.gif                         // 用当前的 Key 属性和初始化向量 IV 创建对称加密器对象 
InBlock.gif                        ICryptoTransform encryptTransform = aes.CreateEncryptor(); 
InBlock.gif 
InBlock.gif                         // 加密后的输出流 
InBlock.gif                        MemoryStream encryptStream =  new MemoryStream(); 
InBlock.gif 
InBlock.gif                         // 将加密后的目标流(encryptStream)与加密转换(encryptTransform)相连接 
InBlock.gif                        CryptoStream encryptor =  new CryptoStream(encryptStream, encryptTransform, CryptoStreamMode.Write); 
InBlock.gif 
InBlock.gif                         // 将一个字节序列写入当前 CryptoStream (完成加密的过程) 
InBlock.gif                        encryptor.Write(data, 0, data.Length); 
InBlock.gif                        encryptor.Close(); 
InBlock.gif 
InBlock.gif                         // 将加密后所得到的流转换成字节数组,再用Base64编码将其转换为字符串 
InBlock.gif                         string encryptedString = Convert.ToBase64String(encryptStream.ToArray()); 
InBlock.gif 
InBlock.gif                         return encryptedString; 
InBlock.gif                }                 
InBlock.gif        } 
InBlock.gif}
 
 
      本文转自webabcd 51CTO博客,原文链接:http://blog.51cto.com/webabcd/343130,如需转载请自行联系原作者
相关文章
|
前端开发
WCF更新服务引用报错的原因之一
WCF更新服务引用报错的原因之一
|
C# 数据安全/隐私保护
c#如何创建WCF服务到发布(SqlServer版已经验证)
c#如何创建WCF服务到发布(SqlServer版已经验证)
62 0
|
安全 数据库连接 数据库
WCF服务创建到发布(SqlServer版)
在本示例开始之前,让我们先来了解一下什么是wcf? wcf有哪些特点? wcf是一个面向服务编程的综合分层架构。该架构的项层为服务模型层。 使用户用最少的时间和精力建立自己的软件产品和外界通信的模型。它使得开发者能够建立一个跨平台的安全、可信赖、事务性的解决方案。且能与已有系统兼容写作。 简单概括就是:一组数据通信的应用程序开发接口。
93 0
|
XML 网络协议 IDE
WCF基础教程(三)——WCF通信过程及配置文件解析
WCF基础教程(三)——WCF通信过程及配置文件解析
348 0
WCF基础教程(三)——WCF通信过程及配置文件解析
|
C++
WCF基础教程(二)——解析iis8和iis8.5+VS2013发布wcf服务问题
WCF基础教程(二)——解析iis8和iis8.5+VS2013发布wcf服务问题
125 0
WCF基础教程(二)——解析iis8和iis8.5+VS2013发布wcf服务问题
|
JavaScript
|
3天前
|
SQL 安全 网络安全
网络安全与信息安全:构建防线的三大支柱在数字时代,网络安全和信息安全成为了我们不可忽视的重要议题。本文将深入探讨网络安全漏洞、加密技术以及安全意识这三大支柱,帮助您建立更全面的安全防护体系。
本文旨在分享有关网络安全漏洞、加密技术和安全意识的知识。首先,我们将介绍常见的网络安全漏洞及其形成原因;接着,我们将探讨几种主要的加密技术及其应用;最后,我们将强调提高安全意识的重要性并提供实用的建议。通过这些内容,读者可以更好地理解如何在日常生活和工作中保护自己的信息安全。
22 9
|
1天前
|
SQL 安全 算法
网络安全的盾牌与矛:加密技术与安全意识的双重防线
【9月更文挑战第16天】在数字世界的海洋中,网络安全的灯塔照亮着信息流通的每一个角落。本文将深入探讨网络安全的核心——加密技术,以及构筑坚不可摧防线的另一块基石——安全意识。我们将从密码学的基础知识出发,逐步揭开加密技术的神秘面纱,并分析当前最常见的网络攻击手段,揭示安全意识的重要性。通过理论与实践的结合,旨在为读者提供一套实用的网络安全知识体系,以应对日益复杂的网络威胁。