使用在python中使用grpc和共享内存,实现功能调用。

初衷:grpc可以很好的作为跨语言,跨机器的服务模式。但是对于大量数据的传输,进程之间还是共享内存要快一些,所以这里将grpc与共享内存混合使用。

功能实现:

        客户端:从摄像头不断的读取图片,并传输给服务端,并呈现从服务端返回的处理结果。

        服务端:读取客户端的请求,并从中读取共享内存中的图片,在图片上写入”ok“,并将结果传回给客户端。

创建文件:image_processing.proto

syntax = "proto3";package image_processing;import "google/protobuf/empty.proto";// 定义一个请求消息,包含共享内存的名称
message ImageRequest {string input_shm_name = 1;string output_shm_name = 2;
}// 定义一个响应消息,包含处理后的图像的共享内存地址
message ImageResponse {string output_shm_name = 1;
}// 定义清理共享内存的请求消息
message CleanupRequest {string output_shm_name = 1;
}// 定义服务接口
service ImageProcessingService {rpc ProcessImage (ImageRequest) returns (ImageResponse);rpc CleanupSharedMemory (CleanupRequest) returns (google.protobuf.Empty);
}

安装python库:

pip install grpcio grpcio-tools opencv-python-headless

使用grpcio-tools生成Python代码:

python -m grpc_tools.protoc -I. --python_out=. --grpc_python_out=. image_processing.proto

客户端

import grpc
import cv2
import numpy as np
from multiprocessing import shared_memoryimport image_processing_pb2
import image_processing_pb2_grpc
def main():# 捕获图像#img = capture_image()vc=cv2.VideoCapture(0)with grpc.insecure_channel('localhost:50051') as channel:stub = image_processing_pb2_grpc.ImageProcessingServiceStub(channel)while 1:res,img=vc.read()shm = shared_memory.SharedMemory(create=True, size=img.nbytes)shm_array = np.ndarray(img.shape, dtype=img.dtype, buffer=shm.buf)shm_array[:] = img[:]response = stub.ProcessImage(image_processing_pb2.ImageRequest(input_shm_name=shm.name, output_shm_name=""))# 读取处理后的图像output_shm = shared_memory.SharedMemory(name=response.output_shm_name)output_array = np.ndarray(img.shape, dtype=img.dtype, buffer=output_shm.buf)# 显示处理后的图像cv2.imshow("Processed Image", output_array)cv2.waitKey(1)cv2.destroyAllWindows()# 清理共享内存shm.close()shm.unlink()output_shm.close()output_shm.unlink()# 通知服务端清理共享内存with grpc.insecure_channel('localhost:50051') as channel:stub = image_processing_pb2_grpc.ImageProcessingServiceStub(channel)stub.CleanupSharedMemory(image_processing_pb2.CleanupRequest(output_shm_name=response.output_shm_name))if __name__ == '__main__':main()

服务端:

import grpc
from concurrent import futures
import cv2
import numpy as np
from multiprocessing import shared_memory
import time
import threadingimport image_processing_pb2
import image_processing_pb2_grpc# 全局字典存储共享内存对象引用
shared_memory_store = {}
lock = threading.Lock()class ImageProcessingServiceServicer(image_processing_pb2_grpc.ImageProcessingServiceServicer):def ProcessImage(self, request, context):# 读取共享内存中的图像shm = shared_memory.SharedMemory(name=request.input_shm_name)img_array = np.ndarray((480, 640, 3), dtype=np.uint8, buffer=shm.buf)img = img_array.copy()  # 复制图像以确保共享内存可以被释放# 在图像上写“ok”cv2.putText(img, "ok", (50, 50), cv2.FONT_HERSHEY_SIMPLEX, 1, (255, 0, 0), 2, cv2.LINE_AA)# 创建新的共享内存并将处理后的图像写入output_shm = shared_memory.SharedMemory(create=True, size=img.nbytes)output_array = np.ndarray(img.shape, dtype=img.dtype, buffer=output_shm.buf)output_array[:] = img[:]# 将共享内存对象存储在全局字典中with lock:shared_memory_store[output_shm.name] = output_shm# 启动一个线程在一段时间后清理共享内存threading.Timer(60, self.cleanup_shared_memory, args=(output_shm.name,)).start()# 返回处理后的图像的共享内存名称response = image_processing_pb2.ImageResponse(output_shm_name=output_shm.name)return responsedef cleanup_shared_memory(self, shm_name):with lock:if shm_name in shared_memory_store:shm = shared_memory_store.pop(shm_name)shm.close()shm.unlink()def serve():server = grpc.server(futures.ThreadPoolExecutor(max_workers=1))image_processing_pb2_grpc.add_ImageProcessingServiceServicer_to_server(ImageProcessingServiceServicer(), server)server.add_insecure_port('[::]:50051')server.start()print("Server started, listening on port 50051.")try:while True:time.sleep(86400)  # 运行一整天except KeyboardInterrupt:server.stop(0)if __name__ == '__main__':serve()

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

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

相关文章

4 款最佳 C# 无头浏览器

摘要: 在当今大数据时代,高效的数据采集成为众多项目的关键一环。对于偏好C#语言的开发者而言,无头浏览器是实现网页自动化交互、数据抓取的强大工具。本文将深入探讨四款顶尖的C#无头浏览器库,分析它们的特性和应用场景&#xf…

怎么把C盘分成两个盘?让C盘分区更简单,赶快试试!

在日常使用电脑的过程中,有时我们可能希望将C盘分割成两个独立的分区,以便更好地管理文件和数据。这种操作需要谨慎进行,因为错误的分区操作可能导致数据丢失。那么,我们该怎么把C盘分成两个盘呢?下面,我将…

lua 游戏架构 之 游戏 AI (六)ai_auto_skill

定义一个为ai_auto_skill的类,继承自ai_base类。ai_auto_skill类的目的是在AI自动战斗模式下,根据配置和条件自动选择并使用技能。 lua 游戏架构 之 游戏 AI (一)ai_base-CSDN博客文章浏览阅读379次。定义了一套接口和属性&#…

vue3在元素上绑定自定义事件弹出虚拟键盘

最近开发中遇到一个需求: 焊接机器人的屏幕上集成web前端网页, 但是没有接入键盘。这就需要web端开发一个虚拟键盘,在网上找个很多虚拟键盘没有特别适合,索性自己写个简单的 图片: 代码: (代码可能比较垃圾冗余,也没时间优化,凑合看吧) 第一步:创建键盘组件 为了方便使用…

3.2.微调

微调 ​ 对于一些样本数量有限的数据集,如果使用较大的模型,可能很快过拟合,较小的模型可能效果不好。这个问题的一个解决方案是收集更多数据,但其实在很多情况下这是很难做到的。 ​ 另一种方法就是迁移学习(transfer learning…

c++如何理解多态与虚函数

目录 **前言****1. 何为多态**1.1 **编译时多态**1.1.1 函数重载1.1.2 模板 **1.2 运行时多态****1.2.1 虚函数****1.2.2 为什么要用父类指针去调用子类函数** **2. 注意****2.1 基类的析构函数应写为虚函数****2.2 构造函数不能设为虚函数** **本文参考** 前言 在学习 c 的虚…

打造重庆市数字化教育“新名片”,广阳湾珊瑚中学凭实力“出圈”!

分布于教学楼连廊顶部的智能照明设备,根据不同的时间和场景需求自动调节灯光亮度和开关状态;安装于各个教室内的智能黑板、学校同步时钟、学生互动设备,在极简以太全光网的赋能下,为师生提供丰富的教学体验与学习支持......行走于重庆市广阳湾珊瑚中学,像是与充满科技感的“校园…

病理AI领域的基础模型汇总|顶刊专题汇总·24-07-26

小罗碎碎念 本期文献主题:病理AI领域的最新基础模型 今天的推文是一期生日特辑,定时在下午六点二十一分发表(今天农历六月二十一,哈哈),算是自己给自己的24岁生日礼物,希望24岁这一年&#xff0…

ollama本地部署大语言模型记录

目录 安装Ollama更改模型存放位置 拉取模型GemmaMistralQwen1.5(通义千问)codellama 部署Open webui测试性能知识广度问题1问题2 代码能力总结 最近突然对大语言模型感兴趣 同时在平时的一些线下断网的CTF比赛中,大语言模型也可以作为一个能对话交互的高级知识检索…

SSRF中伪协议学习

SSRF常用的伪协议 file:// 从文件系统中获取文件内容,如file:///etc/passwd dict:// 字典服务协议,访问字典资源,如 dict:///ip:6739/info: ftp:// 可用于网络端口扫描 sftp:// SSH文件传输协议或安全文件传输协议 ldap://轻量级目录访问协议 tftp:// 简单文件传输协议 gopher…

【JavaScript】函数声明和函数表达式的区别

文章目录 一、函数声明1. 定义方式2. 作用域提升(Hoisting)3. 块级作用域 二、函数表达式1. 定义方式2. 作用域提升(Hoisting)3. 自引用 三、其他区别1. 函数名2. 可读性和代码组织3. 使用场景 四、总结函数声明函数表达式 在Java…

【大模型系列】Video-LaVIT(2024.06)

Paper:https://arxiv.org/abs/2402.03161Github:https://video-lavit.github.io/Title:Video-LaVIT: Unified Video-Language Pre-training with Decoupled Visual-Motional TokenizationAuthor:Yang Jin, 北大&#x…

Java面试八股之@Qualifier的作用

Qualifier的作用 Qualifier 是 Spring 框架中的一个非常有用的注解,它主要用于解决在依赖注入过程中出现的歧义问题。当 Spring 容器中有多个相同类型的 Bean 时,Qualifier 可以帮助指明应该使用哪一个具体的 Bean 进行注入。 Qualifier 的作用&#x…

外设购物平台

目 录 一、系统分析 二、系统设计 2.1 系统功能设计 2.2 数据库设计 三、系统实现 3.1 注册功能 3.2 登录功能 3.3 分页查询所有商品信息功能 3.4 分页条件(精确、模糊)查询商品信息功能 3.5 购物车功能 3.6 订单管理功能 四、项…

【Opencv】模糊

消除噪声 用该像素周围的平均值代替该像素值 4个函数 blur():最经典的 import os import cv2 img cv2.imread(os.path.join(.,dog.jpg)) k_size 7 #窗口大小,数字越大,模糊越强 img_blur cv2.blur(img,(k_size,k_size)) #窗口是正方形&#xff…

云计算实训16——关于web,http协议,https协议,apache,nginx的学习与认知

一、web基本概念和常识 1.Web Web 服务是动态的、可交互的、跨平台的和图形化的为⽤户提供的⼀种在互联⽹上浏览信息的服务。 2.web服务器(web server) 也称HTTP服务器(HTTP server),主要有 Nginx、Apache、Tomcat 等。…

C#使用csvhelper实现csv的操作

新建控制台项目 安装csvhelper 33.0.1 写入csv 新建Foo.cs namespace CsvSut02;public class Foo {public int Id { get; set; }public string Name { get; set; } }批量写入 using System.Globalization; using CsvHelper; using CsvHelper.Configuration;namespace Csv…

[数据集][目标检测]金属罐缺陷检测数据集VOC+YOLO格式8095张4类别

数据集格式:Pascal VOC格式YOLO格式(不包含分割路径的txt文件,仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数):8095 标注数量(xml文件个数):8095 标注数量(txt文件个数):8095 标注…

使用Process Explorer和Dependency Walker排查dll动态库加载失败的问题

目录 1、问题描述 2、如何调试Release版本的代码? 3、使用Process Explorer查看exe主程序加载的dll库列表,发现mediaplay.dll没有加载起来 4、使用Dependency Walker查看rtcmpdll.dll的库依赖关系和接口调用情况,定位问题 4.1、使用Depe…

Javascript面试基础6【每日更新10】

Gulp gulp是前端开发过程中一种基于流的代码构建工具,是自动化项目的构建利器;它不仅能对网站资源进行优化,而且在开发过程中很多重复的任务能够使用正确的工具自动完成 Gulp的核心概念:流 流,简单来说就是建立在面向对象基础上的一种抽象的…