DP专项训练

第一题

来源:P3609 [USACO17JAN] Hoof, Paper, Scissor G

题意

给你n个蹄子/剪刀/布,在你没改变前出法均相同,可以改k次之后最多的相同的局数有多少。

做法

线性DP

因为手势用的字符表示,为方便可以转换为数字

void calc(int i,char a){if(a=='H'){d[i]=1;}else if(a=='P'){d[i]=2;}else{d[i]=3;}
}

我们还可以写一个check函数,来判断当前出的手势是否可以赢奶牛出的手势

bool check(int x,int y){if((x==1&&y==2)||(x==2&&y==3)||(x==3&&y==1)){return true;}return false;
}

约定m为题中的k

分析

  1. 状态定义:f(i,j,k):前i次中改变j次,且第i次手势为k的最大获胜次数
  2. 状态转移: 不改:f(i,j,k)=f(i-1,j,k)+g
    改:f(i,j,k)=max(f(i-1,j,k)+g,f(i-1,j-1,l)+g); 1<=i<=n,1<=j<=m,1<=k,l<=3 g=check(k,s),s为当前奶牛出的手势 l指除k意外的两种手势

结果: ans=max{f(n,m,l)},l为三种手势

时间复杂度

将字符转为数字为O(n)dp中:遍历1到n为O(n)遍历改变情况为O(m)遍历一遍上一次的手势与当前手势为3*3check函数为O(1)时间复杂度约为(9mn)总时间复杂度约为O(9mn)

代码

这里采用记忆化搜索

#include<iostream>
using namespace std;
int n,m;
const int N=100010;
char s[N];
int d[N];
int f[N][4][110];
void calc(int i,char a){if(a=='H'){d[i]=1;}else if(a=='P'){d[i]=2;}else{d[i]=3;}
}
bool check(int x,int y){if((x==1&&y==2)||(x==2&&y==3)||(x==3&&y==1)){return true;}return false;
}
int dfs(int i,int j,int k){if(f[i][j][k]){return f[i][j][k];}if(i>n){return 0;}int l=check(d[i],j);int x=dfs(i+1,j,k),y=-0x3f3f3f3f;if(k>0){if(j==1){y=max(dfs(i+1,2,k-1),dfs(i+1,3,k-1));}else if(j==2){y=max(dfs(i+1,1,k-1),dfs(i+1,3,k-1));}else{y=max(dfs(i+1,1,k-1),dfs(i+1,2,k-1));}}return f[i][j][k]=max(x,y)+l;
}
int main(){freopen("hps.in","r",stdin);freopen("hps.out","w",stdout);cin>>n>>m;for(int i=1;i<=n;i++){cin>>s[i];calc(i,s[i]);}int a=dfs(1,1,m);int b=dfs(1,2,m);int c=dfs(1,3,m);cout<<max(a,max(b,c));
}

第二题

来源:P3140 [USACO16FEB] Circular Barn Revisited G

题意:

即求满足给定序列所需要的最小总代价

做法:

考虑线性DP
因为仓库是圆形的,即构成了一个环 所以我们应破环成链

约定m为开门数,即文中给出的k

分析

  1. 状态定义:f(i,k):开启第i扇门时共开了k扇门的最小总路程
  2. 状态计算:f(i,k)=min{f(j-1,k)+g(i,j)};

           1<=i<=n,1<=j<=i,1<=k<=m由于我们破环成链,所以应以1..n为开头依次遍历l指开头,r指结尾g(i,j)指由i门转移到j门的路程和
    

为了节省时间复杂度,g(i,j)可以预处理

结果:ans=min{f(l,m)}; 1<=l<=n;

时间复杂度

瓶颈在dp部分
以1..n为开头依次遍历为O(n)
计算f(i,k)为三重循环,时间复杂度为O(mn^2)
总时间复杂度约为O(mn^3)
注意:不开long long见祖宗

代码

#include<iostream>
#include<cstring>
#define int long long
using namespace std;
const int N=210;
int n,m;
int a[N];
int f[N][8],g[N][N];
signed main(){cin>>n>>m;for(int i=1;i<=n;i++){scanf("%lld",&a[i]);a[i+n]=a[i];}for(int i=1;i<=2*n;i++){int x=0;int d=min(i+n-1,2*n);for(int j=i+1;j<=d;j++){x=x+(j-i)*a[j];g[i][j]=x;}}//预处理g数组int ans=1e18;for(int l=1;l<=n;l++){//遍历开头memset(f,0x3f,sizeof f);f[l-1][0]=0;//初始化f数组int r=l+n-1;for(int i=l;i<=r;i++){for(int j=1;j<=m;j++){for(int k=l;k<i;k++){f[i][j]=min(f[i][j],f[k-1][j-1]+g[k][i]);//状态转移}}}ans=min(ans,f[r][m]);}cout<<ans;
}

第三题

来源:P5424 [USACO19OPEN] Snakes G

题意:

将一个长为n的区间分为最多分为 (k+1) 个子区间,每个子区间的代价为 (区间最大值*长度-区间和) ,求最小代价

约定m=k+1

做法:

分段式线性dp

分析

  1. 状态定义:f(i,j):将前i个数分为j段时,所需要的最小代价
  2. 状态转移:f(i,j)=min{f(k-1,j)+(g(k,i)*(i-k+1)-sum(k,i))}

       1<=i<=n,1<=j<=i,1<=k<=m意思是:将前(k-1)个数分为(j-1)段,让k到i为一段g(k,i)为区间最大值,sum(k,i)为区间和这两个数组可以预处理
    

结果:ans=f(n,m)

时间复杂度

预处理 sum() ,g() 可以在O(n^2)的时间中求得DP部分除计算代价的地方不同以外,其他地方都与分段DP的模板相同
时间复杂度约为O(mn^2)总时间复杂度约为O(mn^2)

代码

#include<iostream>
#include<algorithm>
#include<cstring>
#define int long long
using namespace std;
const int N=410;
int f[N][N],d[N][N];
int n,m,a[N];
int sum[N][N];
signed main(){cin>>n>>m;m++;for(int i=1;i<=n;i++){scanf("%lld",&a[i]);}memset(f,-1,sizeof f);for(int i=1;i<=n;i++){int ans=0;for(int j=i;j<=n;j++){ans=max(ans,a[j]);d[i][j]=ans;sum[i][j]=sum[i][j-1]+a[j];}}memset(f,0x3f,sizeof f);f[0][0]=0;for(int i=1;i<=n;i++){for(int j=1;j<=m;j++){for(int k=1;k<=i;k++){f[i][j]=min(f[i][j],f[k-1][j-1]+(d[k][i]*(i-k+1)-sum[k][i]));}}}cout<<f[n][m];
}

第四题

来源:UVA11584 Partitioning by Palindromes

题意

当一个字符串正序和反序是完全相同时,我们称之为“回文串”,例如“racecar”就是一个回文串,而“fastcar”就不是
现在给一个字符串s,把它分割成若干个互不相交的回文子串,求分割的回文子串的最少个数

判断是否为回文串可以写一个 check() 函数

bool check(int l,int r){for(int i=l,j=r;i<=j;i++,j--){if(s[i]!=s[j]){return 0;}}return 1;
}

做法:

分段式线性DP

分析:

  1. 状态定义:f(i):由 1..i 的字符串中,最少可划分为多少个回文串
  2. 状态转移:f(i)=min{f(j-1)+1}

       1<=i<=n,1<=j<=i意思为:当满足check(j,i)(即满足 j..i 的字符串为回文串)时,将 1..j-1 的字符串划分为最少回文串数,并加上该串,与f(i)取最小值
    

结果: ans = f(n)

时间复杂度:

主要是dp部分,时间复杂度为O(n^3),较高可以设一个p(i,j),判断由 i..j 的字符串是否为回文串
O(n^2)的时间预处理出所有p(i,j),并O(n^2)的时间dp
可以自行了解

代码

#include<iostream>
#include<cstring>
using namespace std;
int t;
const int N=1010;
char s[N];
int f[N];
bool check(int l,int r){for(int i=l,j=r;i<=j;i++,j--){if(s[i]!=s[j]){return 0;}}return 1;
}
int main(){cin>>t;while(t--){memset(f,0x3f,sizeof f);scanf("%s",(s+1));int x=strlen(s+1);f[0]=0;for(int i=1;i<=x;i++){for(int j=1;j<=i;j++){if(check(j,i)){f[i]=min(f[j-1]+1,f[i]);}}}cout<<f[x]<<endl;}
}

第五题

来源:UVA11400 Lighting System Design

题意

你的任务是设计一个照明系统,一共有n(n<=1000)种灯泡可供选择,不同种类的灯泡必须用不同的电源,同一种灯泡则可以用一个
输入为一个n,以下n行,每行四个数值,代表电压V,电源费用K,每个灯泡费用C,所需灯泡数量L。n=0为结束标志

做法:

分段式线性dp

首先明确一个性质: 如果当前灯泡不如另外某种的灯泡,那么这类灯泡应全部换成其他比它优的灯泡 否则全部不换

因为只有电压比当前灯泡电压大的灯泡才能将当前灯泡换成该灯泡 所以我们应对灯泡按电压大小排序

分析

  1. 状态定义:f(i):前i种灯泡在进行操作后的最小总成本
  2. 状态转移:f(i)=min{f(j-1)+wb(i)+wc(i)*(sum(i)-sum(j-1))}

       1<=i<=n,1<=j<=i意思是:将 j..i 的灯泡类型全换成第i种,并加上前 j-1 种灯泡在操作后的最小成本,与f(i)取minwc(i)*(sum(i)-sum(j-1)) 指换后的成本其中:sum指灯泡数量的前缀和
    

结果: ans = f(n)

时间复杂度

排序为O(nlogn)
预处理前缀和为O(n)
dp套分段dp板子即可,复杂度为O(n^2)总时间复杂度约为O(n^2)

代码

#include<iostream>
#include<cstring>
#include<algorithm>
#define int long long
using namespace std;
int n;
const int N=1010;
struct node{int a,b,c,d;
}w[N];
int f[N],sum[N];
bool cmp(node a,node b){return a.a<b.a;
}
signed main(){while(cin>>n,n){memset(f,0x3f,sizeof f);memset(sum,0,sizeof sum);for(int i=1;i<=n;i++){cin>>w[i].a>>w[i].b>>w[i].c>>w[i].d;}sort(w+1,w+n+1,cmp);for(int i=1;i<=n;i++){sum[i]=sum[i-1]+w[i].d;}f[0]=0;for(int i=1;i<=n;i++){for(int j=1;j<=i;j++){f[i]=min(f[i],f[j-1]+w[i].b+w[i].c*(sum[i]-sum[j-1]));}}cout<<f[n]<<endl;}
}

第六题

来源:P5924 [IOI2004] Phidias 菲迪亚斯神

题意

为了这座雕像他需要大小为 W1*H1,W2*H2...Wn*Hn​ 的矩形大理石板最近菲迪亚斯获得一块矩形大理石块。菲迪亚斯想把这块石板切成所需要的大小石板或是石板所切割出的部分都可以由垂直(或水平)方向纵贯(或是横贯)加以切割到底成为两块矩形石板,同时切割出的这两块矩形石板都必须具有整数的宽度与高度石板只能以此种方法加以切割,同时石板不能粘合成较大石板因为石板具有花纹,所以石板也不能旋转菲迪亚斯想知道如何切割最初的石板,才能让所产生的废料最少

设整个长方形成为W,宽为H

约定a[]为w[],b[]为h[]

做法

记忆化搜索

分析

  1. 状态定义:dfs(i,j):切割长为i,宽为j的长方形所浪费的最小值
  2. 状态转移:dfs(i,j)=min{min(dfs(i,j-b(k))+dfs(i-a[k],b(k)),dfs(i-a(k),j)+dfs(a(k),j-b(k)))}

        1<=i<=W,1<=j<=H,1<=k<=n可知:a(k)<=i,b(k)<=j其中:dfs(i,j-b(k))+dfs(i-a[k],b(k))为竖着切dfs(i-a(k),j)+dfs(a(k),j-b(k)))为横着切因为是记忆化搜索,所以每搜完一个状态就记录它的最小值f(i,j)初始值(搜索内)为i*j,及最多将整个长方形都浪费,后面挨个挨个取min注意f要全部初始化为-1(搜索外),因为有可能可以不浪费
    

结果:ans=dfs(W,H)

时间复杂度

因为记忆化搜索每个状态只会被搜一遍 所以最多搜W*H种状态

总时间复杂度约为O(W*H)

代码

#include<iostream>
#include<cstring>
using namespace std;
int w,h,n;
const int N=1010;
int a[N],b[N];
int f[N][N];
int dfs(int i,int j){if(f[i][j]!=-1){return f[i][j];}int res=i*j;for(int k=1;k<=n;k++){if(a[k]>i||b[k]>j){continue;}int x1=dfs(i,j-b[k])+dfs(i-a[k],b[k]);int x2=dfs(i-a[k],j)+dfs(a[k],j-b[k]);int cnt=min(x1,x2);res=min(res,cnt);}return f[i][j]=res;
}
int main(){memset(f,-1,sizeof f);cin>>w>>h>>n;for(int i=1;i<=n;i++){scanf("%d%d",&a[i],&b[i]);}cout<<dfs(w,h);
}

第七题

来源:P4267 [USACO18FEB] Taming the Herd G

题意

农夫想要知道从他开始记录以来发生过多少次出逃。但是,他怀疑奶牛们篡改了它的记录,现在他所确定的只有他是从发生出逃的某一天开始记录的。
请帮助他求出,对于每个从他开始记录以来可能发生的出逃次数,他被篡改了的记录条数的最小值

约定m=n,即可能的出逃最大次数

做法

仍然是分段式线性DP

分析

  1. 状态定义:f(i,j):前i天中,出逃j次记录的与给定序列不同的最小数目
  2. 状态转移:f(i,j)=min{f(k-1,j-1)+g(k,i)}

       1<=i<=n,1<=j<=i,1<=k<=i因为前i天中最多出逃i次,所以j<=i(但设为j<=m也没关系)意思是:前 (k-1) 天出逃 (j-1) 次(且第k天出逃),一直到第i天再出逃的最小不同数量,与f(i,j)取min唯一与模板不同的是g(k,i)g(k,i)表示在第k天开始出逃,到第i天时有多少个不同的记录条数可以预处理得到
    

结果:ans=f(n,i),1<=i<=n

时间复杂度

预处理g()约为O(n^2) dp约为O(n^3)

总时间复杂度约为O(n^3)

代码

#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
const int N=110;
int n;
int a[N];
int f[N][N];
int g[N][N];
int main(){memset(f,0x3f,sizeof f);cin>>n;for(int i=1;i<=n;i++){cin>>a[i];}for(int i=1;i<=n;i++){for(int j=i;j<=n;j++){g[i][j]=g[i][j-1]+(a[j]!=(j-i));}}f[0][0]=0;for(int i=1;i<=n;i++){for(int j=1;j<=n;j++){//n可改为i,降低时间for(int k=1;k<=i;k++){f[i][j]=min(f[i][j],f[k-1][j-1]+g[k][i]);}}}for(int i=1;i<=n;i++){cout<<f[n][i]<<endl;}
}

第八题

来源:牛客假日团队赛54-A

题意

在Farmer John最喜欢的节日里,他想要给他的朋友们赠送一些礼物
由于他并不擅长包装礼物,他想要获得他的奶牛们的帮助。你可能能够想到,奶牛们本身也不是很擅长包装礼物,而Farmer John即将得到这一教训Farmer John的N头奶牛(1≤N≤10000)排成一行,方便起见依次编号为1…N。奶牛i的包装礼物的技能水平为si
她们的技能水平可能参差不齐,所以FJ决定把她的奶牛们分成小组。每一组可以包含任意不超过K头的连续的奶牛(1≤K≤1000),并且一头奶牛不能属于多于一个小组由于奶牛们会互相学习,这一组中每一头奶牛的技能水平会变成这一组中水平最高的奶牛的技能水平请帮助FJ求出,在他合理地安排分组的情况下,可以达到的技能水平之和的最大值

做法

还是分段式线性DP

分析

  1. 状态定义:f(i):前i头牛中,分组后可取到的最高技术和
  2. 状态转移:f(i)=min{f(j-1)+w(i,i-j+1)*(i-j+1)}

       1<=i<=n,d<=j<=i其中:d为max(1,i-k+1)含义就不用多说了w(i,x)指以i为结尾,取第 i-x+1...i 的最大值同样预处理
    

结果:ans=f(n)

时间复杂度

易得:约为O(nm)

代码

#include<iostream>
using namespace std;
const int N=10010;
int f[N];
int n,a[N],k;
int w[N][1010];
int main(){freopen("teamwork.in","r",stdin);freopen("teamwork.out","w",stdout);cin>>n>>k;for(int i=1;i<=n;i++){cin>>a[i];}for(int i=1;i<=n;i++){int d=max(1,i-k+1),ans=0;for(int j=i;j>=d;j--){ans=max(ans,a[j]);w[i][i-j+1]=ans;}}for(int i=1;i<=n;i++){int d=max(1,i-k+1);for(int j=d;j<=i;j++){f[i]=max(f[i],f[j-1]+w[i][i-j+1]*(i-j+1));}}cout<<f[n];
}

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

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

相关文章

【服务的主从切换实现原理】

文章目录 主从架构介绍zookeeper利用ZK实现主从架构 主从架构介绍 主从服务架构是一种常见的分布式系统设计模式&#xff0c;常用于提高系统的性能、可用性和扩展性。在这种架构中&#xff0c;系统中的节点被分为两类&#xff1a;主节点&#xff08;Master&#xff09;和从节点…

斯坦福的新工具,生物计算,操作系统与AI融合之路

一支烟花官网&#xff1a; https://agifun.love 智源社区 斯坦福让“GPU高速运转”的新工具火了&#xff0c;比FlashAttention2更快 西风 发自 凹非寺量子位 | 公众号 QbitAIAI算力资源越发紧张的当下&#xff0c;斯坦福新研究将GPU运行效率再提升一波——内核只有100行代码…

Java基础面试重点-1

0. 符号&#xff1a; *&#xff1a;记忆模糊&#xff0c;验证后特别标注的知识点。 &&#xff1a;容易忘记知识点。 *&#xff1a;重要的知识点。 1. 简述一下Java面向对象的基本特征&#xff08;四个&#xff09;&#xff0c;以及你自己的应用&#xff1f; 抽象&#…

R可视化:ggpubr包学习

欢迎大家关注全网生信学习者系列&#xff1a; WX公zhong号&#xff1a;生信学习者 Xiao hong书&#xff1a;生信学习者 知hu&#xff1a;生信学习者 CDSN&#xff1a;生信学习者2 介绍 ggpubr是我经常会用到的R包&#xff0c;它傻瓜式的画图方式对很多初次接触R绘图的人来…

淘宝/1688获得店铺的所有商品(商品列表)

通过以下步骤&#xff0c;可以获得淘宝或1688店铺的所有商品。请注意&#xff0c;具体步骤可能会因为平台的更新而有所改变&#xff0c;可以根据实际情况进行操作。 更多API调用展示以及获取Key和secret请移步 返回数据格式&#xff1a; {"user": null,"ite…

python 简单demo

import pyttsx3def read_poem_with_voice(filename):"""使用女声从文本文件中读取古诗词并朗读:param file_path: 文本文件的路径"""engine pyttsx3.init() # 初始化语音引擎# 设置语速&#xff08;范围通常是50-200&#xff09;engine.setPro…

提升你的编程体验:自定义 PyCharm 背景图片

首先&#xff0c;打开 PyCharm 的设置菜单&#xff0c;点击菜单栏中的 File > Settings 来访问设置&#xff0c;也可以通过快捷键 CtrlAItS 打开设置。 然后点击Appearance & Behavior > Appearance。 找到Background image...左键双击进入。 Image:传入自己需要设置…

c++_0基础_讲解4 变量定义

变量 C 中的变量是存储数据值的容器&#xff0c;这些值可以在程序执行过程中被修改和使用。在 C 中&#xff0c;变量必须先声明后使用&#xff0c;声明变量也可以称之为定义变量&#xff0c;它告诉编译器在何处以及如何去分配存储空间。接下来我将对 C 中的变量定义进行详细的介…

常见的 EVM 版本以及它们的区别

EVM&#xff08;以太坊虚拟机&#xff09;版本的演进是为了引入新的特性和改进以太坊平台的安全性、效率和功能性。每个版本通常伴随着以太坊网络的硬分叉&#xff0c;这是以太坊协议的重大升级。以下是一些常见的EVM版本及其主要区别&#xff1a; Homestead (2016年3月)&…

中国首台!紧随美国,重磅发布100比特中性原子量子计算机

2024年6月11日上午&#xff0c;“武汉量子论坛—2024”隆重开幕&#xff0c;国家自然科学基金委员会主任窦贤康院士&#xff0c;武汉大学校长张平文院士&#xff0c;以及叶朝辉、徐红星、祝世宁等院士出席大会。在会议上&#xff0c;中科酷原重磅发布国内首台原子量子计算机——…

利用 HTML5 Canvas 实现在线签字功能

目录 前言 一、HTML5 Canvas 简介 二、签字功能的实现 效果演示 完整代码 前言 在现代互联网应用中&#xff0c;有时我们需要让用户在网页上进行签字操作&#xff0c;比如确认文件、填写电子表格或者签署合同。利用 HTML5 的 canvas 画布&#xff0c;我们可以轻松地实现这一…

事务中存在多线程,怎么处理?

在 Spring 框架中&#xff0c;Transactional 注解作为一种声明式事务管理的关键机制&#xff0c;其背后的工作原理远比简单的 AOP&#xff08;面向切面编程&#xff09;和 ThreadLocal 存储更为细腻。该注解的实现核心在于 Spring 的 TransactionInterceptor&#xff08;事务拦…

图片转Base64

在Python中, 可以使用内置的base64模块以及图像处理库(如PIL, 也称为Pillow)来将图片转换为Base64编码的字符串. 以下是一个简单的示例, 说明如何实现这一过程:首先, 需要安装Pillow库(如果尚未安装), 可以使用pip来安装: pip install pillow然后, 可以使用以下Python代码将图片…

项目中常量的定义方式

方式一 在常量个数少的时候&#xff0c;通常情况下使用这种方式。 public class MqConstants {public static final String EXCHANGE_1 "exchange1";public static final String EXCHANGE_2 "exchange2";public static final String EXCHANGE_3 "…

RabbitMQ实践——交换器(Exchange)和绑定(Banding)

大纲 direct型交换器默认交换器命名交换器 fanout型交换器topic型交换器headers型交换器 RabbitMQ在概念上由三部分组成&#xff1a; 交换器&#xff08;Exchange&#xff09;&#xff1a;负责接收消息发布者发布消息的结构&#xff0c;同时它会根据“绑定关系”&#xff08;Ba…

基于SpringBoot+VueBBS论坛系统设计和实现(源码+LW+调试文档+讲解等)

&#x1f497;博主介绍&#xff1a;✌全网粉丝1W,CSDN作者、博客专家、全栈领域优质创作者&#xff0c;博客之星、平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌&#x1f497; &#x1f31f;文末获取源码数据库&#x1f31f; 感兴趣的可以先收藏起来&#xff0c;还…

sku与spu的区别!!!

一个 SPU 可以有多个 SKU。

攻防演练之-成功的钓鱼邮件溯源

书接上文&#xff0c;《网络安全攻防演练风云》专栏之攻防演练之-网络安全产品大巡礼二&#xff0c;这里。 演练第一天并没有太大的波澜&#xff0c;白天的时间过得很快。夜色降临&#xff0c;攻防演练中心内的灯光依旧明亮。对于网络安全团队来说&#xff0c;夜晚和白天并没有…

教育培训知识付费在线课程小程序开发

教育培训知识付费在线课程小程序功能概述 核心功能 课程报名与缴费&#xff1a;支持线上报名、缴费&#xff0c;自定义课程时间、人数等。 砍价功能&#xff1a;用户通过分享邀请好友参与砍价&#xff0c;享受低价购买课程的优惠。 视频课程&#xff1a;支持倍速播放&#x…

全面对比DAMO ModelScope OCR与Paddle OCR技术

本文深入比较了DAMO ModelScope OCR与Paddle OCR在功能、性能、应用场景及技术细节上的差异&#xff0c;旨在为读者提供全面的OCR技术选型参考。 文章目录 引言OCR技术的重要性概述DAMO ModelScope和Paddle OCRDAMO ModelScope OCRPaddle OCR 技术背景与概述自底向上的文字检测…