ASP.NET Core SignalR案例:导入英汉词典

Ecdict

  1. 下载词典文件stardict.7z,解压,stardict.csv是一个CSV格式的文本文件,文件的第一行是表头,除第一行外,其他每行文本是一个单词的相关信息,用逗号分隔的就是各个列的值。
  2. 英汉词典ECDICT中导入单词到数据库。
  3. T_WordItems:Id(主键)、Word(单词)、Phonetic(音标)、Definition(英文解释),Translation(中文翻译)

https://github.com/skywind3000/ECDICTicon-default.png?t=O83Ahttps://github.com/skywind3000/ECDICT

实现

  1. ImportExecutor中注入IHubContext<ImportDictHub>等服务。
  2. 暂时用字符串Split解析CSV,或者用更专业的库。
  3. 用SqlBulkCopy 进行分批快速导入:

Program.cs

builder.Services.AddSignalR();string[] urls = new[] { "http://localhost:5173" };
builder.Services.AddCors(options =>options.AddDefaultPolicy(builder => builder.WithOrigins(urls).AllowAnyMethod().AllowAnyHeader().AllowCredentials()));builder.Services.AddScoped<ImportExecutor>();app.MapHub<ImportHub>("/ImportHub");

ImportExecutor.cs

public class ImportExecutor
{private readonly IHubContext<ImportHub> hubContext;public ImportExecutor(IHubContext<ImportHub> hubContext){this.hubContext = hubContext;}public async Task ExecuteAsync(string connectionId){string[] lines = await File.ReadAllLinesAsync(@"F:\Demo\stardict\stardict.csv");int totalCount = lines.Length - 1;string connStr = "Data Source=.;Initial Catalog=demo;Integrated Security=SSPI;TrustServerCertificate=true";SqlBulkCopy bulkCopy = new SqlBulkCopy(connStr);bulkCopy.DestinationTableName = "T_WordItems";bulkCopy.ColumnMappings.Add("Word", "Word");bulkCopy.ColumnMappings.Add("Phonetic", "Phonetic");bulkCopy.ColumnMappings.Add("Definition", "Definition");bulkCopy.ColumnMappings.Add("Translation", "Translation");int counter = 0;using DataTable dataTable = new DataTable();dataTable.Columns.Add("Word");dataTable.Columns.Add("Phonetic");dataTable.Columns.Add("Definition");dataTable.Columns.Add("Translation");foreach (var item in lines){string[] str = item.Split(',');string word = str[0];string? phonetic = str[1];string? definition = str[2];string? translation = str[3];DataRow row = dataTable.NewRow();row["Word"] = word;row["Phonetic"] = phonetic;row["Definition"] = definition;row["Translation"] = translation;dataTable.Rows.Add(row);counter++;Console.WriteLine($"已加载{counter}");if (dataTable.Rows.Count == 100){await bulkCopy.WriteToServerAsync(dataTable);dataTable.Clear();}await hubContext.Clients.Client(connectionId).SendAsync("ImportProgress",totalCount,counter);}await bulkCopy.WriteToServerAsync(dataTable);await hubContext.Clients.Client(connectionId).SendAsync("ImportProgress", totalCount, counter);}
}

ImportEcdict.cs

public class ImportHub : Hub
{private readonly ImportExecutor importExecutor;public ImportHub(ImportExecutor importExecutor){this.importExecutor = importExecutor;}public Task ImportEcdict(){_ = importExecutor.ExecuteAsync(this.Context.ConnectionId);return Task.CompletedTask;}
}

Vue

<template><div><input type="button" value="导入" v-on:click="importEcdict"><progress :value="state.importedCount" :max="state.totalCount"></progress><span>{{ state.importedCount }},{{ state.totalCount }}{{ ((state.importedCount / state.totalCount)*100).toFixed(2)}}%</span></div>
</template><script>
import { reactive, onMounted } from 'vue';
import * as signalR from '@microsoft/signalr';
import axios from 'axios';let connection;
export default {name: 'Login',setup() {//创建响应式对象const state = reactive({ importedCount: 0, totalCount: 0 });onMounted(async function () {startConn()})//SignalR连接const startConn = async function () {const transport = signalR.HttpTransportType.WebSockets;const options = { skipNegotiation: true, transport: transport };connection = new signalR.HubConnectionBuilder().withUrl('https://localhost:7222/ImportHub', options).withAutomaticReconnect().build();try {await connection.start();} catch (err) {alert(err);return;}//接收消息connection.on('ImportProgress', (totalCount, counter) => {//监听服务器端发送过来的信息state.importedCount = counter;state.totalCount = totalCount});}//导入const importEcdict = async function (e) {await connection.invoke("ImportEcdict");// alert("启动导入")}//返回响应式对象和方法return { state, importEcdict };}
}
</script>

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

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

相关文章

计算机自学资源分享

分享本硕期间学习过、收藏过的一些计算机相关的课程、书籍、资源。大部分都是开源资源和免费资源&#xff0c;部分优质资源可能收费。 文章目录 综合资源PythonMachine LearningC/CJava前端SQL数据结构与算法计算机基础AIGC类求职电子资源GitHub 优质项目 综合资源 【指南】ka…

[7] 游戏机项目说明

[7] 游戏机项目说明 在这节课中&#xff0c;我们将学习如何基于FreeRTOS开发一个简单的游戏项目。我们会使用一个开源项目nwatch&#xff0c;它是一个基于STM32的开源手表&#xff0c;包含了三个游戏。我们的目标是将这个游戏移植到我们的开发板上&#xff0c;并逐步使用FreeR…

【清晰教程】通过Docker为本地DeepSeek-r1部署WebUI界面

【清晰教程】本地部署DeepSeek-r1模型-CSDN博客 目录 安装Docker 配置&检查 Open WebUI 部署Open WebUI 安装Docker 完成本地DeepSeek-r1的部署后【清晰教程】本地部署DeepSeek-r1模型-CSDN博客&#xff0c;通过Docker为本地DeepSeek-r1部署WebUI界面。 访问Docker官…

shell脚本控制——处理信号

Linux利用信号与系统中的进程进行通信。你可以通过对脚本进行编程&#xff0c;使其在收到特定信号时执行某些命令&#xff0c;从而控制shell脚本的操作。 1.重温Linux信号 Linux系统和应用程序可以产生超过30个信号。下表列出了在shell脚本编程时会遇到的最常见的Linux系统信…

2025web寒假作业二

一、整体功能概述 该代码构建了一个简单的后台管理系统界面&#xff0c;主要包含左侧导航栏和右侧内容区域。左侧导航栏有 logo、管理员头像、导航菜单和安全退出按钮&#xff1b;右侧内容区域包括页头、用户信息管理内容&#xff08;含搜索框和用户数据表格&#xff09;以及页…

Node.js怎么调用到打包的python文件呢

在 Node.js 中调用打包后的 Python 可执行文件&#xff08;如 PyInstaller 生成的 .exe 或二进制文件&#xff09;&#xff0c;可以通过以下步骤实现&#xff1a; 一、Python 打包准备 假设已有打包好的 Python 文件 your_script.exe&#xff08;以 Windows 为例&#xff09;&…

【开源AI】AI一页一页读PDF

【开源AI】AI一页一页读PDF 可以在这里看 : 让AI 处理 PDF 文件,提取其中的知识点,并生成总结。 只是无法修改,后续若有更新在csdn这里。 【OpenAI】 API 更新: JSON 结构化输出约束机制( JSON Schema) 的一次实战。知识库的JSON Schema形式 每一页都要总结,总结的知识…

如何在 Qt 中添加和使用系统托盘图标

在 Qt 中实现系统托盘图标是一个常见的需求&#xff0c;尤其是在桌面应用程序中。系统托盘图标可以让应用程序在后台运行时仍然具有可见性&#xff0c;同时避免占用过多的桌面空间。本文将详细介绍如何在 Qt 项目中添加托盘图标&#xff0c;并通过资源系统&#xff08;.qrc 文件…

探索B-树系列

&#x1f308;前言&#x1f308; 本文将讲解B树系列&#xff0c;包含 B-树&#xff0c;B树&#xff0c;B*树&#xff0c;其中主要讲解B树底层原理&#xff0c;为什么用B树作为外查询的数据结构&#xff0c;以及B-树插入操作并用代码实现&#xff1b;介绍B树、B*树。 &#x1f4…

Python的

& 运算符可用于不同集合类型&#xff0c;它主要用于集合的交集操作 下面分别介绍它在 set&#xff08;集合&#xff09;和 frozenset&#xff08;不可变集合&#xff09;这两种常见集合类型中的使用 set 类型 set 是 Python 中内置的可变集合类型&#xff0c;使用 & …

深入与浅出-Python爬虫逆向实战

一、什么是爬虫逆向&#xff1f; 爬虫逆向&#xff0c;简单来说&#xff0c;就是通过分析网页的前端和后端行为&#xff0c;找出数据的来源和获取方式&#xff0c;从而实现自动化抓取。很多时候&#xff0c;直接使用requests和BeautifulSoup可能无法获取到目标数据&#xff0c…

使用 POI-TL 和 JFreeChart 动态生成 Word 报告

文章目录 前言一、需求背景二、方案分析三、 POI-TL JFreeChart 实现3.1 Maven 依赖3.3 word模板设置3.2 实现代码 踩坑 前言 在开发过程中&#xff0c;我们经常需要生成包含动态数据和图表的 Word 报告。本文将介绍如何结合 POI-TL 和 JFreeChart&#xff0c;实现动态生成 W…

Java网络编程学习(一)

网络相关概念 网络体系结构 OSI体系结构&#xff08;七层&#xff09; OSI&#xff08;Open Systems Interconnection&#xff0c;开放系统互联&#xff09;体系结构将整个计算机网络分为七层&#xff0c;从上到下依次为&#xff1a;应用层、表示层、会话层、传输层、网络层…

flutter ListView Item复用源码解析

Flutter 的 ListView 的 Item 复用机制是其高性能列表渲染的核心&#xff0c;底层实现依赖于 Flutter 的渲染管线、Element 树和 Widget 树的协调机制。以下是 ListView 复用机制的源码级解析&#xff0c;结合关键类和核心逻辑进行分析。 1. ListView 的底层结构 ListView 的复…

粒子群优化算法:像鸟群一样找到最优解

前言 在人工智能的浩瀚星空中,粒子群优化算法(PSO)如同一颗熠熠生辉的明星,吸引了无数科研人员的目光。它的名字听起来好像非常高大上,仿佛只有数学天才和算法大师才能理解。但实际上,PSO的原理并没有那么复杂。想象一下,一群聪明的小鸟在天空中自由飞翔,大家互相呼唤…

QT修仙之路2-2 对话框 尚欠火候

警告对话框 相关代码 错误对话框 相关代码 消息对话框 相关代码 询问对话框 相关代码 相关代码 警告对话框 QMessageBox::warning(this,"错误","账号密码不能为空",QMessageBox::Ok);错误对话框 QMessageBox msgBox(QMessageBox::Critical,"错误…

Python 字典(一个简单的字典)

在本章中&#xff0c;你将学习能够将相关信息关联起来的Python字典。你将学习如何访问和修改字典中的信息。鉴于字典可存储的信息量几乎不受限制&#xff0c;因此我们会演示如何遍 历字典中的数据。另外&#xff0c;你还将学习存储字典的列表、存储列表的字典和存储字典的字典。…

conda 修复 libstdc++.so.6: version `GLIBCXX_3.4.30‘ not found 简便方法

ImportError: /data/home/hum/anaconda3/envs/ipc/bin/../lib/libstdc.so.6: version GLIBCXX_3.4.30 not found (required by /home/hum/anaconda3/envs/ipc/lib/python3.11/site-packages/paddle/base/libpaddle.so) 1. 检查版本 strings /data/home/hum/anaconda3/envs/ipc/…

RTD2775QT/RTD2795QT瑞昱显示器芯片方案

RTD2775QT与RTD2795QT&#xff1a;高性能4K显示驱动芯片 RTD2775QT与RTD2795QT是瑞昱半导体公司推出的两款高性能显示驱动芯片&#xff0c;专为满足现代显示设备对高清、高分辨率的需求而设计。这两款芯片不仅支持4K分辨率&#xff0c;还具备丰富的功能和卓越的性能&#xff0…

Linux路径中的‘~‘

本文来自DeepSeek 在Linux中&#xff0c;~ 是用户主目录的简写。具体含义如下&#xff1a; 当前用户的主目录&#xff1a; ~ 代表当前登录用户的主目录。例如&#xff0c;用户 alice 的主目录通常是 /home/alice&#xff0c;~ 就指向 /home/alice。 其他用户的主目录&#xff…