当前位置: 首页 > news >正文

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>

相关文章:

  • 开发中用到的设计模式
  • 前端面试题目---页面抖动的原因、如何避免、如何解决
  • Spring Boot过滤器链:从入门到精通
  • 最新版Edge浏览器集成ActiveX控件之金山WpsDocFrame控件
  • redis之数据库
  • redis 缓存击穿问题与解决方案
  • Fabric.js、leaferjs、pixi.js 库的对比分析
  • 清华大学《DeepSeek:从入门到精通》
  • 零基础入门机器学习 -- 第四章分类问题与逻辑回归
  • 自动化测试 - 黑马头条测试项目笔记
  • IPoIB模块初始化过程详解
  • 企业使用统一终端管理(UEM)工具提高端点安全性
  • Django项目中创建app并快速上手(pycharm Windows)
  • 【论文笔记】ZeroGS:扩展Spann3R+GS+pose估计
  • Ubuntu 22.04 LTS 安装MinerU
  • window 安装GitLab服务器笔记
  • Python 数据结构速成教程
  • 数据结构:图论入门
  • 深入理解 C# 顶级语句:告别 Main 方法
  • Java Swing-5.jar 使用 jpackage 打包成 windows 可安装应用(exe,msi,免安装版exe)
  • 大连三合一网站制作/希爱力的功效及副作用
  • 做影视网站关停/免费推广途径
  • 襄阳做网站的公司有哪些/百度站长平台怎么用
  • 做网站桂林/互联网营销师证书含金量
  • 做漫画在线观看网站/新闻软文推广案例
  • wordpress 轮播广告/seo外链自动群发工具