C# 封送和远程编程介绍


.NET学习资料

.NET学习资料

.NET学习资料


在 C# 编程领域中,封送(Marshaling)和远程编程(Remote Programming)是两个极为重要的概念,它们为开发者提供了与不同环境、不同进程或不同机器上的代码进行交互的能力,极大地拓展了应用程序的功能和适用范围。

一、C# 封送

(一)定义与作用

封送是指在托管代码(如 C# 编写的代码,由公共语言运行时 CLR 管理)和非托管代码(如 C、C++ 编写的代码,没有 CLR 参与)之间进行数据转换和传递的过程。不同编程语言和环境在调用约定、布局约定、基本数据类型大小、对象创建与销毁约定以及设计准则等方面存在差异。例如,C# 中的char类型和 C 中的char类型在大小和编码方式上可能不同。因此,需要封送来解决这些差异,确保数据能够在不同环境之间正确传递和理解 。它就像是一座桥梁,连接着托管世界和非托管世界,使得两者能够进行有效的通信。

(二)常见的封送类型及处理方式

blittable 类型

像byte、short、int、long等及其无符号对应类型,这些类型在托管和非托管代码之间传递时,不需要特殊的封送处理,因为它们在不同环境中的内存布局和表示方式是一致的,可以直接进行数据传输。

非 blittable 类型

字符串类型:在 C# 与非托管代码交互时,若要将string封送到非托管函数,可以使用Marshal.StringToHGlobalAnsi、Marshal.StringToHGlobalAuto、Marshal.StringToHGlobalUni等方法 ,操作完成后需调用Marshal.FreeHGlobal释放内存;从非托管函数中取出string则可使用Marshal.PtrToStringAnsi、Marshal.PtrToStringAuto、Marshal.PtrToStringUni。例如,当调用一个非托管的 C 函数,该函数接收一个 ANSI 编码的字符串参数时,就可以使用Marshal.StringToHGlobalAnsi将 C# 中的string转换为符合要求的非托管内存中的字符串形式。

数组类型:对于byte[](或其他基础类型的数组),可以使用Marshal.Copy方法实现从byte[]拷贝封送到非托管函数,或从非托管函数拷贝封送到byte[] 。另外,还可以采用固定内存直接传送byte[]的原始地址的方式(这种方式省去了申请内存和拷贝的开销,速度更快),具体是使用GCHandle并指定GCHandleType.Pinned类型。假设需要将一个字节数组传递给非托管代码进行快速处理,固定内存地址的方式就能提高数据传输效率。
委托类型:从delegate封送到非托管函数可使用Marshal.GetFunctionPointerForDelegate,从非托管函数中取出则使用Marshal.GetDelegateForFunctionPointer 。在封送到非托管函数时,要确保垃圾回收器(GC)不会回收委托,通常需要保持引用但不需要将其固定。比如在实现回调函数机制时,就可能涉及委托类型的封送。

结构体类型:在 C# 中按相同顺序和对应数据类型声明一个同样的struct(只能使用基础类型和固定长度的数组),并且标记StructLayout的LayoutKind.Sequential属性,然后利用Marshal.StructureToPtr和Marshal.PtrToStructure进行封送。当与非托管代码进行结构体数据交互时,这种方式能保证数据结构的正确传递。

(三)自动封送与手动封送

自动封送:当 C# 声明的extern函数返回类型和参数类型中有托管类型(如string、byte[]、委托等)时,CLR 会自动进行封送处理,这大大节省了开发者编写封送代码的时间和精力。例如,在调用一个非托管的 DLL 函数,该函数的参数是一个字符串时,C# 代码中无需额外编写复杂的封送代码,CLR 会自动完成字符串类型的封送转换。
手动封送:在某些特殊情况下,自动封送无法满足需求,就需要手动进行封送。比如当string的编码是 UTF - 8 之类的非 ANSI 非 UTF - 16 编码时,必须手动进行封送并同时转换编码;又或者委托转换成函数指针的操作比较耗时,如果有频繁的对同一委托进行封送调用,预存转换后的结果能够显著提升性能,此时也需要手动进行封送处理。

二、C# 远程编程

(一).NET Remoting 框架概述

.NET Remoting 是一种分布式处理框架,它允许对象通过应用程序域与另一对象进行交互 ,可以看作是 DCOM 的一种升级,并且很好地融合到了.NET 平台下。其核心优势在于提供了一种灵活且抽象的进程间通信方式,使得开发者无需关注具体的客户端或服务器应用程序域,也无需关心特定的通信协议,就能实现不同应用程序域甚至不同机器上的对象之间的通信。

(二)主要组件

可远程处理的对象(Remoteable Object):这是需要在远程环境中被访问和调用的对象。它必须继承自MarshalByRefObject类,以便能够跨越应用程序域进行通信。例如,一个实现了业务逻辑的服务类,希望被远程客户端调用其方法,就可以让这个类继承MarshalByRefObject。

远程监听应用程序(Remote Listener Application):负责监听对远程对象的请求。它创建并注册通信通道,同时注册远程对象,使其能够被远程客户端发现和访问。比如在服务器端,通过创建一个 TCP 或 HTTP 通道,并将远程对象注册到该通道上,等待客户端的连接请求。

远程客户端应用程序(Remote Client Application):向远程对象发出请求。客户端通过 Remoting 框架,访问通道以获取服务器端对象的引用,然后通过该引用调用远程对象的方法。在客户端代码中,首先创建一个与服务器端对应的通道,然后使用Activator.GetObject等方法获取远程对象的引用,进而调用其公开的方法。

(三)通信原理与通道

通信原理:在 Remoting 中,客户端获取服务器端对象时,得到的并不是实际的服务端对象,而是它的引用,这保证了客户端和服务器端对象的松散耦合,同时优化了通信性能。客户端通过通道发送请求消息,服务器端通过相应的通道接收请求并处理,然后将结果通过通道返回给客户端 。整个过程就像是客户端和服务器端通过一条 “通信管道” 进行交互,而这条 “管道” 就是通道。

通道类型:主要支持 TCP(传输控制协议)和 HTTP(超文本传输协议)通道。使用 TCP 通道时,数据以二进制形式传输,效率较高,适合在内部网络环境中使用;而 HTTP 通道则基于 HTTP 协议,数据以 SOAP(简单对象访问协议)格式传输,具有更好的跨平台和跨网络环境的兼容性,适合在 Internet 等复杂网络环境中使用。

(四)代码示例

定义远程对象
using System;
namespace RemoteObject
{public class MyRemoteObject : MarshalByRefObject{public MyRemoteObject(){}public int Add(int a, int b){return a + b;}}
}
服务器端代码
using System;
using System.Runtime.Remoting.Channels;
using System.Runtime.Remoting.Channels.Tcp;
using RemoteObject;namespace RemoteServer
{class Program{static void Main(string[] args){// 创建TCP服务器通道,端口号为8080TcpServerChannel channel = new TcpServerChannel(8080);// 注册通道ChannelServices.RegisterChannel(channel, false);// 注册远程对象,对象URI为"MyRemoteObject",激活模式为单例RemotingConfiguration.RegisterWellKnownServiceType(typeof(MyRemoteObject), "MyRemoteObject", WellKnownObjectMode.Singleton);Console.WriteLine("服务器已启动,等待客户端连接...");Console.Read();}}
}
客户端代码
using System;
using System.Runtime.Remoting.Channels;
using System.Runtime.Remoting.Channels.Tcp;
using RemoteObject;namespace RemoteClient
{class Program{static void Main(string[] args){// 创建TCP客户端通道TcpClientChannel channel = new TcpClientChannel();// 注册通道ChannelServices.RegisterChannel(channel, false);// 获取远程对象引用MyRemoteObject remoteObject = (MyRemoteObject)Activator.GetObject(typeof(MyRemoteObject), "tcp://localhost:8080/MyRemoteObject");// 调用远程对象的方法int result = remoteObject.Add(3, 5);Console.WriteLine("调用远程方法结果: " + result);Console.Read();}}
}

通过以上对 C# 封送和远程编程的介绍,希望能帮助你深入理解这两个重要概念及其在实际开发中的应用。如果你对代码实现细节、原理有进一步的疑问,或者想要了解更多相关应用场景,欢迎随时交流。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/15139.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot接入Deep Seek的API

1,首先进入deepseek的官网:DeepSeek | 深度求索,单击右上角的API开放平台。 2,单击API keys,创建一个API,创建完成务必复制!!不然关掉之后会看不看api key!!&…

【C++学习篇】C++11第二期学习

目录 1. 可变参数模板 1.1 基本语法及原理 1.2 包扩展 1.3empalce系列接⼝ 2. lamba 2.1 lambda的语法表达式 2.2 捕捉列表 2.3 lamba的原理 1. 可变参数模板 1.1 基本语法及原理 1. C11⽀持可变参数模板,也就是说⽀持可变数量参数的函数模板和类模板&…

开放式TCP/IP通信

一、1200和1200之间的开放式TCP/IP通讯 第一步:组态1214CPU,勾选时钟存储器 第二步:防护与安全里面连接机制勾选允许PUT/GET访问 第三步:添加PLC 第四步:点击网络试图,选中网口,把两个PLC连接起…

迁移学习 Transfer Learning

迁移学习(Transfer Learning)是什么? 迁移学习是一种机器学习方法,它的核心思想是利用已有模型的知识来帮助新的任务或数据集进行学习,从而减少训练数据的需求、加快训练速度,并提升模型性能。 &#x1f…

爬虫技巧汇总

一、UA大列表 USER_AGENT_LIST 是一个包含多个用户代理字符串的列表,用于模拟不同浏览器和设备的请求。以下是一些常见的用户代理字符串: USER_AGENT_LIST [Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.1; Trident/4.0; Hot Lingo 2.0),Mozilla…

我们来学人工智能 -- 将Ollama已下载的模型从C盘迁出

题记 未配置OLLAMA_MODELS系统变量导致模型下载到了C盘 迁移步骤 退出ollama 配置OLLAMA_MODELS系统变量 OLLAMA_MODELS:D:\ollama\models 直接将C盘下的models目录剪切到指定目录 检查 cmd命令窗口退出重新打开

Redis 集群原理、主从复制和哨兵模式的详细讲解

引言:本文记录了博主在学习Redis的过程中的原理,了解为什么使用与怎么样使用 Redis 集群,在使用 Redis 集群时出现的主从复制和哨兵模式的相关知识。本文并不涉及Redis安装。 文章目录 一、简单介绍什么是 Redis二、为什么要使用 Redis 集群三…

Java数据结构 | TreeMap 和 TreeSet

TreeMap 和 TreeSet 1. 搜索树1.1 概念1.2 搜索树的查找、插入、删除思路及代码1.2.1 查找1.2.2 插入1.2.3 删除&#xff08;难点&#xff09; 1.3 二叉搜索树的性能分析 2. Map 和 Set2.1 Map 接口2.1.1 Map.Entry<K,V>2.1.2 Map的常用方法 2.2 Set 接口2.2.1 Set 的常用…

智能理解 PPT 内容,快速生成讲解视频

当我们想根据一版 PPT 制作出相对应的解锁视频时&#xff0c;从撰写解锁词&#xff0c;录制音频到剪辑视频&#xff0c;每一个环节都需要投入大量的时间和精力&#xff0c;本方案将依托于阿里云函数计算 FC 和百炼模型服务&#xff0c;实现从 PPT 到视频的全自动转换&#xff0…

小鹅通首页网页开发

一、小鹅通首页开发 二、代码&#xff1a; index.html: <!DOCTYPE html><html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title&…

离散型变量的 PSI-群体稳定性指标计算

文章目录 PSI-群体稳定性指标(离散型)单个指标计算所有指标计算 PSI-群体稳定性指标(离散型) 单个指标计算 代码 import pandas as pddf pd.read_csv(/Users/mengzhichao/Desktop/文件/图表/小微企业用电量数据.csv)X_train df.sample(n7000) X_test df.sample(n3000)计算单…

STM32G474--Whetstone程序移植(单精度)笔记

1 准备基本工程代码 参考这篇笔记从我的仓库中选择合适的基本工程&#xff0c;进行程序移植。这里我用的是stm32g474的基本工程。 使用git clone一个指定文件或者目录 2 移植程序 2.1 修改Whetstone.c 主要修改原本变量定义的类型&#xff0c;以及函数接口全部更换为单精度…

【电机控制器】STC8H1K芯片——低功耗

【电机控制器】STC8H1K芯片——低功耗 文章目录 [TOC](文章目录) 前言一、芯片手册说明二、IDLE模式三、PD模式四、PD模式唤醒五、实验验证1.接线2.视频&#xff08;待填&#xff09; 六、参考资料总结 前言 使用工具&#xff1a; 1.STC仿真器烧录器 提示&#xff1a;以下是本…

Neo4j图数据库学习(二)——SpringBoot整合Neo4j

一. 前言 本文介绍如何通过SpringBoot整合Neo4j的方式&#xff0c;对图数据库进行简单的操作。 Neo4j和SpringBoot的知识不再赘述。关于Neo4j的基础知识&#xff0c;有兴趣可以看看作者上一篇的文章&#xff1a;Neo4j图数据库学习(一)——初识CQL 二. 前置准备 新建SpringBo…

【后端开发】系统设计101——Devops,Git与CICD,云服务与云原生,Linux,安全性,案例研究(30张图详解)

【后端开发】系统设计101——Devops&#xff0c;Git与CICD&#xff0c;云服务与云原生&#xff0c;Linux&#xff0c;安全性&#xff0c;案例研究&#xff08;30张图详解&#xff09; 文章目录 1、DevopsDevOps与SRE与平台工程的区别是什么&#xff1f;什么是k8s&#xff08;Ku…

01_Machine Vision_LSI及傅立叶变换

outline 图像分解和线性时不变系统二维傅立叶变换图像采样 图像分解和线性时不变系统 图像数学表达 图像由基本的像素点组成&#xff0c;如果将每一个像素点看作一个脉冲&#xff0c;则每个像素点的值可以看作是脉冲的幅值&#xff0c;这样图像就可以看作是由一系列脉冲组成…

elasticsearch实战三 elasticsearch与mysql数据实时同步

一 介绍 elasticsearch数据不是一直不变的&#xff0c;需要与mysql、oracle等数据库的数据做同步。 本博客里涉及到的项目地址&#xff1a;https://www.aliyundrive.com/s/7bRWpTYsxWV 方案一&#xff1a; 同步调用&#xff0c;即操作mysql数据后&#xff0c;接着操作elastic…

智能化食品安全管理:AI视频监控在大型商场的技术方案

前言 在卖场中&#xff0c;尤其是熟食区&#xff0c;AI视频监控的应用对于食品安全至关重要。通过AI视频监控系统&#xff0c;卖场可以实时监测食品处理环节中的每一个细节&#xff0c;从员工的个人防护到清洁操作&#xff0c;再到区域管理&#xff0c;全面提升食品安全管理的…

分析模式应用――帐务模式02

Party 模式中的层次结构模型支持多种灵活的层次结构&#xff0c;但这里我们只要关心上下级的包含关系就可以了&#xff0c;参加结算的称为结算实体BalanceEntity&#xff0c; 不可再拆分的称为LeafEntity&#xff0c; 可以包含下级结算实体的称为CompositeEntity&#xff0c;因…

什么是网络安全

1) 什么是网络安全 作为程序员&#xff0c;主要是面向产品的安全的问题。比如sql注入&#xff0c;xss&#xff0c;csrf&#xff0c;cookie窃取等等&#xff0c;都值得我们去思考。保证网站运行正常&#xff0c;客户数据安全。 2) sql注入 简单的说&#xff0c;就是利用表单提…