C#通讯——关于Winform中的简单的Http服务器与客户端
C#通讯——关于Winform中的简单的Http服务器与客户端
- 前言
- 一、Http是什么?
- 二、简单的Http服务器
- 三、简单的Http客户端
- 四、实际调用
- 五、Winform中Http服务器和WebApi的区别?
前言
在实际项目中通讯的交互的过程中,遇见数据传输时同事和我说用WebApi,Get,Post等等,以前没用搞过这个块的我傻傻分不清。查询资料了解过后发现只是个简单的Http交互通讯,在此记录下Winform中的简单服务器和客户端的编写,并简要描述webapi与http通讯的差异。
一、Http是什么?
HTTP(Hypertext Transfer Protocol)是一种用于在网络上传输超文本的协议。它是一个用于传输和交换超文本、图像、视频、音频以及其他多媒体资源的应用层协议。
HTTP是Web的基础,它定义了客户端和服务器之间的通信规则。通常,当你在浏览器中输入网址、点击链接或者提交表单时,浏览器会发送HTTP请求到服务器,服务器收到请求后进行处理,然后返回HTTP响应给客户端(浏览器),最终在你的浏览器上显示网页内容。
HTTP协议的主要特点包括:
- 无状态性(Stateless):
HTTP协议本身是无状态的,每个请求都是相互独立的,服务器并不会保留前后请求之间的状态信息。这导致每个请求都需要包含足够的信息来被服务器正确处理,比如会话标识或者其他验证信息。 - 基于请求-响应模型:
客户端发送请求给服务器,服务器处理请求并返回响应。这个请求-响应模型是HTTP的核心概念,通过这种方式客户端能够请求不同资源并且服务器能够提供响应。 - 支持多种请求方法:
HTTP定义了一系列请求方法(也称为HTTP谓词),包括GET(用于获取资源)、POST(用于提交数据给服务器)、PUT(用于更新资源)、DELETE(用于删除资源)等。 - URL定位资源:
每个HTTP请求都包含一个URL(Uniform Resource Locator),用于指定被请求的资源的位置。 - 无连接性:
HTTP协议本身是无连接的,即每次请求完成后,连接会立即关闭。这意味着每个请求都需要重新建立连接,可能会造成一定的性能开销。
二、简单的Http服务器
using System;
using System.Collections.Generic;
using System.Net;
using System.Threading.Tasks;namespace httpServer
{class HttpServer_DP{private readonly string mUrl; // 服务器监听的URLprivate readonly HttpListener mListener; // HttpListener实例private readonly Dictionary<string, Func<HttpListenerContext, Task<string>>> mRoutes; // 路由映射// 构造函数,接收服务器监听地址和端口的数组public HttpServer_DP(string[] prefixes){if (!HttpListener.IsSupported){throw new NotSupportedException("HttpListener is not supported on this platform.");}mListener = new HttpListener(); // 初始化HttpListener实例mRoutes = new Dictionary<string, Func<HttpListenerContext, Task<string>>>(); // 初始化路由映射字典// 为服务器添加监听地址和端口foreach (string prefix in prefixes){mListener.Prefixes.Add(prefix);}mUrl = prefixes[0]; // 记录第一个监听地址}public bool IsOpen { get { return mListener.IsListening; } }// 启动服务器public void Start(){mListener.Start();Console.WriteLine($"Server started and listening on {mUrl}");mListener.BeginGetContext(ProcessRequestCallback, mListener); // 处理客户端请求}// 停止服务器public void Stop(){mListener.Stop();mListener.Close();Console.WriteLine("Server stopped.");}// 添加路由和处理程序的映射关系public void AddRoute(string route, Func<HttpListenerContext, Task<string>> handler){mRoutes.Add(route, handler);}// 处理客户端请求的回调函数private async void ProcessRequestCallback(IAsyncResult result){HttpListener listener = (HttpListener)result.AsyncState;// 开始下一个请求的监听listener.BeginGetContext(ProcessRequestCallback, listener);try{// 获取传入的请求HttpListenerContext context = listener.EndGetContext(result);// 获取请求方法和URL路径string httpMethod = context.Request.HttpMethod;string responseString = "No Data!"; // 默认响应字符串string url = context.Request.Url.AbsolutePath;Func<HttpListenerContext, Task<string>> handler;// 如果请求路径存在于路由映射中,执行相应的处理程序if (mRoutes.TryGetValue(url, out handler)){responseString = await handler(context); // 获取处理程序返回的响应数据// 将响应数据编码成字节数组byte[] buffer = System.Text.Encoding.UTF8.GetBytes(responseString);// 设置响应的内容长度和状态码context.Response.ContentLength64 = buffer.Length;context.Response.StatusCode = (int)HttpStatusCode.OK;// 将响应写入输出流并关闭输出流context.Response.OutputStream.Write(buffer, 0, buffer.Length);context.Response.OutputStream.Close();}else{// 如果请求路径不存在于路由映射中,返回404 Not Foundcontext.Response.StatusCode = (int)HttpStatusCode.NotFound;context.Response.Close();}}catch (Exception ex){Console.WriteLine($"Error processing request: {ex.Message}");}}}
}
需要注意的是:Dictionary<string, Func<HttpListenerContext, Task>> mRoutes; // 路由映射
为了方便根据客户端传入的URL的不同路径来选择进行不同的处理流程
三、简单的Http客户端
using System;
using System.Net.Http;
using System.Threading.Tasks;namespace Http
{class HttpClient_DP{private readonly HttpClient _client; // HttpClient 实例// 构造函数,初始化 HttpClient 实例public HttpClient_DP(){_client = new HttpClient();}// 设置超时时间public TimeSpan TimeOut { set { _client.Timeout = value; } }// 发送 GET 请求并返回响应内容public async Task<string> Get(string url){HttpResponseMessage response = await _client.GetAsync(url); // 发送 GET 请求if (response.IsSuccessStatusCode){string content = await response.Content.ReadAsStringAsync(); // 读取响应内容return content;}else{return $"Error: {response.StatusCode}"; // 返回错误信息}}// 发送 POST 请求并返回响应内容public async Task<string> Post(string url, string data){HttpContent content = new StringContent(data); // 创建包含请求数据的 HttpContent 实例HttpResponseMessage response = await _client.PostAsync(url, content); // 发送 POST 请求if (response.IsSuccessStatusCode){string responseData = await response.Content.ReadAsStringAsync(); // 读取响应内容return responseData;}else{return $"Error: {response.StatusCode}"; // 返回错误信息}}}
}
其中只实现的Get与Post的相关代码,这也是当前我最常用的两个方法。
其中 public TimeSpan TimeOut为超时时间,不设置时将永久等待服务器响应。当响应超时时,外部可以通过(try-catch)捕获TaskCanceledException异常。
四、实际调用
初始化服务器:
httpServer.HttpServer_DP HttpServer;public static void IniHttpServer(){HttpServer = new httpServer.HttpServer_DP(new string[]{"http://127.0.0.1:8080/"});//绑定映射,处理函数//当传入URL为"http://127.0.0.1:8080/PostAGVC_Data/"时将调用PostAGVC_Data函数接收和解析HttpServer.AddRoute("/PostAGVC_Data/", PostAGVC_Data);//当传入URL为"http://127.0.0.1:8080/PostAGV_Data/"时将调用PostAGV_Data函数接收和解析HttpServer.AddRoute("/PostAGV_Data/", PostAGV_Data);HttpServer.Start();}public static async Task<string> PostAGVC_Data(HttpListenerContext context){string httpMethod = context.Request.HttpMethod;string responseString = "";// 处理 POST 请求if (context.Request.HasEntityBody){// 从请求主体中获取数据using (System.IO.Stream body = context.Request.InputStream){using (System.IO.StreamReader reader = new System.IO.StreamReader(body, context.Request.ContentEncoding)){string postData = reader.ReadToEnd(); // 读取 POST 数据//处理数据//。。。。。。。。。。。。//。。。。。。。。。。。。//返回字符串responseString = "OK";}}}return responseString;}//与PostAGVC_Data内部类似,处理数据的代码不一样public static async Task<string> PostAGV_Data(HttpListenerContext context){//省略}
由于Http的无连接性,客户端只需要实例化后,需要时进行连接就行,简单的初始化(如果需要的话)如下:
Http.HttpClient_DP HttpClient ;public static void IniHttpClient(){if (HttpClient == null){HttpClient = new Http.HttpClient_DP();HttpClient.TimeOut = TimeSpan.FromSeconds(10);}}
调用:
其中JsonConvert只是将我的类实例序列化成Json字符串,在此可以仅当字符串来看。
string ret = await Global_Value.HttpClient.Post(url, JsonConvert.SerializeObject(new PLC_WorkState(){name = agvName,workState = workState}));
五、Winform中Http服务器和WebApi的区别?
在WinForms中编写的HTTP服务器类与WebAPI之间有几个主要区别:
1.目的和设计用途:
- WinForms HTTP服务器类:
通常是基于Socket或其他网络库手动实现的简单HTTP服务器。这样的服务器可能是为了特定目的而创建,比如在本地进行简单的通信或提供特定服务。 - WebAPI: 是一个在Web开发中常用的框架,用于构建基于HTTP的RESTful
API。它是ASP.NET框架的一部分,提供了更完整、功能更丰富的API开发环境,包括路由、中间件、控制器、模型绑定等。
2.功能和特性:
- WinForms HTTP服务器类:
往往具有较少的功能,可能只实现了基本的HTTP请求/响应处理,而缺少高级特性和工具。它通常用于较小规模的需求。 - WebAPI:
提供了许多功能强大的特性,例如路由,HTTP谓词(GET、POST、PUT等)的自动映射到相应的处理方法,参数绑定、模型验证等,使得开发者能够更轻松地构建和管理API。
3.集成和扩展性:
- WinForms HTTP服务器类: 通常需要手动处理连接、请求、响应和其他网络层细节,灵活性较高,但也需要开发者自行处理许多功能。
- WebAPI:
集成在ASP.NET中,可以利用ASP.NET提供的丰富特性和中间件,容易与其他ASP.NET组件集成,并且具有更强的扩展性。
4.适用场景:
-
WinForms HTTP服务器类: 适用于需要在WinForms应用程序中实现简单的HTTP通信或提供基本服务的情况。
-
WebAPI: 更适合构建大型的Web应用程序或服务,提供多种HTTP服务,满足各种复杂的需求,例如构建RESTfulAPI供移动应用或Web前端使用。
总体来说,WinForms中的简单HTTP服务器类适用于简单的、局部的HTTP通信需求,而WebAPI更适用于构建和管理完整的Web服务或RESTful API,并提供了更多现成的工具和功能来简化开发流程。
相关文章:
C#通讯——关于Winform中的简单的Http服务器与客户端
C#通讯——关于Winform中的简单的Http服务器与客户端 前言一、Http是什么?二、简单的Http服务器三、简单的Http客户端四、实际调用五、Winform中Http服务器和WebApi的区别? 前言 在实际项目中通讯的交互的过程中,遇见数据传输时同事和我说用…...
Mendelson AS2 介绍下载和配置
最近与一家国外公司做EDI对接,并且EDI通讯工具是基于AS2协议的。目前开源的as2的开源项目有openas2,Mendelson AS2,和国人写的freeas2但是,现在freeas2已经被从开源中国不能下载了,变为收费的版本了。 如果你需要使用基于AS2协议…...
旅游海报图怎么做二维码展示?扫码即可查看图片
现在旅游攻略的海报可以做成二维码印刷在宣传单单页或者分享给用户来了解目的地的实际情况,出行路线、宣传海报等。用户只需要扫描二维码就可以查看内容,更加的方便省劲,那么旅游海报的图片二维码制作的技巧有哪些呢?使用图片二维…...
常用git指令
初始化Git仓库:git init 添加文件到暂存区:git add <file> 提交更改到本地仓库:git commit -m "commit message" 查看本地仓库的提交历史:git log 创建分支:git branch <branch_name> 切换分支:git checkout <branch_name> 查看所有分支:git…...
【FPGA】分享一些FPGA协同MATLAB开发的书籍
在做FPGA工程师的这些年,买过好多书,也看过好多书,分享一下。 后续会慢慢的补充书评。 【FPGA】分享一些FPGA入门学习的书籍【FPGA】分享一些FPGA协同MATLAB开发的书籍 【FPGA】分享一些FPGA视频图像处理相关的书籍 【FPGA】分享一些FPGA高速…...
幺模矩阵-线性规划的整数解特性
百度百科:幺模矩阵 在线性规划问题中,如果A为幺模矩阵,那么该问题具有最优整数解特性。也就是说使用单纯形法进行求解,得到的解即为整数解。无需再特定使用整数规划方法。 m i n c T x s . t . { A x ≥ b x ≥ 0 \begin{align*} min \quad…...
数据分析思维
Why&What 数据分析是为了驱动决策赋能业务。在数据分析过程中需要对目标进行拆解量化,如何拆解量化目标便是数据分析思维。 在任务拆解过程中使用的软件、统计模型、分析方法等为分析工具和手段,如何在恰当的场景合理的使用这些工具、模型、方法、手…...
C++ boost planner_cond_.wait(lock) 报错1225
1.如下程序段 boost unique_lock doesn’t own the mutex: Operation not permitted 问题: 其中makePlan是一个线程。这里的unlock导致错误这个报错 boost unique_lock doesn’t own the mutex: Operation not permitted bool navigation::makePlan(){ //cv::named…...
LeetCode刷题--- 字母大小写全排列
个人主页:元清加油_【C】,【C语言】,【数据结构与算法】-CSDN博客 个人专栏 力扣递归算法题 http://t.csdnimg.cn/yUl2I 【C】 http://t.csdnimg.cn/6AbpV 数据结构与算法 http://t.csdnimg.cn/hKh2l 前言:这个专栏主要讲述递归递归、搜索与回…...
165. 小猫爬山(DFS之剪枝与优化)
165. 小猫爬山 - AcWing题库 翰翰和达达饲养了 N 只小猫,这天,小猫们要去爬山。 经历了千辛万苦,小猫们终于爬上了山顶,但是疲倦的它们再也不想徒步走下山了(呜咕>_<)。 翰翰和达达只好花钱让它们…...
【Linux系统基础】(6)在Linux上大数据NoSQL数据库HBase集群部署、分布式内存计算Spark环境及Flink环境部署详细教程
大数据NoSQL数据库HBase集群部署 简介 HBase 是一种分布式、可扩展、支持海量数据存储的 NoSQL 数据库。 和Redis一样,HBase是一款KeyValue型存储的数据库。 不过和Redis设计方向不同 Redis设计为少量数据,超快检索HBase设计为海量数据,…...
多维时序 | MATLAB实CNN-BiGRU-Mutilhead-Attention卷积网络结合双向门控循环单元网络融合多头注意力机制多变量时间序列预测
多维时序 | MATLAB实现CNN-BiGRU-Mutilhead-Attention卷积网络结合双向门控循环单元网络融合多头注意力机制多变量时间序列预测 目录 多维时序 | MATLAB实现CNN-BiGRU-Mutilhead-Attention卷积网络结合双向门控循环单元网络融合多头注意力机制多变量时间序列预测预测效果基本介…...
vs快捷键
ctrlMo 折叠代码块 ctrlML 打开代码块...
linux 内核时间计量方法
定时器中断由系统定时硬件以规律地间隔产生; 这个间隔在启动时由内核根据 HZ 值来编 程, HZ 是一个体系依赖的值, 在 <linux/param.h>中定义或者它所包含的一个子平台文 件中. 在发布的内核源码中的缺省值在真实硬件上从 50 到 1200 嘀哒每秒, 在软件模拟 器中往下到 24.…...
循环神经网络中的梯度消失或梯度爆炸问题产生原因分析(二)
上一篇中讨论了一般性的原则,这里我们具体讨论通过时间反向传播(backpropagation through time,BPTT)的细节。我们将展示目标函数对于所有模型参数的梯度计算方法。 出于简单的目的,我们以一个没有偏置参数的循环神经…...
JWT signature does not match locally computed signature
1. 问题背景 最近在协助团队小盆友调试一个验签问题,结果还“节外生枝”了,原来不是签名过程的问题,是token的问题。 当你看到“JWT signature does not match locally computed signature. JWT validity cannot be asserted and should not…...
vitepress项目使用github的action自动部署到github-pages中,理论上可以通用所有
使用github的action自动部署到github-pages中 创建部署的deploy.yml文件,在项目的根目录下面 .github\workflows\deploy.yml 完整的代码:使用的是pnpm进行依赖安装。 name: 部署VitePresson:push:branches:- docs # 这段是在推送到 docs 分支时触发该…...
Python爬虫---解析---JSONPath
Xpath可以解析本地文件和服务器响应的文件,JSONPath只能解析本地文件 1. 安装jsonpath:pip install jsonpath 注意:需要安装在python解释器相同的位置,例如:D:\Program Files\Python3.11.4\Scripts 2. 使用步骤 2.1 导入&…...
路由器介绍和命令操作
先来回顾一下上次的内容: ip地址就是由32位二进制数组 二进位数就是只有数字0和1组成 网络位:类似于区号,表示区域作用 主机位:类似于号码,表示区域中编号 网络名称:网络位不变,主机位全为0 …...
Hadoop——分布式计算
一、分布式计算概述 1. 什么是计算、分布式计算? 计算:对数据进行处理,使用统计分析等手段得到需要的结果分布式计算:多台服务器协同工作,共同完成一个计算任务2. 分布式计算常见的2种工作模式分散->汇总 (MapReduce就是这种模式)将数据分片,多台服务器各自负责一…...
设计模式和设计原则回顾
设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...
转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...
苍穹外卖--缓存菜品
1.问题说明 用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大 2.实现思路 通过Redis来缓存菜品数据,减少数据库查询操作。 缓存逻辑分析: ①每个分类下的菜品保持一份缓存数据…...
HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...
pikachu靶场通关笔记22-1 SQL注入05-1-insert注入(报错法)
目录 一、SQL注入 二、insert注入 三、报错型注入 四、updatexml函数 五、源码审计 六、insert渗透实战 1、渗透准备 2、获取数据库名database 3、获取表名table 4、获取列名column 5、获取字段 本系列为通过《pikachu靶场通关笔记》的SQL注入关卡(共10关࿰…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
【Go语言基础【13】】函数、闭包、方法
文章目录 零、概述一、函数基础1、函数基础概念2、参数传递机制3、返回值特性3.1. 多返回值3.2. 命名返回值3.3. 错误处理 二、函数类型与高阶函数1. 函数类型定义2. 高阶函数(函数作为参数、返回值) 三、匿名函数与闭包1. 匿名函数(Lambda函…...
音视频——I2S 协议详解
I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议,专门用于在数字音频设备之间传输数字音频数据。它由飞利浦(Philips)公司开发,以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...
并发编程 - go版
1.并发编程基础概念 进程和线程 A. 进程是程序在操作系统中的一次执行过程,系统进行资源分配和调度的一个独立单位。B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位。C.一个进程可以创建和撤销多个线程;同一个进程中…...
