QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3574|回复: 0
打印 上一主题 下一主题

[其他资源] 基于Python+JavaScript的面向文本分析的交互式主题建模可视化分析系统

[复制链接]
字体大小: 正常 放大
杨利霞        

5273

主题

82

听众

17万

积分

  • TA的每日心情
    开心
    2021-8-11 17:59
  • 签到天数: 17 天

    [LV.4]偶尔看看III

    网络挑战赛参赛者

    网络挑战赛参赛者

    自我介绍
    本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。

    群组2018美赛大象算法课程

    群组2018美赛护航培训课程

    群组2019年 数学中国站长建

    群组2019年数据分析师课程

    群组2018年大象老师国赛优

    跳转到指定楼层
    1#
    发表于 2022-9-13 12:40 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
    基于Python+JavaScript的面向文本分析的交互式主题建模可视化分析系统) }/ F4 g- b( L8 ^; m- C6 f7 ~
      p( N- v) r& r: a
    面向文本分析的交互式主题建模
    ' l+ U" j8 E* v  e
    ! I) s! K& K2 T( A. }; }5 H, R目录
    5 d, A2 [2 R$ @9 S# d面向文本分析的交互式主题建模 1* O3 E/ m. ~+ K- w' j
    一、绪论 2- W! [0 s) c& k9 q
    1.1 本课题的研究背景和意义 2
    ) b$ Z* M/ A+ H7 {, T  i7 q* j1.1.1 主题模型的发展及研究现状 2# B  T. ]0 g+ J$ X0 J
    1.1.2 目前存在的问题 3
    9 Y* g) }$ d9 R0 g1 m' `1.1.3 本课题的研究意义 30 |4 `* L$ M: {' M; S! f; R3 v* S
    1.2 研究内容和主要工作 3% g' V. r0 Z9 _6 H3 D6 d; c2 j1 x; L2 e
    1.3 本文的组织结构 3- y- L( |( _/ o# X. R; z4 L
    二、核心算法 3
    7 q' N" L4 ^* H' |5 \' [2.1 文本预处理 4& u7 k2 j+ _7 J/ r$ C: I2 s
    2.2 大型语料库的内存优化 63 ^5 |  ~+ L/ }3 w; V
    2.3 UMAP 数据降维&可视化 77 z: R* t; I3 z9 x
    三、系统设计与实现 9, U# i% `/ m. C
    3.1 系统介绍及流程图 9, N! U5 i' r$ g4 `
    3.2 后端实现过程 106 E& V, G: ^. j- m
    3.2.1 框架介绍 10# b8 F3 g! p- Z
    3.2.2 数据库 ORM 映射类 10
    ( f# U5 U: M+ V5 H3 k/ _% c- y& u3.2.3 Pydantic 模型类 125 A3 V& F) l7 k* z" w, e
    3.2.4 TextPreprocessing 文本预处理类 120 g7 Y" H* `: S1 h1 a
    3.2.5 SSNMFTopicModel 非负矩阵分解类 12
    0 @- W) \. C, D! w7 c- y7 N0 Q3.2.6 TopicModelTrainingTask 训练任务类 12
    & n* s5 L& }0 L- Y: J& I" d5 |" Z3.2.7 Web API 接口 13  k8 U1 m" y! ]6 R2 P% H8 `8 T
    3.3 前端实现过程 16; V$ U. P5 b  ~  {7 l' q
    3.3.1 框架介绍 16
    ! ?5 E+ D  X; H3.3.2 介绍界面 162 N) N6 U% ~$ t1 V8 e- b+ P: d6 ?) c
    3.3.3 Bilibili 视频评论爬虫界面 17' O; M' m5 l% M4 c( J/ \# R
    3.3.4 语料库查看/选择界面 18
    ) L8 v) j% L* m! o, }" l$ n3.3.5 侧边栏主控制界面 19
    8 v1 a7 e9 i( h, l7 M3.3.6 右侧训练状态 Tab 193 _5 @" _% K7 e& R+ h9 z5 A  J* H
    3.3.7 词云图 Tab 19
    0 E' y) x( [8 u  Y$ e0 I3.3.8 NMF 迭代误差折线图 Tab 20% S3 c4 B' _& i" \. ]' y
    3.3.9 主题聚类可视化 Tab 20
    0 r* w$ S3 C" [# i3.3.10 文档详细信息抽屉 22
    8 J6 ~) x" F+ b6 n) [, K' f* t3 i3.3.11 主题详细信息抽屉 23
    3 b" ?* ?: L/ Z3 c9 ^$ @6 T3.3.12 用户交互 23; @# L# F8 B( c
    3.3.13 新文档主题分布预测 29- ^( R: R& W3 _1 B) h
    四、算法定量分析及比较 29
    , h5 n& |. Y: B4 E9 K4.1 性能及收敛速度 30
    ) M5 [; X" O* j8 \% C( \4.2 多次运行的一致性 307 c& {* }$ V6 ?) X' Q, {
    五、使用案例 31
    / A, T' C& X' W5 b7 R" ~5.1 Bilibili 视频评论数据 31
    . Y# ]5 J- g: Q7 k) C5 J7 p( {5.2 外卖用户评价数据 39: x/ _" n* d& [$ `
    5.3 新冠病毒新闻数据 44% l, _" o% c1 t- s+ {0 n0 m  ~" U
    六、总结及展望 475 R7 t$ E  h7 p1 |+ ]
    一、绪论+ b3 ~3 i' t4 }( p$ `
    1.1 本课题的研究背景和意义
    - M# P6 _" v" r7 X% R7 E6 e2 t近年来,随着网络以及社交媒体的不断发展及活跃,互联网上每天都会产生大量的文本信息。在这些海量的文本数据中,对其进行分析并提取出有价值的内容,并由此指导用户的决策过程一直是研究的热点领域。在解决这些问题的各种方法中,主题建模方法(从文档语料库中发现语义上具有意义的主题)在数据挖掘/机器学习和可视化分析领域得到了广泛的应用。顾名思义,这是一个自动识别文本对象中存在的主题并派生文本语料库显示的隐藏模式的过程。  n1 ]4 z% f# @& u* G
    主题建模不同于使用正则表达式或基于字典的关键字搜索技术的基于规则的文本挖掘方法。这是一种无监督的方法,用于查找和观察大型文本集中的一堆单词(称为“主题”)。主题可以定义为“语料库中共同出现的术语的重复模式”。6 J, K9 b, p6 f4 n# y
    主题模型有着广泛的应用场景,特别是对于文档聚类,组织大量文本数据,从非结构化文本中检索信息以及选择功能非常有用。例如,《纽约时报》正在使用主题模型来提升其用户–文章推荐引擎;各种专业人士正在针对招聘行业使用主题模型,他们旨在提取职位描述的潜在特征并将其映射到合适的候选人。
    5 \5 a, m" s( s! i  e! J2 \5 J3 @在本节接下来的内容中,我们将回顾主题模型的发展及研究现状,提出目前存在的问题,以及本课题的研究意义。- H+ M' l5 O( O' d3 x+ W
    1.1.1 主题模型的发展及研究现状1 x6 [$ g$ j, w) R# ~9 ~/ }/ w/ p
    早在上世纪 80 年代,研究者们常常使用 TF-IDF 方法来进行文档的检索和信息的提取。在此方法中,先选择一个基于词或者词组的基本词汇,然后对于语料库中的每个文档,分别统计其中每个单词的出现次数。经过合适的正则化后,再将词的频率计数和反向文档频率计数相比较,得到一个词数乘以文档数的向量 X。向量的每个行中都包含着语料库中每个文档的 TF-IDF 值。
    , p% N) a( D: _: `9 p# m但是整个 TF-IDF 算法是建立在一个假设之上的:一个单词出现的文本频数越小,它区别不同类别文本的能力就越大。这个假设很多时候是不正确的,尤其是在引入 IDF 的过程中,单纯地认为文本频率小的单词就越重要,文本频率大的单词就越无用,显然这并不是完全正确的。其也不能有效地反映单词的重要程度和特征词的分布情况,因此精度有限。/ A& y5 F# w2 M0 b* y3 F; [
    上世纪 90 年代,Deerwester S 等人提出了潜在语义分析(LSA),它是一种用于知识获取和展示的计算理论和方法,出发点就是文本中的词与词之间存在某种联系,即存在某种潜在的语义结构。Thomas Hofmann 于 1999 年提出了概率潜在语义分析(pLSA),并在文中描述了 pLSA 与 LSA 的区别,即 LSA 主要基于奇异值分解(SVD),而 pLSA 则依赖混合分解。他随后进行了一系列实证研究,并讨论了 pLSA 在自动文档索引中的应用。他的实证结果表明 pLSA 相对于 LSA 的表现有明显进步。
    ) u* c, r7 t3 O8 R$ ~# Q年,潜在语义分析被 Jerome Bellegarda 提出使用在自然语言处理中。2003 年 Andrew Y. Ng 等人在论文中提出用于 pLSA 的 aspect model 具有严重的过度拟合问题,他们提出了隐含狄利克雷分布(LDA),这可以看作是结合了贝叶斯思想的 pLSA。LDA 是目前使用的最常见的主题模型。
    , l8 E. z; c" q7 \) A' f: w8 C年,针对当前主题模型可以为文档构建可解释的向量表示,而词向量在句法规则方面十分有效,Christopher E Moody 提出了lda2vec,一个学习密集单词向量的模型,将上述两类模型的优点结合。他们的方法很容易融入现有的自动微分框架,并允许科学家使用无监督文档表示,同时学习单词向量和它们之间的线性关系。$ ^6 a2 L" {. s
    1.1.2 目前存在的问题
    8 |7 V+ T# V$ y* ~6 a! o8 t& D主题模型是一种无监督学习模型,其结果的好坏取决于所选的模型参数和训练集,并且具有很高的不确定性。用户通常无法在训练过程中对模型的结果进行修正,特别是一些专业领域,用户无法向主题模型提供一些领域知识来提高主题建模的质量。已有一些研究提出,通过对主题模型添加约束的形式来解决这一问题。Hu 等人提出了交互式主题模型,然而主题建模的结果十分不直观,从中寻找不恰当的结果并添加合适的约束非常耗时、费力。同时现在基于概率图模型的主题建模算法结果往往具有不确定性,很难引入用户的交互反馈操作,比如让用户交互调整建模的中间结果和参数,从而得到更优的主题分析结果。! q0 `4 _! U: s6 p4 G
    1.1.3 本课题的研究意义$ x, m0 i9 @% V3 }8 \4 k0 J8 C; ~
    本课题的研究意义是对当前主题建模算法进行优化改进,解决算法结果的不确定性和用户交互反馈引入的困难性这 2 个问题,将可视化分析技术与主题模型相结合,提供有效的交互手段,让人们充分参与到分析主题模型的结果中来,利用人的认知能力,从数据中挖掘有效信息,达到基于用户驱动的文本主题模型交互优化。; k$ Q) z6 w! d" U6 q0 t4 e6 r
    1.2 研究内容和主要工作
    7 j  r3 d% h, j2 F2 q7 r3 o$ n本课题的研究主要内容是在国内外交互式主题建模的研究基础上,设计一个面向文本分析的交互式主题建模可视化分析系统。本文转载自http://www.biyezuopin.vip/onews.asp?id=15157系统使用非负矩阵分解(NMF)来作为主题建模的主要算法,并参考了 Jaegul Choo 等人提出的 UTOPIAN 主题建模可视化分析系统,对其中的半监督非负矩阵分解(SS-NMF)算法和多种用户交互方式进行了实现及改进。4 g! m- C# t$ @% Q( U
    本系统采用 Web 技术,后端算法和接口使用 Python 语言和 FastAPI Web 框架编写,前端界面使用 React&Ant Design 框架编写。主要工作可分为以下几个部分:! b& B- c+ L; ~
    语料库的爬取、收集和整理,并写入数据库。
    ) Q3 n+ B/ A! s" I- Y# }# W后端主题建模相关算法的编写。% q" b  v- r' J7 U& n
    后端 Web API 接口的编写。  O% A- K1 n& o
    前端界面的设计。
    $ `% E9 n( R1 o" n+ c  J前端与后端的对接,调用后端数据接口,并将数据可视化。+ p# g' p. x( R/ r0 A/ i) d& e
    前端用户交互逻辑的编写。* J1 H! m! a9 c' h
    : F8 O1 v# Z7 E6 J0 h5 ?+ u
    from typing import List
    ) v, |4 G) l. d; [  N; I
    : @8 [4 ^% U6 C6 [6 Wfrom fastapi import Depends, FastAPI, BackgroundTasks, Response, Cookie, HTTPException! v& E  v9 S* ]4 X7 v
    from fastapi.middleware.cors import CORSMiddleware
    0 U: n. w1 X4 b1 Pfrom sqlalchemy.orm import Session
    ) r* N9 \0 M* j0 kfrom typing import Optional, Dict
    9 q( ]9 D4 Y# `import crud, models, schemas
    0 H% h9 R  @" a# s+ V' zfrom database import SessionLocal
    , u' r! N; A7 Pfrom TopicModelTrainingTask import *
    ! l  e) }) P: F, t  H% {import uuid4 y1 T$ @: y: @
    + h: p2 G& ~; Y5 `; H% K
    app = FastAPI()- a2 b1 E4 z. f) D

    / w) H! q% Y+ X, L; h2 B# 配置跨域问题! l! I7 L0 ^+ @* |
    origins = [7 x& _  N0 F2 }* ]# F6 J- C
        "http://localhost:8000",! }7 E: _4 f% ]2 @
        "http://localhost:3000"6 S0 L% j, \. D1 [
    ]% }0 i/ K0 g3 K" g! V* T' ^4 J
    0 k+ m% \1 R. a1 C( ]
    # 全局主题模型训练对象7 Q  F, F& V2 A, [; S6 [  s
    topic_model_training_tasks: Dict[str, TopicModelTrainingTask] = {}+ l7 _" E, x( D5 I% q
    8 J; ]5 X2 B/ Q& x; w
    app.add_middleware(
    ( {; Y' C8 [4 J1 i) ]. ^    CORSMiddleware,) z0 k" m3 f* R  O
        allow_origins=origins,& x# G5 T# m5 g0 v& U
        allow_credentials=True,0 g; v: y( y- Z' ^) j# w
        allow_methods=["*"],0 ~) F5 c2 D! B7 A7 S
        allow_headers=["*"],: B9 z3 _( K2 K) N* B
    )
    5 d6 m: z: M- G! Y5 V0 C! i3 F6 p5 ~; T
    4 v. b' w5 C- Y8 E& X6 |  B
    # 数据库连接依赖; V3 Z. R" A2 Q4 p
    def get_db():
    9 I" b* }$ c# }6 B; b    try:
    2 e. Y8 I4 ~- V. q* e+ z        db = SessionLocal()
    , p4 Q. f; u8 |& B& j        yield db
    * ~& o! F( A5 r5 G/ w" b    finally:4 P6 t+ {6 L* ~; B4 b
            db.close()8 r* X4 F8 \  d, B& e, t
    0 v# g' S! e. e: E5 g5 ^4 a

    : m  B5 x  o" S@app.get() V2 t, C3 v5 r0 }  B% S/ ]
        "/bilibilivideos/",* B# w1 L0 L" T" q+ f+ _+ }
        response_model=List[schemas.BilibiliVideo],
    3 [% {! }3 p; f6 }6 m" B& A    tags=['bilibili视频评论语料库数据接口']1 T2 ^  H. k; t8 |0 o2 S' J
    )
    $ Z& C, [' G3 z1 g' A! v/ @0 Edef get_bilibili_videos(db: Session = Depends(get_db)):# M; g( O: z& d( P5 C0 d$ x9 ]
        db_videos = crud.get_bilibili_videos(db)2 f$ F  Q2 B$ P- ^
        return db_videos1 K# A* N$ z) W! K3 @3 S

    ( T7 G7 o) H, s+ h: e$ \, M8 {! h
    @app.get(6 A' ?+ o/ }; B9 x6 b& e
        "/bilibilivideos/{vid}/",5 a5 J7 c  C: B( f! L
        response_model=schemas.BilibiliVideo,+ s9 i& b5 w) T
        tags=['bilibili视频评论语料库数据接口']6 M" }! L2 ]3 E+ o4 a! w) Y
    )
    6 O5 y  y/ A4 p' j. b2 L7 A7 P6 m+ odef get_bilibili_videos_by_vid(vid: int, db: Session = Depends(get_db)):, c+ m, U. O" U% z6 H& V. Y
        db_video = crud.get_bilibili_videos_by_vid(vid, db)" [6 i8 Q1 ~' F* r
        return db_video0 Q2 O0 ?: y/ R8 C' s0 P

    ; o: B; f# i& C. [  [% G
    : B0 p6 ]8 n' X- r- P@app.get(
    8 d0 k) C1 x  |    "/bilibilivideos/{vid}/comments/",- L5 X5 `/ R! U# s. f
        response_model=List[schemas.BilibiliVideoComment],% j: Y9 e0 Q5 D6 ]
        tags=['bilibili视频评论语料库数据接口']; G. r; |: k0 k/ ]$ C$ p& q
    )$ p  j- a; W  H# c/ C1 Y
    def get_bilibili_video_comments_by_vid(vid: int, db: Session = Depends(get_db)):6 t7 I3 C+ U  Y1 s5 m  n) r' y
        db_comments = crud.get_bilibili_video_comments_by_vid(vid, db)$ C" D$ b* B: h& Y: R
        return db_comments
    ) m! Q- ^) b* E; ^2 P3 U' [: ]0 O& `% G& l6 ?
    - u4 S' l) @8 `- ?; U3 G! {
    @app.get(
    - s3 e+ |/ I* w$ Z' k) H    "/onlineShoppingReviews/",
    + n5 R1 K4 c( A' Y    response_model=List[schemas.OnlineShoppingReview],
    # g$ }, L) C1 u0 R9 N    tags=['电商购物评价语料库数据接口']
    4 H# j0 E8 `. b: |& D)
    6 a  f! F* k& x" o$ U. odef get_online_shopping_reviews(db: Session = Depends(get_db)):/ R+ b' q4 @) o$ _
        db_reviews = crud.get_online_shopping_reviews(db)
    2 Y- t: i* o" g6 m, x    return db_reviews
    0 e/ n( i* r9 X" _  \
    * Y2 V' ]/ j, P. t; \: n; P9 y0 Y4 e! S0 `) c
    @app.get(" K: G9 R4 Z1 \8 M: w0 d- w- A- l
        "/takeawayReviews/",
    # a3 I1 V% u" H" S0 G; P9 k    response_model=List[schemas.TakeawayReview],
    * K. d  C) B! B5 \    tags=['外卖评价语料库数据接口']
    0 T% ]$ ^/ U" V: t7 H2 R)
    * ~. T# {& h! _  H; I7 `7 r  xdef get_takeaway_reviews(db: Session = Depends(get_db)):* B* Q- n. ?$ }. c* D& I- t
        db_reviews = crud.get_takeaway_reviews(db)- M& C: p; [/ ]& E" N& G
        return db_reviews$ N; ^+ Q* W; @; }+ n: g# W
    % _7 d( r0 b; b& I  H4 }

    ' G2 \/ F& I, T+ ^/ y6 |2 G@app.get(
    1 h+ f5 t5 ~, t4 j    "/chineseLyrics/",) c4 n, K( V$ w* E
        response_model=List[schemas.ChineseLyrics],  U. `$ X- V: |: X: @! v2 k
        tags=['中文歌歌词语料库数据接口']
    ' ]% j( c; l" V8 B)
    9 Y* E9 ]+ n% ]# G9 E* {$ W- Ddef get_chinese_lyrics(db: Session = Depends(get_db)):+ H9 U: u& H" _, f
        db_reviews = crud.get_chinese_lyrics(db)
    $ z% d  l9 g* k1 O: ?& S4 w+ m    return db_reviews
      D" a: j. O! U& Y5 d5 z9 R" g; I0 v9 b% M2 b: {# b( b
    / p2 V8 X( \0 S2 ?9 V
    @app.get(6 N9 x) S" w0 x
        "/COVID19News/",* v& p) R1 u! L2 v4 B( N
        response_model=List[schemas.COVID19News],1 z& y9 i1 T1 |& e8 q
        tags=['新冠病毒新闻语料库数据接口']
    9 m$ U$ w; K$ B) L)& C# M7 y0 I0 S& w# O& c2 e; J
    def get_COVID19_news(db: Session = Depends(get_db)):2 V4 R; e# S7 _# ]* V
        db_reviews = crud.get_COVID19_news(db)% q( {. E( Y' S5 Y9 I6 X* `: b
        return db_reviews
    * D5 d/ X9 x- |9 x3 a7 e5 C! C8 \
    4 \' l5 h# E1 o. ]( j: z
    9 R% S# g6 y& A  E4 Y$ n# 为每个用户设置uuid标识符,并存入Cookie9 c  t* Z6 [5 U; `
    def create_user_uuid_and_set_cookie(response: Response):! P5 }; l5 J7 y. \# m' }3 t9 e  j3 X
        user_uuid = str(uuid.uuid1())
    2 [3 N, ^. r( n/ y8 B    response.set_cookie(key="user_uuid", value=user_uuid)
    7 r/ x# L, ?1 B% |1 ]- y# z    return user_uuid
    8 B2 @& H* j# ^3 ~8 J! v! n) [7 [# x/ h
    ' U3 C: `/ Z) ^( y. Z/ E
    @app.post(
    ; U- u/ L) I- g* I2 B" V+ G3 L, h    "/calculate/preprocessing/",
    8 V( ~% `, I3 n: O# |; P. |    tags=["计算模块数据接口"]" u2 L0 `. K" T+ ?) R8 S' l7 s( T
    ), i, z1 J. r/ L& Y. }9 w
    async def run_text_preprocessing_task(text_preprocessing_params: TextPreprocessingParams,& O) ^9 C, n# y" x9 k
                                          background_task: BackgroundTasks,
    : o' q2 C9 h1 I4 g- P/ [                                      response: Response,
    4 W; n+ g2 c2 |1 n7 Z) ?6 z                                      user_uuid: str = Cookie(None)):
    3 `$ K; g; g- q7 i$ B+ J9 ~    if not user_uuid:; A  w/ c! L/ I& K
            user_uuid = create_user_uuid_and_set_cookie(response), V2 S9 g) z5 V8 c

    , e' K: B' g0 W  r9 f+ O; E    global topic_model_training_tasks$ x# L+ z2 k) p4 s4 K; h+ D, A
        task = TopicModelTrainingTask(text_preprocessing_params)
    / |  ?0 g0 r; w( a2 G2 t  b9 D- Q    topic_model_training_tasks[user_uuid] = task5 J8 o0 p# [1 A6 Z: q0 V

    + f* R$ m; l: L3 Y: M) T  |4 X    background_task.add_task(task.preprocessing)- M& M  L9 q; I
        return {"message": "已添加文本预处理后台任务"}
    & s# c$ {2 D$ f- x# m$ u7 F7 R
    " p; u0 u/ s3 d6 s
    * W0 D5 \, D9 N- r& g@app.post(* J3 @/ M* f) |
        "/calculate/nmftraining/",
    7 w1 Z1 o: t: w( P    tags=["计算模块数据接口"]
    4 r5 B) @+ o- q3 K+ e)
    ( t5 j2 f! y: F. Wasync def run_nmf_training_and_tsne_task(nmf_training_params: NMFTrainingParams,3 H: z9 s) z: I$ j$ P+ _9 H- q
                                             background_task: BackgroundTasks,
    5 H# V1 O' D% i9 ^5 M& _/ d$ p7 f- U- N                                         response: Response,& G' X3 O8 }; U: b, i" R0 e! ~
                                             user_uuid: str = Cookie(None)):
    ) c  h1 @( w( E2 l3 j9 M# U1 D1 q    if not user_uuid:
    3 i3 e+ z5 Y2 U! U' d& H  F        user_uuid = create_user_uuid_and_set_cookie(response)
    4 U( P7 o3 |: m0 b7 y' Q8 U& F& c& _4 d# m" ]6 U' I* s
        global topic_model_training_tasks7 E- l# A. x, Y1 N* v- R# O
    & h* O9 D8 R1 Y9 M" T0 S
        if user_uuid and user_uuid in topic_model_training_tasks:% ~" d9 C" |4 S5 V# `+ o1 Z7 S3 H2 m
            task = topic_model_training_tasks[user_uuid]4 r9 B) s: y4 ^5 e$ e) {$ n
            if task.text_preprocessing_progress.status_code == 2:$ g2 ^4 N9 T1 ^9 n
                background_task.add_task(task.nmf_training, nmf_training_params)7 W; N$ t, `+ @. ^, a* A
                return {"message": "已添加NMF主题模型训练后台任务"}
    5 _2 ~  I, A4 |+ h: M        else:+ a2 C- o) t. C% M& N
                raise HTTPException(status_code=404, detail="请先进行文本预处理任务!")
    7 u5 A+ W2 J& N# e. }    else:
    : O5 h1 h/ I, |; t" U; y        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    # B- y# Q  Y7 T8 A% m! ~2 E5 c9 ]: I, G2 P
    , I2 r- A( D. O$ |2 [
    @app.get(% \6 E& K! `! V0 F5 K! B
        "/calculate/nmftraining/keywordsearch/",! `1 T% s4 m* I4 Y- i1 J& b" N
        response_model=KeywordSearchResult,2 c" M1 e3 M, D+ A! n. j' w
        tags=["用户交互"]* P- A# c3 b2 u, R6 V
    )
    " Y; M. r  [9 n0 V/ Z* N3 c- dasync def search_keyword(search_text: str,
    3 Y3 l2 o/ l% \: v' ?7 l                         response: Response,
    " g- u+ v+ P6 y+ z1 q: n  c# e1 }                         user_uuid: str = Cookie(None)):; O0 {" g* Y, Z# r4 i9 H
        if not user_uuid:
      P' L7 }0 f9 {3 G# j        user_uuid = create_user_uuid_and_set_cookie(response)# j& a2 G# a7 `% l2 W% w) O

    3 `! `# E2 x; w    global topic_model_training_tasks
    / e; U$ d  _# U0 f: r; H3 M. ]$ A) G7 F7 p( q$ K
        if user_uuid and user_uuid in topic_model_training_tasks:
    / B/ Y. p# e) n        task = topic_model_training_tasks[user_uuid]
    0 s8 e, j) |, u9 s9 B' q        if task.text_preprocessing_progress.status_code == 2:0 B% A. t% ^  a; c
                search_text = search_text.lower()6 r( W8 m! F2 E5 ]
                bag_words = task.text_preprocessing.bagWords# Y! E; F' V( M4 c) ^* M
                keyword_search_result = KeywordSearchResult()
    1 U7 |, g! g& t0 A7 ^            for wi, word in enumerate(bag_words):
    6 _+ p+ y9 w& _. r& }                if search_text in word:. h) v4 V7 f* n$ b, S$ q1 M
                        keyword_search_result.word_id_list.append(wi)
    / i% X) f+ f. Z                    keyword_search_result.word_list.append(word)6 S+ _% y3 ^# X
                return keyword_search_result
    # h) c/ _- c4 s        else:
    , [$ W$ a1 Q5 E            raise HTTPException(status_code=404, detail="请先进行文本预处理任务!")* J  M3 i' K8 C  I
        else:5 Q4 [# Y5 n6 {4 D1 E: v
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")- g' I6 z7 Z8 e& l/ Q

    1 V. I/ @. E- A, V. q8 D1 s* _: k: q3 ?2 i; E9 A; {
    @app.post() E& w  Y, \) [8 Y
        "/calculate/nmftraining/topickeywordoptimization/",
    0 z9 w9 W& Q, a$ t; H1 ^# K    tags=["用户交互"]* A# H8 O3 H" s4 y
    )" C$ t4 i7 j& d; f. Z' N
    async def run_topic_keyword_optimization_task(tko_params: TopicKeywordOptimizationParams,: U! ^; `+ H; U
                                                  background_task: BackgroundTasks,  n. Z' }, [$ y$ p( [+ L
                                                  response: Response,& F, G% \7 |$ A$ U/ {
                                                  user_uuid: str = Cookie(None)):
    7 C2 @, q% e% ]! N9 F3 A$ G" G    if not user_uuid:
    " H" U2 _+ r- C# S2 E9 g        user_uuid = create_user_uuid_and_set_cookie(response)
    9 ?6 ?8 `1 Z" L8 P5 R5 J$ _
    % [2 G, H# E5 C% i, o% v2 s' Z' i    global topic_model_training_tasks
      o9 \. h0 J8 t2 s% o
    / [- _# u$ ?' n! U    if user_uuid and user_uuid in topic_model_training_tasks:
    * n6 J! ~+ q6 \" i! d8 f        task = topic_model_training_tasks[user_uuid]
    + ^3 {0 k" e0 d8 p# G" }" y* f& M        if task.nmf_training_progress.status_code == 2:
    9 f* |' q9 h+ ]: C8 G7 S            background_task.add_task(task.topic_keyword_optimization, tko_params)
    $ d- m& s$ d8 X& m3 X            return {"message": "已添加主题关键词优化后台任务"}% B+ f" Y# K# F0 L+ l9 h
            else:
    8 i$ B1 e2 b9 ]2 t            raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")
    3 O; P8 d% s3 U7 P: V; q% g1 l    else:* |) x1 n/ ^& ?
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")/ @- c& @$ D6 K# S' w
    4 [/ d+ @( m& X4 X

    ) ?8 A  e& {! `$ y0 k$ @5 C( \3 S% T@app.post(# }6 L) e- k2 }2 W
        "/calculate/nmftraining/topicsplit/",& ^6 i( C* k- I' e7 _  @
        tags=["用户交互"]% m) A; z4 ^$ G- |4 s
    )" x* R. T, |4 _& A/ [. c# Y' Q- Q
    async def run_topic_split_task(ts_params: TopicSplitParams,9 E. d/ c+ \# M7 l- v" X
                                   background_task: BackgroundTasks,
    2 X4 t6 j1 `# Z1 _7 n                               response: Response,6 o  F3 P6 a# g; C. x, e
                                   user_uuid: str = Cookie(None)):
    - |3 r: V. e( _# V    if not user_uuid:
    ) _$ a/ T# G* _2 s* Z        user_uuid = create_user_uuid_and_set_cookie(response)5 O! [" O% J2 b4 _' A' P
    6 B* o2 l+ B) [/ h# @# d0 `" P: m
        global topic_model_training_tasks
    ; s" Q! Q' b9 y. {
    6 Q% p# M. d& y% [2 M, H9 S* I    if user_uuid and user_uuid in topic_model_training_tasks:
    - p' G+ {: s& T' u        task = topic_model_training_tasks[user_uuid]
    / u9 f+ M- l3 w/ x! a7 B: A        if task.nmf_training_progress.status_code == 2:
    ( e7 ?* p0 \4 A" M0 r4 L6 ~            background_task.add_task(task.topic_split, ts_params)1 W$ t5 N0 Q& I2 J  l3 V7 e
                return {"message": "已添加主题拆分后台任务"}
    8 F: d( e! ~) {6 p/ ~  ~# |* C        else:
    / N/ [$ y9 G3 e5 u7 I% S+ e8 w8 O5 x            raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")9 a) o; x8 j- K/ b& }; |" |( W
        else:; a4 q* z4 ~/ N3 U
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    4 y% D" C! y: D: [
    9 }7 H! x2 ~( J% I
    / h6 f# B  V* [9 E* t7 S@app.post(: A- k* [% b, M& I  ?
        "/calculate/nmftraining/topicmerge/",3 e! \" A. b! Y- {, w
        tags=["用户交互"]% o8 g5 G1 x8 K1 `0 F
    )6 I0 o3 E6 ]. C- v1 ~; R
    async def run_topic_merge_task(tm_params: TopicMergeParams,/ G: a" |4 i/ ~# O6 j4 c' y
                                   background_task: BackgroundTasks,
    7 M# a8 ~% X1 X' _0 R                               response: Response,
    4 z3 B$ P: ^4 ]! m                               user_uuid: str = Cookie(None)):2 t0 _2 ~( {5 n
        if not user_uuid:; A% I: v9 _  e
            user_uuid = create_user_uuid_and_set_cookie(response)
    . ^7 S5 F5 v. j* ^  H0 j, S6 F& V" q& ^4 c8 X
        global topic_model_training_tasks
    4 C" q8 e7 L- S3 h+ i9 O  W% J0 e- W
        if user_uuid and user_uuid in topic_model_training_tasks:: h( u; u5 S1 e1 H
            task = topic_model_training_tasks[user_uuid]
    : g  S% ~$ u/ p7 V, X: ]) o* B+ q        if task.nmf_training_progress.status_code == 2:
    , x. e  Q# n$ `* {# ]            background_task.add_task(task.topic_merge, tm_params)  {" f7 M, O" ^) r. a
                return {"message": "已添加主题合并后台任务"}
    , {8 ]2 i# \: y$ h& Q! l        else:
    % U2 ]- t5 f9 g            raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")& U% \% r8 g$ N: B' }7 U9 q9 V: K
        else:
    5 Q0 b7 q$ J$ c2 L3 P' n        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    ) `+ p/ s! N$ n$ d( P/ K6 ^  Q3 [4 c4 |' r
    . ?; ]" J9 }4 s7 s
    @app.post(
    9 t: X  \" N1 E: A$ f7 ]( i2 M& p    "/calculate/nmftraining/keywordinducedtopiccreate/",
    0 F5 X3 M6 V& f' [; V    tags=["用户交互"]
    ( l7 r5 ^) S" i9 o  W1 D5 U)1 u, P- ]; D& w* k0 I
    async def run_keyword_induced_topic_create_task(kitc_params: KeywordInducedTopicCreateParams,
    ' h$ E; {2 r' x( R                                                background_task: BackgroundTasks,- m! U' n( J9 r+ l- q
                                                    response: Response,
    8 j: A& d  b- B# I7 O, _                                                user_uuid: str = Cookie(None)):/ A( @* A0 A3 n! o
        if not user_uuid:& T1 G, r7 e4 Q- h3 i
            user_uuid = create_user_uuid_and_set_cookie(response)
    3 F1 m% A* h; K
    7 I: G# S& j5 R/ W    global topic_model_training_tasks
    7 A. P( p! v9 p# x" Q) i* W5 S' d  c5 @. B
        if user_uuid and user_uuid in topic_model_training_tasks:' P/ }9 [- k- A. D
            task = topic_model_training_tasks[user_uuid]
    0 p: q" j- l6 M$ i* \1 I        if task.nmf_training_progress.status_code == 2:
    2 u) I2 z( w% S5 {) n3 [% n            background_task.add_task(task.keyword_induced_topic_create, kitc_params)
    5 N, l: K# v& V9 n: h3 \            return {"message": "已添加关键词诱导主题创建后台任务"}3 |( ]# G5 P# b' R) M# C: s
            else:5 U1 L4 C9 G9 q8 s- U1 Z
                raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")
    : j+ m7 Q; ~# G. b: O& V    else:
    - [) O& Y- w* T/ y  D        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    $ _  Q  X& B* n. L+ j8 H, q/ h5 G8 `" P8 F
    9 C) d, g! I: L- G  Q4 u
    @app.post() Y" S3 D, r* ~) V
        "/calculate/nmftraining/documentinducedtopiccreate/",$ ?: G, g, y5 s' h) k
        tags=["用户交互"]
    : c% O" d( U! B)
    ; F/ T* |0 R* J, n+ o' L1 Rasync def run_document_induced_topic_create_task(ditc_params: DocumentInducedTopicCreateParams,
    . z2 {3 K/ r$ t* P                                                 background_task: BackgroundTasks,; L1 A) n2 A# g0 v: ]; U; U
                                                     response: Response,
    1 ~1 y; a# a7 G6 Z                                                 user_uuid: str = Cookie(None)):  G  r( d  o/ e8 h: H# p
        if not user_uuid:
    % E. r/ f* ~9 g* W% H7 ~        user_uuid = create_user_uuid_and_set_cookie(response)$ n7 g, _; V) S, i+ H1 \0 z
    + q5 W3 i/ V% S5 E
        global topic_model_training_tasks' L0 s6 a1 _0 A
    ; d2 Y8 e# n$ ~* h8 g' z& i
        if user_uuid and user_uuid in topic_model_training_tasks:- S3 h, _! S* _" X0 S& f
            task = topic_model_training_tasks[user_uuid]& p, f8 U% f1 T6 b( U$ _$ U! |
            if task.nmf_training_progress.status_code == 2:
    ! t/ s& o/ N% [# k8 w/ y& B            background_task.add_task(task.document_induced_topic_create, ditc_params)
    / m8 q9 t& M4 A& T8 I' D2 N9 G            return {"message": "已添加文档诱导主题创建后台任务"}
    " d: K1 d6 @8 ?0 a4 n6 f        else:1 u* |  y  d6 u! R' ^
                raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")/ ?& b* i; \# r# b+ T( i6 E9 i
        else:: @( @  [# g$ K! c
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    9 P: D6 M; U$ U* Z8 r" U% C2 h3 }0 Q* u" n+ q
    2 ?1 E( W1 t5 `) Q
    @app.get(, S( e; n1 Z3 T; f, K8 W' O
        "/calculate/nmftraining/newdoctopicdistributionpredict/",- e9 H9 S8 M1 B; Y: Y- `. o3 s
        tags=["主题预测"]. x( [! o: A$ q4 n4 K. v6 U
    )
    5 G. B7 E+ V! U' i" b8 Y1 z9 V& Pasync def run_new_document_topic_distribution_predict_task(9 E; `, p) P" M
            new_doc_text: str,9 i% p5 R" q5 }5 X( n" N
            background_task: BackgroundTasks,1 P2 f+ T, ?' Z) U
            response: Response,
    ( U; D2 T& R9 H& E, u        user_uuid: str = Cookie(None)):" y7 L7 W! \" i2 Y! Z
        if not user_uuid:
    $ f. z3 _/ s. h3 ?) ]: B        user_uuid = create_user_uuid_and_set_cookie(response)& z+ J: F  [2 m  Y; M% E* X

    : V. `& T: w6 S8 J6 _1 T    global topic_model_training_tasks
    3 E3 V$ G+ p" f5 S5 T9 z4 {" D$ B  Z: i( \0 {
        if user_uuid and user_uuid in topic_model_training_tasks:7 ?, g3 K9 z; g& a) X' H/ l, m
            task = topic_model_training_tasks[user_uuid]& H: n! J' T1 F
            if task.nmf_training_progress.status_code == 2:) ~4 U  K% P/ _9 z& B6 g
                background_task.add_task(9 l/ o3 u* I* W' h
                    task.new_doc_topic_distribution_predict,3 v, u; D5 X2 X0 G  v
                    new_doc_text
    8 J" ^& U8 u6 I: W% t# T+ u2 h' K            )
    7 K- T( @/ G6 U; D$ ]3 j            return {"message": "已添加新文档主题分布预测后台任务"}
    & B# [$ h- ^+ n+ L& j. E8 e        else:% |4 \% c5 m7 w4 d: l- G
                raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")
    , O3 N  t6 g+ @& G3 j6 y    else:) z4 K/ O2 j8 q6 a* K. b4 y
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!"): [! g4 @- f3 V  H7 ]& y1 r. T; ^& {

    + d  J( R3 L7 \1 I5 f' ]. u/ {1 }" [: }  `
    @app.get(# H- Z6 T6 k: \" U) K; A
        "/calculate/preprocessing/progress/",1 c! r7 b. R3 w  o& m, _
        tags=["计算模块数据接口"],# e+ y/ ^" z# T1 F0 J  b
        response_model=TextPreprocessingProgress& ]: }6 z; s' L6 a  ?
    )
    * ]$ @! X& g" \/ c4 h( adef get_text_preprocessing_task_progress(user_uuid: str = Cookie(None)):0 G0 N  ?5 x/ w, L- X) I) X
        global topic_model_training_tasks
    % Y: E) P3 V" l) ~  r: U  L! C' r; s9 w0 l; g% s
        if (not user_uuid) or (user_uuid not in topic_model_training_tasks):
    3 M+ m) n* m3 \' M4 D        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")0 V5 x: P. ~* E3 j7 D+ `& ]5 f
        return topic_model_training_tasks[user_uuid].text_preprocessing_progress8 j6 L' K" C" h) X- }

    ; v) c& C1 e: y6 n9 r4 v; T, P2 [
    ; S! i( ?" ~$ M2 N3 Q- I@app.get() q& j6 P* ~6 h: R; F$ H( R
        "/calculate/nmftraining/progress/",1 K/ F$ |( ]0 r# Y/ S" p
        tags=["计算模块数据接口"],  f& H, s4 R# C
        response_model=NMFTrainingProgress5 [6 [" n. G% [. I: O% c' ^
    )
    / B% G9 R' E, s) K! udef get_nmf_training_task_progress(user_uuid: str = Cookie(None)):
    0 x% N3 |8 h/ b1 ?    global topic_model_training_tasks1 S8 C' I3 D, j0 D2 s4 o6 g
    ( a# G" k' i0 Q7 @& w
        if (not user_uuid) or (user_uuid not in topic_model_training_tasks):
    5 |4 y2 O; `. D& `9 y; a        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")% o2 ~- w8 }6 F/ O1 X& h* I# J
    1 P8 j( ~- u" h" |; X" J" n' x; l4 B5 ?
        task = topic_model_training_tasks[user_uuid]
    ' q/ e2 @5 z2 E1 h' h8 P    if task.text_preprocessing_progress.status_code == 2:* H9 `# X3 G0 R& g- Z" i
            return task.nmf_training_progress2 g3 P. B: A" F9 z/ X3 [# h
        else:
    2 y: M  R1 A/ f" w4 _2 [/ h( G        raise HTTPException(status_code=404, detail="请先进行文本预处理任务!")
    1 ?3 {: o: ^: o
    ( c  W+ T$ w5 ~! N5 F9 t3 D3 O- `$ v" c
    @app.get(
    $ Q0 |. ^2 O/ E8 N4 g5 l+ p( j    "/calculate/umap/progress/",9 r. X# z  o" a1 w
        tags=["计算模块数据接口"],& e6 c$ t8 W, K% D* {
        response_model=UMAPProgress
    / I+ d( V* o- m, J  N)5 E; `8 h, K) L% V$ |! h/ n
    def get_umap_task_progress(user_uuid: str = Cookie(None)):5 U+ J9 F# \3 f) F- @0 N* y% s1 P
        global topic_model_training_tasks
    $ \# ^* `" r9 Z/ @& B3 {8 T! i
    4 l2 a. |: Y* M7 d    if (not user_uuid) or (user_uuid not in topic_model_training_tasks):: w/ f% X: A1 S+ G: @3 d
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")! A8 E: J: A9 Z5 M
    ! `$ h0 H6 L9 p; V% ?
        task = topic_model_training_tasks[user_uuid]5 i3 K5 ^, D8 g/ q4 F5 b3 e9 l
        if task.nmf_training_progress.status_code == 2:
    5 a: K6 c1 o2 M+ \8 w        return task.umap_progress
    3 z5 Y: w& I, z. B3 Y    else:% @6 I! _- ]$ J: u' v; n* h
            raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")1 K% t: w: \3 V
    - |& F2 k' @9 a' M, \6 h+ ]

    6 P: g2 c! U6 ~7 }@app.get(  @+ t+ D9 F4 @* b7 a
        "/calculate/nmftraining/predict/progress/",
    $ o& Z( t0 Y, I( A% ^    tags=["计算模块数据接口"],
    1 F' a" M2 }4 H/ S( t( _- L8 s7 r( h    response_model=NewDocTopicDistributionPredictProgress
    / |- }) C8 B1 y2 O2 ^)
    ) S, X7 k, V6 r8 t% N7 s3 P9 Ydef get_new_doc_topic_distribution_predict_task_progress(user_uuid: str = Cookie(None)):# v: K: F. r+ @$ E
        global topic_model_training_tasks, Z; z- R' z* n9 W) n

    0 O0 X; T( l7 M6 Q- T& \    if (not user_uuid) or (user_uuid not in topic_model_training_tasks):4 ~* g' e7 H: q
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    8 _; S& B1 P8 A7 C7 {& d
    , p/ z; W4 E( G: [2 G    task = topic_model_training_tasks[user_uuid]  F+ b0 s- t2 M- ^) K: S1 H
        if task.nmf_training_progress.status_code == 2:3 s3 B/ V: b( O8 E
            return task.topic_distribution_predict_progress
    2 ^% _# y; I7 z: B! J8 g' S" u    else:( A  r5 o8 a8 W% N4 b
            raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")
    5 w5 T$ W- @. U& _! Y# t) `+ u" F, I' u/ v' G  |

    , B( l# e7 S4 V  _( b1 e' r! G@app.get(& h" L! i" \1 `9 g+ P
        "/calculate/details/document/",; Z  J5 ?' |/ M) g; y$ h
        tags=["计算模块数据接口"],. b$ q2 F8 }% @5 D/ m
        response_model=DocumentDetails+ r% i9 D6 b: J- a5 J3 C
    )
    % h9 H! `/ h0 y/ O. ?+ i/ S- Ydef get_document_details(doc_id: int, user_uuid: str = Cookie(None)):
    6 d8 Q3 j/ `3 n# z2 B6 z1 B    global topic_model_training_tasks
    % n. ^/ y! _, Y- {) Q
    1 r+ J* q, J' Q$ p( q' a: w2 A    if (not user_uuid) or (user_uuid not in topic_model_training_tasks):
    . }, G, Z5 u) j/ z        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")3 x- `/ A5 N5 n* V$ P

    : d; ~9 q6 E9 N4 Z& Y! t2 U    task = topic_model_training_tasks[user_uuid]
    " I9 w1 v# @) I    if task.nmf_training_progress.status_code == 2:. P) z3 A, K9 c4 ^$ F; R
            return task.get_document_details(doc_id)
    6 i3 j% `- B0 Q5 R/ |# V: ?    else:
    . X$ N! A( q8 {- Y# f* K. T6 I        raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")( ~. f* u) ~1 S. Q9 M4 L# D# L
    & ^% j- a( m3 Y

    0 S# D' X" e% n- L) t- Z@app.get(& O) M' g3 o5 e6 o( r% n# F3 |
        "/calculate/details/topic/",% I2 A& R# h0 }8 H) ?4 n
        tags=["计算模块数据接口"],
    9 z' E4 I7 q% K7 W! b2 U3 w: f* P1 j" e    response_model=TopicDetails
    6 X- d4 F3 S; F" t  f) O( u" m)
    ( n) Y3 m( Q4 k3 J6 kdef get_topic_details(topic_id: int, user_uuid: str = Cookie(None)):
    3 D. r- n5 E8 ]  z0 {/ v6 L2 G    global topic_model_training_tasks7 M) D7 Q1 U& d4 Q- A& p  X5 S

    - ?- w# g" L6 x6 g  R( O- C% X/ H/ i    if (not user_uuid) or (user_uuid not in topic_model_training_tasks):
    ; `- Z0 o9 O0 g0 I        raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    ; H$ {+ d! F& ^9 y' Q/ j: _& ]& t
    5 k# g& \6 H& S! v/ K7 w    task = topic_model_training_tasks[user_uuid]
    + Z7 Z; O9 @! J: D2 o8 d    if task.nmf_training_progress.status_code == 2:
    : i# ^3 ~3 `8 Q9 b; v        return task.get_topic_details(topic_id)
    1 f  }7 _, d  N% Y    else:# c' w4 ^9 q8 K# \
            raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")
    . U1 `1 d" E' x7 \( N5 `1 n
    8 V- \$ M! j" K+ M1 \. i
      q% G# w% J7 N- |9 x$ g@app.get(, Y' ~% q# W5 @, X# R- k7 P) C( W
        "/calculate/userinteraction/info/keyword/",
      N4 {/ c( ~$ R) U9 P. j    tags=["计算模块数据接口"],
    6 K# C$ Q* x) m    response_model=TopicKeywordInfo
    ; E& n1 y5 b$ d8 Z" J$ u' [* @)7 E6 q  a) @3 h- P3 l. D6 ~
    def get_topic_keyword_info(topic_id: int, user_uuid: str = Cookie(None)):
    : j& B! s  m( N3 C8 w( X) u. r    global topic_model_training_tasks
    : o5 e  A+ G- N6 M$ b9 A7 S
    $ |. K+ n* P% K8 a8 w+ }    if (not user_uuid) or (user_uuid not in topic_model_training_tasks):. \% l8 k& a& J
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
    * p5 q* Y/ H7 p$ S1 N$ ?5 [2 |8 b# S# e
        task = topic_model_training_tasks[user_uuid]5 ^/ H; l& r+ Y, s7 a  _* p1 W# i
        if task.nmf_training_progress.status_code == 2:! @, x* m3 `& ~2 I. S) |1 H6 A
            return task.get_topic_keyword_info(topic_id)
    ) y: j/ b, G% `: [' B+ h" \- X    else:
    7 ?1 L% |' P' c        raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")) `2 x! _' G* L/ T& F
    1 X1 _- \9 t9 e

    ' m- R  n9 \( x7 X( A8 j" U$ ^@app.get(
    / S" `/ o3 C' e' g% A: u( H* L" y    "/calculate/userinteraction/info/topicmergekeyword/",& R! ^( i, \7 i  O0 w" c" F
        tags=["计算模块数据接口"],0 I) J, T, ]* J( B% E" s5 L+ R6 B
        response_model=TopicMergeKeywordInfo
    % P' B- V$ \9 N/ H- T$ q2 `)
    8 l( E  I) B, Wdef get_topic_merge_keyword_info(topic1_id: int, topic2_id: int, user_uuid: str = Cookie(None)):) Q8 Q8 Q$ M( {
        global topic_model_training_tasks
    , ^2 d/ v* w  A) c7 q3 [4 i& G- z
        if (not user_uuid) or (user_uuid not in topic_model_training_tasks):6 E  N2 I% P2 N. u$ }6 O& x9 U7 M
            raise HTTPException(status_code=404, detail="未找到相应的训练对象!")
      V5 X- }* ~. D
    - O, _% P5 c3 Q$ Y    task = topic_model_training_tasks[user_uuid]$ p3 i7 m7 Y0 ]; T) R3 F' e
        if task.nmf_training_progress.status_code == 2:, s- }* u* K% S$ l" s
            return task.get_topic_merge_keyword_info(topic1_id, topic2_id)- b. H# e# J; T& U. ]+ i
        else:
    ( m8 i8 W1 g' E1 ^( S1 o; P! m' t        raise HTTPException(status_code=404, detail="请先等待NMF训练结束!")
    . ?6 L2 g) {: Y) W! O& n; j% k6 C4 Y: J; [- [

    " L! V/ v/ _& S; s! Z1
    ) K  {. ]% f. R* J- K21 D' r' j$ l% J2 {$ D- F
    32 c; f, `3 C/ P' ?7 |
    4
    - m- Y1 p6 t4 K5
    9 F6 C& X" D. m, A% w65 P) K: S- ^7 i; }4 @) \" R$ f
    73 i$ Q& N! y2 o: n
    8. }& e6 Q. A! p3 Q
    9' X& @, p9 r; d. J
    10
    6 p1 F. x- q# ^! [8 ?' `& ^112 v; J/ t% p6 K) F" ~5 u
    12& D* `. r5 p8 a: R) w
    13# S6 T& @, r: |1 c- E, |9 x/ \
    14+ B0 H+ g7 D' f4 U. K) U
    159 U7 `+ z# H0 y5 g8 K
    16  _% Y- I  y+ s( \2 ?/ V
    17% H. Z- X" d- E( n. H" z6 E$ S8 ]6 O
    18
    ; o4 I0 b, U) Z194 e& a7 g: R% p( m) t
    20% C, `: e% L& t+ X  g) ]
    21
    ; w# f/ g5 J8 _. o3 X22
    9 _' H# _3 e9 e" P+ n239 T) M& O8 N6 [& I  U
    24
    4 H( }9 A# z% b" N25
    7 K' v6 n/ ~  m9 a' P' }# K26
    + v- \5 T0 |& B- R: j7 s( d27
    1 j( q$ ?- U5 O+ x28
    . |$ R/ F3 j7 T' A1 W29( Z' O3 z$ t8 R- W& U1 Y4 y" E
    30
    9 \' G( U+ P. g' z5 B$ I310 {) v# [8 P! W+ B9 D
    32
    $ z5 O, h* v: n33
    ) p1 c  q( B3 k( x% t0 v' i7 x( x+ X346 J# V4 a) g0 n# a
    35
    6 D1 e+ s  v8 Z2 L8 A8 Z$ C36
    $ l1 c# T% @" J37' V! N3 Y0 ?/ R6 V9 B0 R: j
    380 o0 W% T1 e; |' C! v
    39
    3 @' L. Y0 Y3 G40# {1 W! |5 ^: @6 o1 m# i& M' L
    41
    " v; ^9 s9 j" t7 [% ~42
    - k$ Z+ ?, ?1 y% n6 }) p4 @43; E/ D; b' b! ~
    44
    ! M9 n4 W! h- W$ l8 ?  S45
    ' V: p+ F$ ?# d4 k& F  q46
    & a% \7 X( r# i6 m  ]4 Q471 {& X$ q& }% X( {- R
    48. w3 L/ s$ _3 @' T, Z- N" v
    49
    3 w- c) Z2 X+ D3 O50# y7 w% t1 g% T4 [' a7 z
    51
    + [; x1 w( l. v+ g52
    / q7 a. n4 _9 l531 j$ |0 ~4 x, _, Q3 F
    54' J: q! [+ Y. W0 |
    55
    8 ]6 c: ^% ], }2 Z4 i563 O6 ]8 u* o3 w  h; {  l! h
    57/ }/ @; B: S% f
    58
    2 y& I" I6 b. }6 U; `59
    / p7 R# @+ n- c0 Y/ C60
    * q: R8 N! E) }2 d& i/ @4 R61
      H/ B6 a+ R% b% ]62% t+ x" k6 W  D
    63) ?$ r3 e( Q& i
    64
    " h! c. `! G) u65. o; G: s- g5 V: ?, |
    661 l: ~4 x& C2 e, K8 _$ A6 Y
    67
    & U2 e. x" `$ R. _+ V, H68
    + d) E& L. P% u+ ~69
    - r0 h+ Y  b: y& X$ n70: Q' O" y3 [) \  `: B: T" n
    71
    0 _* f9 y/ T( }' k, l, {3 s72/ W- ]/ L$ M5 y
    732 ?( Q, w7 z/ j1 H
    742 t% `1 \9 F1 U& C: [* ~8 u
    75
    % S; U' y: y( f' Z7 m768 J: W* G" T! C8 E
    776 {" J& X, p9 |% t9 U% B+ }% J4 s
    78, \0 Z& Y2 z8 w  P+ O7 P  `; t
    79
    8 S3 {  U2 o/ _9 V* a" F: j80
    8 i4 _; x' G' P8 m) B0 d6 B  W81
    ' Y5 {* Z$ p0 Q. z, g# u% U$ g% E825 H2 P1 A. u! N0 f  Q6 ?0 O3 X
    83
    ( N' G+ H  a+ d84
    . q5 J+ w  |# g2 }2 i" g85! a1 ]4 K& ~" c. m3 r* D7 q
    86* q+ u( y) v4 i: ^' j6 C$ Q
    874 u1 t" R( {' i/ Q
    88
    & W/ @0 ]. ?) l4 ~1 u, \896 X( E8 Q0 l$ W$ I" e
    90
    $ `( v" {' j0 L1 o. ~91/ ~4 u* V/ E& ~  a
    92; D( _* U% _3 ?4 {% K
    93
    + g1 s/ ?3 }) q94/ [5 D; I9 \' v
    95
    . y! I4 }: e/ l96
    ( y6 I+ L: P; t0 m% I$ L97
    4 |# h4 z$ ]6 l988 h; {% f5 r5 P- J& r
    994 ]* g, l: H/ G5 G6 O8 Q
    1009 Y7 E6 V% R# v- |
    101
    / Q: ]6 W9 b: F) \, }$ o3 |102
    : b" z4 I1 d1 [" g4 C' W7 s* @( {103* y; B# X- d6 N1 v
    104% ]( |! M8 P5 U
    105
    4 z: L4 v; u) G2 l/ b% Y& k106
      g+ t8 y0 {& y: c6 y0 ]9 c107
    7 l8 T0 W6 S, N) _* w% C1086 ?5 D% R4 L& R3 D! W1 f
    1094 e6 n+ B5 h5 k# W- U5 o7 F
    1108 Q% w: J4 L" F
    111
    ' u- ^9 v: `/ V$ Z112
    6 d' |. V% ]8 R  k: ?) Q! t+ F) `113
    8 Z7 v* |' O8 k- K, f9 x1 ?114
    3 ]$ ~, v% r+ t: t. U% B4 e+ Z+ a115
    # w) m  s+ p# W2 l- A  R5 |116
    % M. f- d, K" d4 \2 r% J117- V7 D7 N7 R; _* ]
    1187 h: [( D) V3 k+ X  t/ R# i$ O
    119
    / ], {9 G. n) @120: \; o' w, H2 S  ^3 m
    1212 X. ~' |4 C: M9 N1 q) s. k- [* w
    1225 R6 `( V0 z# r4 q
    123
    ) a+ x6 c" |6 M. m124
    5 F0 x- C/ `6 G" K$ w( u125
    . t+ k3 R, m$ S5 I" M- V126
    1 W/ U. O. n! e/ B127
    1 G# o- t- K. u9 k" \) y- C128/ Q3 m! H: @# P) k5 O* T7 f* e
    129
    6 A5 p. l0 [3 Z0 z% C" @, r: N+ g130
    3 i2 a4 M% g3 j6 t( l1 K0 x131: _$ v% C4 `( c8 y4 Y* `* ?
    132: @. \/ D1 a" ^9 Q5 |; L) L
    1338 M8 |! C" Q+ J. @- N
    134
    3 o; D. ~0 h; X- o4 f135
    # O( ], F+ U3 @% V. B* u/ H( F$ Q136
    - d" I# E, V: `( n9 `( C137, f0 l1 N! s/ {0 n. q6 T" D
    138
    & U( p$ b5 \' x5 x3 n4 T9 t139
    0 W) H& \4 G4 b; o; ~# ?140
    ; Y: R% B2 D# X- Z: O8 w141! I3 X+ [6 Z" ]: B
    142& g- `$ g; ^2 T, ]
    1436 {6 \8 o8 e& a- h
    144
      o& l8 P9 K5 H6 u% e145  z, f; n7 e7 J2 \/ k( `% d$ S* u
    146
    : f: I& H' N) E  l' W* t1 }& O0 j& U% C147
    2 L/ [8 F- r6 @5 @% I8 C4 Z: g9 g1484 C) T% b- c. e5 A8 M. U9 l
    149
    8 ]& \- h& t  `1507 v' l3 n1 G6 L  p, n$ G
    151
    9 Z9 e/ I; z* P4 o! O# O: {152
    - d8 ?4 \- F) G153; V0 n7 R% V! _
    154" h- L% C! `$ V8 M8 K- `( ]4 K- L
    155* F8 z5 I# Q  o1 g0 q1 Q
    156
    # Q* X0 I0 Y7 V% L4 M+ y3 f157
    8 e$ d8 s# x) G. O, |- T158& J' p& b+ o8 J0 P3 t- c
    159
    0 p: Z: H* l" z; a160
    6 |: c8 n" _$ P6 l1 L$ W! L6 p161" y2 `/ n( ~0 f) j; l# }
    162
    9 e/ A6 \( Y& P1 n( L, ~163
    9 I- G3 Q& [* V164
    3 ?7 I9 ]: |$ K3 m) X: L, ^! ^  l165
    6 m' N* r( v3 Q" V166
    ) b2 v1 f) {  `. b# Y4 ]$ i167) h# k! Z3 x$ F' L
    168
    , p9 h* j- `! ?0 Z; }- k, R1696 g, s1 u2 `+ S& S0 s
    1703 X; J6 B3 P# M" S; `9 ^
    171
    7 X9 [9 F/ _) y3 C1722 q+ {8 m- y. I
    173: k2 r" t0 a8 J6 K3 x2 R
    1743 s  a, z' N. W" C
    175
    6 i  M' p/ @9 w. P6 x176
    8 m- J' T- N9 n6 ]5 j177
      F' h# E% x; }) C178& m$ g: _) E) A7 D% S" h
    179- X4 ~# q0 v" _9 F
    180
    , U& A% E3 n( }* `8 ]181% Q6 P% v9 @% W
    182& f1 L: ?, ]3 l+ {
    183
    3 y, l  S" r, m0 z% {2 F! o184
    + r9 o3 f/ H/ W3 @( m  ~185  V- g/ I) g6 K( F
    1868 r" N  @* ?' v) e+ @1 }! U
    187
    0 Y) V- F! x5 H* n- o$ J188& c1 A& R+ c( x4 o& u7 ~
    189
    1 H4 W/ l1 }" O) _; s$ M190
    3 ^( A! H4 U; S+ _& v! N' t4 s191! L5 V6 n$ a+ m" S6 ?1 h, M
    1922 g) v& `" N3 j% a2 @! O
    193  u$ [! V# Z2 D) m5 W: P/ j' N9 f
    194
    8 b" S1 I) t  p4 ?  u4 v# J+ Q+ Z195! i  J4 t+ _. o+ H- F) P0 P* N
    196
    7 V) }* P% B9 N' x197
    . y$ X, u' W9 }7 {/ F198
    " G+ H$ W/ I9 ]1 Z1992 U" i  u9 U7 y4 r" X( B
    200! K+ g% ?. _5 c4 K. B
    201
    % c4 ^3 x6 [  h2 S: _% Y! I8 i202
    / A( E" d% b  A3 \203, _. i6 J0 e- `* g, F4 P/ y& _
    204
    0 R5 F" @  M3 {- w6 q% ]7 Z4 h4 F205+ T$ m$ X3 Q" k7 t7 j! w9 C% f9 S
    206
    . ~! N, j; j* u/ K2074 i  ]' Y2 K# z% [
    2087 y7 U4 ^& u* }5 Z) g2 e. }# q0 ?
    2096 h( ]/ k3 M; P( t, `8 |
    210) q& T) v5 [" _+ u  r; D2 U4 ?# |$ c
    2112 }2 L8 x' U. U8 z8 E2 F0 j4 s
    212
    1 e2 b. V: W' C4 k" s. W, T* N2134 M* f2 O8 S$ j; I% z' U4 |
    214
    0 L6 T% ^; C& P4 d* y215
    ; x! b) j, ~+ l3 M+ {4 M216! u2 u) r3 r* x6 ~
    217+ E" v/ W! t, e6 t
    218
    6 X3 A# _, t( r+ n, t( m219+ A% B4 n0 q0 V% P( }
    220
    / M1 V2 ?, ]* `- b221+ A5 z: n" g2 N. S, F& _$ D0 ^/ b1 i
    222
    ' Q" t5 C, H- j( s223
    6 Y' B8 J. C. r224
    . ^9 S# U* b* H+ J1 r" Q% _225
    1 b7 A( x0 o# M7 w. H5 V0 \226/ f& S9 {6 |- H; P4 n" g
    2272 u# Z; B: a! ?  g* L
    228" F" ?  k) P! i" \. O
    229
    7 l6 g: R' \* d7 m8 e2307 @4 g0 |) S0 H7 F
    231/ @1 |4 ^3 c- n. C. O
    2328 J$ R$ g9 o0 _+ O; V: U- p, q+ ]: s. J
    233
    + R' k/ k. L( s5 c234) N; J; z+ Y8 I5 k7 N+ C3 e5 Y
    2355 x/ f! i5 B5 M5 X- j
    236; j6 w8 [  Q$ f, |' j2 n
    237
    8 B) O8 S5 d+ Y2384 O+ ?6 S: P; e% N, z9 t
    239
    / L& f4 K1 @8 u2 R% D# Z( U240+ q* c  \$ b% N5 t6 k* I+ w5 d
    241
    9 \. c+ S# o; M6 n1 z2424 [1 {8 h' w2 _$ \' E, e! q
    243* \- a. G3 ^  P$ }- B  v5 S) n5 D
    244
    / n! b  K2 V! H245
      W' Y; G2 t# l, p2466 W: ~  f$ {& }( |! Z
    2471 s7 h& M  q6 l8 }7 U4 O1 ?8 r
    2483 o3 ^2 p  K, |5 D1 @, Z
    2496 X+ n, C; w1 {8 ?2 k5 g& o
    250, j' f/ d5 j$ s" Z4 \
    2511 y( |$ S. v2 y- [' n
    2524 {& B! C% l3 z
    253
    - O0 M" S; ^! g7 o, \# k254
    8 O% v, y; X+ {8 J! L7 d. O255
    # N8 _- S( h' J8 r! U" x256
    1 W! h( w# t1 ~0 j" l8 o! V2 A257
    4 f% y/ ?- m0 N$ x+ m258
    . R" }) R* Q7 I- u  m7 M$ y8 O259' J+ A, p' V- S) N( ~) D7 G9 Q' U
    260
    ; L; T5 {& R) i* i& M# |9 ?261
    . T$ v: A; g8 s6 d$ r262) M$ F! K% e( a1 I$ k# Y4 i+ j$ V- o
    263
    9 X, T# Z9 F$ h* C8 R2647 j9 ?  j7 Q) P' y3 Q# D6 [0 j
    265
    " _) O" j- W0 ]1 U! M266
    # H. O. A$ g8 \267
    7 L1 `+ q0 X, v: t268
    ! T1 s# ?( y. Q269
    . t6 u! \  x8 E; n6 x270
    9 z8 p* ?* _8 Z271
    ) y6 [! R: K% f1 `) t4 u272
    , x$ q9 O; p& V3 O( a1 O- H4 f273
    ! ~& }2 h- |: M4 C( G274
    2 X% }- c  g5 |  \2 [( n, C275
    8 U5 J3 K- E' g# F. Q+ ~276# {7 _1 Z, n% l8 I# V) L
    2771 X# `( n4 C7 e$ {6 r9 K. r
    278
    ( {* f- W3 B9 z  _3 r3 P0 s279
    * c/ x! o' ?  `' x/ \2804 V% J: X9 n  v' G6 O( |
    281
    0 a" ?  w2 ]4 l- Y282* ^  k, W7 e* m; q! e/ f
    283$ g9 x* r8 ~) h& G9 B' d( i4 L
    284
    , O0 H1 T: |& }  W8 _2 f285" k3 F+ m" Z" }! y
    286. Q' R' B' Z5 |  }9 w6 [/ n( ]
    287' o+ x# n. D4 A8 L/ `) M5 R
    2886 d3 h& |% w& ^6 y; f  ^& q$ E
    289
    " q6 f, N) [0 U7 A4 A2906 ?& s% s( ]- ]9 E: y8 c
    291! j% X* @, `: \0 \
    292! E. L, r: K3 K& _& D2 b
    293
    $ p0 |- x' o9 f# ?1 r) b  i294
    , ?5 ~' A* ?' u: B2 c* D; m3 r295# x  q7 H5 E' n" n/ Q' S1 o# L
    296
    ' A5 R+ P  K, r# W4 M9 Y. e" G297' p1 M( {5 d" a6 E+ i
    298" X# ~/ d( w5 Q9 u% _( o
    299
    ! T6 g% S) w5 |4 k8 F5 @3007 W7 ~9 o+ n5 z5 t
    301" ^3 q6 j/ a0 ?9 U5 u. T
    302
    3 _. ~& c! k  j) A% E1 p+ R303
    / V: ]8 U, Q% l! }304: @* {& ~5 V( E! R1 V
    3059 Q$ Y! o. g9 z7 J
    306: b1 w9 a/ p/ i5 j7 g" E
    3070 w0 O/ U& F& {' D" G: J
    308
    ' B( C3 q- s# e, B- T309
    5 X8 E1 e1 m" W- I  g310
    3 `4 Q* r$ J3 V2 F3117 a8 P- s9 S% t+ B% `/ Y
    3122 o5 g. \6 X4 {: a0 x1 Y( E9 N
    313
    0 v6 d' A1 g, A  [. w& }, L3142 @/ p& c6 x/ @  h6 a  e$ A! \
    315
    6 s+ x8 f+ x1 S( i4 t3166 U- ?+ _# y/ _1 h# O/ O! t8 X1 Y
    317
    & A) Q1 _: `+ e- `318
    ) ?+ ^* X0 f- d* _- {6 D1 c319
    6 S5 n" ~7 X$ a" q4 B% E320& v+ T5 g& G, ]
    321
    3 h7 {9 @  R1 ~- w" ^322( Z. _) @/ N/ l; M* O8 B3 G  F
    3238 D3 g% f# ^% J8 z/ j# r& I! b
    324" A. \" y* e% o  V
    325/ I3 j/ V6 `* R+ P9 \$ |/ J
    326
    # m. Y+ w: D. a) }327* P. Q  [5 m* s
    328
      r. N( r# n- c& a/ l329
    7 E& A: J2 x+ ~- h330
    6 ^; b* i" i1 X+ Q7 u331" g/ X8 K; k% F) k% b" M
    332
    : V& l4 M5 h+ c6 x5 M  w0 E333  J+ k6 h0 ]) p( f8 q) D  V. \
    334% V4 e* _0 p/ x. C0 ?- }1 m8 t+ G! Z
    335
    % w0 Z' I! k1 K$ b, G$ Q2 |. e4 h336
    & v: f; k# X9 N# X7 i8 c+ w337% J' k! t$ @/ Z. t3 e3 K3 K$ e( ]
    338
    ) ?# b/ L  o7 h339: G1 O3 \4 r4 R. O; P
    340
    8 q9 e/ j, V8 g- F9 D; B3415 g4 z# c* O5 [1 a! s4 l! P1 U0 U7 S$ Z
    342  q: L( x* {# m
    343# \& d* Z8 ^  W' L
    344" Q0 y# K. U! t2 l4 _
    345
    5 s' D5 I' L' `4 U+ U346
    9 Q, N4 L) R' q! U% A347
    , k1 r3 J  V! _! K348
      t( b2 |4 O3 [/ u349
    9 G% L5 U4 G/ D( n/ i350
    ' J# F* H$ ^* v/ R+ N" I351) n" T* \* U4 S7 e1 S
    3525 J" A# ]6 r% P- f
    353
    $ R% o; x+ R  q354* s# M+ S2 V* P8 E# [6 ^
    355
    0 H$ _! x1 z, H, u1 T: h3562 F2 R0 T* D7 p* W
    357  t5 v$ _5 _7 S6 z
    358) _& _2 k  X+ C$ [5 ]
    359" o' v# Y) l+ q% f
    360
    * D( Y' H8 C% \' C- S361
    5 E, J: a9 M0 ~% ~362
    3 D" J/ F3 r  k1 n% [! j. V) H363/ u# T, Q+ p& E: Q# P8 p
    364
    . w2 y# [1 x$ }3 y  T3 c. d: F8 Q" Q365/ b7 U/ [, j. z' c
    366% i$ u5 {! R2 l7 ^
    367
    # ^, b3 |0 H$ \1 Y. h! T9 ?368
    8 K8 l, c  Z1 h" V9 G5 _3691 x  {  R( |1 t1 O8 G3 Y
    370
    ) V; W0 P8 [4 O1 c% s3716 Y& [4 d" e9 @0 r& v" a
    372
    6 i2 K4 X  ~, B373
    4 p: U- z: S8 M: y% n- X) s3 l6 I374
    - i1 K' ?( R/ s! }) L* G0 y; P375
    2 @. F* G, [2 L4 g) E376
    9 W* ?6 t+ _: n- Z377  g+ `9 v! P+ l# V, W
    378$ Q  \  n* T) L# n
    379
    8 Y) c! ^* L5 @. Q/ V3809 B; P, m0 W# w% D9 \
    381
    3 b1 c( u6 O! q& i, n1 S382
    ( t& y9 m" B; V) J383
    - j2 d" W, V1 L. @; ^384
    6 T# Z' q9 [# {; {  C385
    - q& V( J. e6 P/ o; U1 m; X386
    1 X/ ]$ D+ \3 b! c387
    7 V6 E* ~  ]1 x* }3 v388
    $ e: i. ]2 n6 w389
      H4 h/ l" {/ D: v390! a# A4 Z" s' P, [$ U- l
    391
    , j0 a9 ~( ^  N392
    8 v+ L# s1 Q0 L. X" y393
    * G% E4 H, H. ?' [1 f5 j! c, J* N3945 q/ R7 Q; k! L+ ]" j9 c; C
    395
    7 S: S' G# _9 \  [2 n+ Q$ m396
    4 M" [% Q! F$ c0 J4 m' R3977 z4 F( p; C2 }7 T2 T4 N
    398! n5 I: n: u: L
    3991 C9 y" ?6 n/ `) }6 X8 q$ Z
    400
    " `; g# j# p4 g" |; Z401
    8 m# L$ A3 O+ F4 D9 w4024 `5 }2 K2 s! T
    403
    5 X# r" B! n. `7 d/ L! j5 w9 ?404
    / ^6 V" X1 h+ e( b* M1 i405
    2 D! X$ U* O. f* B' U" v4 u9 u406
    # R8 Y0 o1 r- O* @& H+ p407
    2 A& h; g  a+ t6 u5 O, E408  w  z( e# l! Z" b3 j
    4091 q8 c% e3 g; s; `5 s) U
    410" }: P+ B- ~- B, r+ N
    4115 R& @* A; H+ g4 M5 o1 Y
    412
    0 u1 ]( a# M3 |0 x  D. w8 e0 l& o- W0 F413
    : `/ a; ]4 }3 I7 {$ ?9 U4148 i, R2 i$ F, |+ X+ q
    415) c- m0 b, _8 ^3 S* j
    4160 j3 a! X  A6 S3 L$ F% P/ n
    417: D- l; e; |3 i+ e
    418
    9 J  H4 p4 h- K. m0 l, c/ P419
    / \& p1 J7 [2 k8 Y( [- h1 a4202 P) l$ u1 h4 k$ K, X- P! R/ `
    421
    / J* Z0 x8 p6 O$ |8 r" \, x2 Q+ |* w4228 Z  Z( k7 [+ ?6 p0 c6 h5 L+ Z
    423
    . k' ]  e0 T& N9 L, x; e7 A  R424/ N. ^4 _! b& m2 S) K1 c5 _. |
    4255 n5 e" Q4 y* `1 o4 q
    426" s+ I0 e4 M7 {, n5 n8 `
    427
    : ?. ^( _8 q5 h/ x0 Y3 {8 F428
    0 Y9 j- K/ p1 w. l. c( v# Z9 u429
    & B) N* W* a6 E) I8 l- Z430
    0 i& [- \- h3 h9 f431- r  c7 e( L* `: ]: y5 ~+ ^5 L, ^, O
    432
    2 y" d2 F  A" C. ~6 V% Q+ g433
    5 y% C, m( @: }2 p434
    " o. C: M6 ^& q/ e435
      t3 [& B" J4 h7 v  l436
    % d; }! l& `9 ?8 j, L437
    6 E! v& ?+ |& @! j+ ]438
    $ r; }  V- j! e1 _) j: t$ \0 Q439! a6 D& [- V0 t/ _1 a, P
    4405 Q, I6 \% S' C
    441# G; S* z. j6 m  f: v
    442
    / H5 I: a! K# |: P443
    # W% s% m2 N* n# P5 A444% u& z7 a! d: B1 L( P
    445
    4 h0 p8 n) I& \8 O7 L" e5 X446
    ( [3 z& C8 y- U- n$ j447  A/ a( G  _# n- b0 b
    4489 |$ r, _* O5 m: _1 h; H' T$ P& ]/ n
    449! w- G* W1 H1 v: r4 j4 Z0 l  Y6 c) U7 v
    450
      T* y6 X3 h$ M' I5 U! g7 m451
    ( |/ |+ _6 K8 Y4 T5 o( m1 ~7 k452% H9 t/ s1 b! A3 b# u
    4537 G# U- s1 C! d1 t0 k# ~/ @
    454
    4 q' r! c4 M  G8 t/ _$ u" n455
    ; l* B7 C. q. l4 M9 ]: m% ?456
    ; x3 U6 H- m. e1 p# z; E4570 _& C" u- E( W, Q4 M
    458: z; x# l: a7 O1 S
    459* Q0 q! {! }1 y4 R3 E6 K; I
    4604 n! g/ N! d; L1 I; i$ i+ }0 n+ u
    461
    4 @" h. B! V, ~) j- a* U462
    1 T! n7 u- x% [: o' ?! b463
    ! X& q4 i1 m* x7 B2 l; o464
    ! h* F5 x1 ^* {' ]465
    2 a( v6 E) @5 [& T  Q466. |' ^0 N* f$ Y* Z" s
    467
    # }: U% ?! P0 b- n+ E# m" `# U5 H4680 j! C6 n8 x! g
    469+ X4 ]( E& Y) S( f* d" b) S4 K
    4700 l. z" L7 X% V& ?# ]
    4714 O3 l& ]& Q1 U- _+ a0 n* {
    472
    2 Q5 v0 u, C2 x5 w6 L473: |, A3 B  {( G0 B' ]
    474! S$ A: B/ H0 c. ?, M$ p# B' v
    475+ r: l4 W$ f$ z+ M% d$ i
    476
    ; F$ V" p7 m- \( x5 R9 h/ }" U$ n, Y1 k, i$ X
    3 M4 @7 G( c5 L* v. s

    , I9 v4 o4 A/ T  v+ i! k
    / ]# ]. [  k0 p  k+ {2 D% `8 W8 T. ^" R; _- i0 k. t
    ( W0 T. P1 J3 S1 E

    5 S5 a3 d0 `3 n- i+ D, h
    2 p( Y$ I/ D. y. m& k& I: O
    7 c: P  \/ b; O* e0 f$ Z( p8 h6 Y7 E2 b3 n+ e9 d& [% d- j/ O) E8 g& b
    " e# M0 `3 h+ d, F6 Q7 W# ?, ]
    2 g4 r& U/ h2 \% }, [0 D9 L, e

    8 p% G4 r. C% d
    ( v- O7 @% f* b# G- o
    0 {5 X: O8 [' h. H# C5 X, v+ Z+ o( D2 B+ Q

    ; s7 O6 U3 v/ S9 C
    ; [- U) Q/ [) t3 H+ d
    5 f5 e5 X. X( Y  P
    - x0 w$ |2 r8 U
    ( @: u6 @0 D7 L+ L4 B0 c- W2 a  C$ V" w, ~* e. Y* C" ?* }: b
    ) U/ K( {+ J5 V6 q' D  w4 k" n
    ! V% E1 L" D9 ]; Z0 {3 H# `
    ————————————————5 ?% |. \$ u+ P; a1 t$ ]
    版权声明:本文为CSDN博主「biyezuopin」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
    8 B: T$ e/ j; f/ v  n9 _. p原文链接:https://blog.csdn.net/sheziqiong/article/details/1268140213 U5 L0 B0 D8 q+ z

    " \4 r/ k2 T, j0 N
    * [# X$ l7 v) O! k" \' R2 M2 M# C
    zan
    转播转播0 分享淘帖0 分享分享0 收藏收藏2 支持支持0 反对反对0 微信微信
    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-6-15 10:07 , Processed in 0.459665 second(s), 51 queries .

    回顶部