RPC

 远程过程调用的详细信息:http://msdn.microsoft.com/en-us/library/windows/desktop/aa378651(v=vs.85).aspx

[
    uuid(7a98c250-6808-11cf-b73b-00aa00b677a7),
    version(1.0)
]
interface hello
{
    void HelloProc([in, string] unsigned char * pszString);
    void Shutdown(void);
}


 

[
    implicit_handle (handle_t hello_IfHandle)
] 
interface hello
{
}


copy好后hello.idl和hello.acf运行midl hello.idl生成hello.h、hello_c.c、hello_s.c,编写helloc.c

#include <stdlib.h>
#include <stdio.h>
#include <ctype.h>
#include "hello.h" 
#include "rpc.h"
#include <windows.h>

void main()
{
	RPC_STATUS status;
	unsigned char * pszUuid             = NULL;
	unsigned char * pszProtocolSequence = "ncacn_ip_tcp";
	unsigned char * pszNetworkAddress   = "127.0.0.1";
	unsigned char * pszEndpoint         = "12345";
	unsigned char * pszOptions          = NULL;
	unsigned char * pszStringBinding    = NULL;
	unsigned char * pszString           = "hello, world";
	unsigned long ulCode;

	status = RpcStringBindingCompose(pszUuid,
		pszProtocolSequence,
		pszNetworkAddress,
		pszEndpoint,
		pszOptions,
		&pszStringBinding);
	if (status) exit(status);

	status = RpcBindingFromStringBinding(pszStringBinding, &hello_IfHandle);

	if (status) exit(status);

	RpcTryExcept  
	{
		HelloProc(pszString);
		//Shutdown();
	}
	RpcExcept(1) 
	{
		ulCode = RpcExceptionCode();
		printf("Runtime reported exception 0x%lx = %ld\n", ulCode, ulCode);
	}
	RpcEndExcept

	status = RpcStringFree(&pszStringBinding); 

	if (status) exit(status);

	status = RpcBindingFree(&hello_IfHandle);

	if (status) exit(status);

	exit(0);
}

/******************************************************/
/*         MIDL allocate and free                     */
/******************************************************/

void __RPC_FAR * __RPC_USER midl_user_allocate(size_t len)
{
	return(malloc(len));
}

void __RPC_USER midl_user_free(void __RPC_FAR * ptr)
{
	free(ptr);
}

在同一个solution中new一个project,添加hellos.c

#include <stdlib.h>
#include <stdio.h>
#include <ctype.h>
#include "hello.h"
#include "rpc.h"
#include <windows.h>

void main()
{
	RPC_STATUS status;
	unsigned char * pszProtocolSequence = "ncacn_ip_tcp";
	unsigned char * pszSecurity         =  NULL; 
	unsigned char * pszEndpoint         = "12345";
	unsigned int    cMinCalls = 1;
	unsigned int    fDontWait = FALSE;

	status = RpcServerUseProtseqEp(pszProtocolSequence,
		2,
		pszEndpoint,
		pszSecurity); 

	if (status) exit(status);

	status = RpcServerRegisterIf(hello_v1_0_s_ifspec,  
		NULL,   
		NULL); 

	if (status) exit(status);

	status = RpcServerListen(cMinCalls,
		RPC_C_LISTEN_MAX_CALLS_DEFAULT,
		fDontWait);

	if (status) exit(status);
}

/******************************************************/
/*         MIDL allocate and free                     */
/******************************************************/

void __RPC_FAR * __RPC_USER midl_user_allocate(size_t len)
{
	return(malloc(len));
}

void __RPC_USER midl_user_free(void __RPC_FAR * ptr)
{
	free(ptr);
}


另建一hellop.c

#include <stdio.h>
#include <windows.h>

void Shutdown(void)
{
	RPC_STATUS status;

	status = RpcMgmtStopServerListening(NULL);

	if (status) 
	{
		exit(status);
	}

	status = RpcServerUnregisterIf(NULL, NULL, FALSE);

	if (status) 
	{
		exit(status);
	}
} 

void HelloProc(char * pszString)
{
	printf("%s\n", pszString);
}


在vs2008的client和server工程里分别修改properties->linker->input->addtional dependencies,输入RpcRT4.lib

在rpc运行的过程中,可能会受到windows的安全限制,修改注册表HKEY_LOCAL_MACHINE \ SOFTWARE\Policies\ Microsoft\Windows NT\RPC ,设置Restrict
RemoteClients
值为0,没有响应的项就添加

若将client/server中的一端拿到没有支持vs2008的c/c++运行库的vm中,设置properties->c/c++->code generation->runtime library为multi-threaded debug

 

07-11
### 三级标题:RPC 基本概念 远程过程调用(Remote Procedure Call,简称 RPC)是一种在分布式系统中广泛使用的通信机制,允许一个程序像调用本地函数一样调用另一个网络节点上的程序[^5]。RPC 的核心思想是隐藏底层的网络通信细节,使得开发者可以专注于业务逻辑的设计与实现。 在 RPC 模型中,通常包含客户端(Client)、服务端(Server)、以及中间的网络通信组件。客户端通过调用本地代理(Stub)来发起请求,该请求被序列化并通过网络传输到服务端。服务端接收到请求后进行反序列化,并执行对应的服务逻辑,最终将结果返回给客户端[^2]。 ### 三级标题:RPC 使用场景 RPC 主要应用于以下几种典型场景: 1. **微服务架构**:在现代的微服务系统中,各个服务之间需要频繁地进行通信。RPC 提供了一种高效、透明的跨服务调用方式,能够显著提升系统的整体性能和可维护性。 2. **高性能计算**:当系统需要处理大量的并发请求时,使用高效的 RPC 协议(如 gRPC 或 Dubbo)可以有效降低网络延迟并提高吞吐量。 3. **异构系统集成**:许多 RPC 框架支持多种编程语言和平台,这使得不同技术栈构建的服务可以在统一的接口下协同工作。 4. **分布式事务管理**:在复杂的分布式系统中,多个服务可能需要共同完成一个事务。RPC 可以作为这些服务之间的协调工具,确保数据的一致性和完整性。 ### 三级标题:RPC 框架实现 #### 1. 核心组件 一个完整的 RPC 框架通常包括以下几个关键组件: - **协议层**:定义了消息的格式和交互规则。常见的协议有 HTTP/REST、gRPC 和 Dubbo 协议等。 - **序列化机制**:负责将数据结构转换为字节流以便在网络上传输,常用的序列化方式包括 JSON、Protocol Buffers、Thrift 和 Hessian 等。 - **网络通信**:基于 TCP/IP 或 HTTP/2 实现可靠的数据传输。 - **服务注册与发现**:用于动态管理和查找可用的服务实例。 - **负载均衡与容错机制**:确保请求能够在多个服务实例之间合理分配,并在出现故障时提供自动恢复能力。 #### 2. 实现示例 以下是一个简化版的 RPC 调用流程的 Python 示例代码: ```python import socket import json # 定义服务端 class RpcServer: def __init__(self, host='localhost', port=8080): self.host = host self.port = port self.services = {} def register_service(self, name, func): self.services[name] = func def start(self): with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s: s.bind((self.host, self.port)) s.listen() print("Server is listening...") while True: conn, addr = s.accept() with conn: data = conn.recv(1024) if not data: break request = json.loads(data.decode()) service_name = request['service'] args = request['args'] result = self.services[service_name](*args) conn.sendall(json.dumps(result).encode()) # 定义客户端 class RpcClient: def __init__(self, host='localhost', port=8080): self.host = host self.port = port def call(self, service_name, *args): with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s: s.connect((self.host, self.port)) request = json.dumps({'service': service_name, 'args': args}) s.sendall(request.encode()) data = s.recv(1024) return json.loads(data.decode()) # 示例服务 def add(a, b): return a + b # 启动服务器并注册服务 server = RpcServer() server.register_service('add', add) # server.start() # 注释掉以避免阻塞主线程 # 客户端调用 client = RpcClient() result = client.call('add', 2, 3) print(f"Result: {result}") ``` #### 3. 高级特性与优化 为了提升 RPC 框架的性能和可靠性,通常会引入以下高级功能: - **异步调用**:支持非阻塞式的调用方式,提升系统的响应速度和资源利用率。 - **缓存机制**:对于重复的请求,可以通过缓存减少对后端服务的压力。 - **安全机制**:采用 SSL/TLS 加密通信,防止数据被窃取或篡改。 - **监控与日志**:记录详细的调用日志,便于故障排查和性能分析。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值