`
hdy007
  • 浏览: 29829 次
最近访客 更多访客>>
文章分类
社区版块
存档分类

常用算法设计方法之动态规划法

阅读更多

经常会遇到复杂问题不能简单地分解成几个子问题,而会分解出一系列的子问题。简单地采用把大问题分解成子问题,并综合子问题的解导出大问题的解的方法,问题求解耗时会按问题规模呈幂级数增加。<o:p></o:p>

为了节约重复求相同子问题的时间,引入一个数组,不管它们是否对最终解有用,把所有子问题的解存于该数组中,这就是动态规划法所采用的基本方法。以下先用实例说明动态规划方法的使用。<o:p></o:p>

【问题】   求两字符序列的最长公共字符子序列<o:p></o:p>

问题描述:字符序列的子序列是指从给定字符序列中随意地(不一定连续)去掉若干个字符(可能一个也不去掉)后所形成的字符序列。令给定的字符序列X=“x0x1xm<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>,序列Y=“y0y1yk<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>X的子序列,存在X的一个严格递增下标序列<i0i1ik-1>,使得对所有的j=01k-1,有xij=yj。例如,X=“ABCBDAB”Y=“BCDB”X的一个子序列。<o:p></o:p>

给定两个序列AB,称序列ZAB的公共子序列,是指Z同是AB的子序列。问题要求已知两序列AB的最长公共子序列。<o:p></o:p>

如采用列举A的所有子序列,并一一检查其是否又是B的子序列,并随时记录所发现的子序列,最终求出最长公共子序列。这种方法因耗时太多而不可取。<o:p></o:p>

考虑最长公共子序列问题如何分解成子问题,设A=“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>B=“b0b1bm<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>,并Z=“z0z1zk<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>为它们的最长公共子序列。不难证明有以下性质:<o:p></o:p>

1   如果am-1=bn-1,则zk-1=am-1=bn-1,且“z0z1zk<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>“b0b1bn<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>的一个最长公共子序列;<o:p></o:p>

2   如果am-1!=bn-1,则若zk-1!=am-1,蕴涵“z0z1zk<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>“b0b1bn<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>的一个最长公共子序列;<o:p></o:p>

3   如果am-1!=bn-1,则若zk-1!=bn-1,蕴涵“z0z1zk<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>“b0b1bn<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>的一个最长公共子序列。<o:p></o:p>

这样,在找AB的公共子序列时,如有am-1=bn-1,则进一步解决一个子问题,找“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>“b0b1bm<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>的一个最长公共子序列;如果am-1!=bn-1,则要解决两个子问题,找出“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>“b0b1bn<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>的一个最长公共子序列和找出“a0a1am<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>“b0b1bn<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>的一个最长公共子序列,再取两者中较长者作为AB的最长公共子序列。<o:p></o:p>

定义c[j]为序列“a0a1ai<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="2" unitname="”">-2”</st1:chmetcnv>“b0b1bj<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="True" hasspace="False" sourcevalue="1" unitname="”">-1”</st1:chmetcnv>的最长公共子序列的长度,计算c[j]可递归地表述如下:<o:p></o:p>

1c[j]=0               如果i=0j=0<o:p></o:p>

2c[j]= c[i-1][j-1]+1         如果Ij>0,且a[i-1]=b[j-1]<o:p></o:p>

3c[j]=maxc[j-1]c[i-1][j]   如果Ij>0,且a[i-1]!=b[j-1]<o:p></o:p>

按此算式可写出计算两个序列的最长公共子序列的长度函数。由于c[j]的产生仅依赖于c[i-1][j-1]c[i-1][j]c[j-1],故可以从c[m][n]开始,跟踪c[j]的产生过程,逆向构造出最长公共子序列。细节见程序。<o:p></o:p>

# include <stdio.h><o:p></o:p>

# include <string.h><o:p></o:p>

# define   N   100<o:p></o:p>

char a[N],b[N],str[N];<o:p></o:p>

<o:p> </o:p>

int lcs_len(char *a, char *b, int c[ ][ N])<o:p></o:p>

{   int m=strlen(a),   n=strlen(b), i,j;<o:p></o:p>

for (i=0;i<=m;i++)     c[0]=0;<o:p></o:p>

for (i=0;i<=n;i++)     c[0]=0;<o:p></o:p>

for (i=1;i<=m;i++)<o:p></o:p>

for (j=1;j<=m;j++)<o:p></o:p>

if (a[i-1]==b[j-1])<o:p></o:p>

c[j]=c[i-1][j-1]+1;<o:p></o:p>

else   if (c[i-1][j]>=c[j-1])<o:p></o:p>

c[j]=c[i-1][j];<o:p></o:p>

else<o:p></o:p>

c[j]=c[j-1];<o:p></o:p>

return c[m][n];<o:p></o:p>

}<o:p></o:p>

<o:p> </o:p>

char *buile_lcs(char s[ ],char *a, char *b)<o:p></o:p>

{   int k, i=strlen(a), j=strlen(b);<o:p></o:p>

k=lcs_len(a,b,c);<o:p></o:p>

s[k]=’\<st1:chmetcnv w:st="on" tcsc="0" numbertype="1" negative="False" hasspace="False" sourcevalue="0" unitname="’">0’</st1:chmetcnv>;<o:p></o:p>

while (k>0)<o:p></o:p>

if (c[j]==c[i-1][j])   i--;<o:p></o:p>

else   if (c[j]==c[j-1])   j--;<o:p></o:p>

else   {   s[--k]=a[i-1];<o:p></o:p>

i--;   j--;<o:p></o:p>

}<o:p></o:p>

return s;<o:p></o:p>

}<o:p></o:p>

<o:p> </o:p>

void main()<o:p></o:p>

{   printf (“Enter two string<%d!\n”,N);<o:p></o:p>

scanf(“%s%s”,a,b);<o:p></o:p>

printf(“LCS=%s\n”,build_lcs(str,a,b));<o:p></o:p>

}<o:p></o:p>

1、动态规划的适用条件<o:p></o:p>

任何思想方法都有一定的局限性,超出了特定条件,它就失去了作用。同样,动态规划也并不是万能的。适用动态规划的问题必须满足最优化原理和无后效性。<o:p></o:p>

1)最优化原理(最优子结构性质)<o:p></o:p>

最优化原理可这样阐述:一个最优化策略具有这样的性质,不论过去状态和决策如何,对前面的决策所形成的状态而言,余下的诸决策必须构成最优策略。简而言之,一个最优化策略的子策略总是最优的。一个问题满足最优化原理又称其具有最优子结构性质。<o:p></o:p>

<o:p> </o:p>

2<o:p></o:p>

例如图2中,若路线IJAC的最优路径,则根据最优化原理,路线J必是从BC的最优路线。这可用反证法证明:假设有另一路径J’BC的最优路径,则AC的路线取IJ’IJ更优,矛盾。从而证明J’必是BC的最优路径。<o:p></o:p>

最优化原理是动态规划的基础,任何问题,如果失去了最优化原理的支持,就不可能用动态规划方法计算。根据最优化原理导出的动态规划基本方程是解决一切动态规划问题的基本方法。<o:p></o:p>

2)无后向性<o:p></o:p>

将各阶段按照一定的次序排列好之后,对于某个给定的阶段状态,它以前各阶段的状态无法直接影响它未来的决策,而只能通过当前的这个状态。换句话说,每个状态都是过去历史的一个完整总结。这就是无后向性,又称为无后效性。<o:p></o:p>

3)子问题的重叠性<o:p></o:p>

动态规划算法的关键在于解决冗余,这是动态规划算法的根本目的。动态规划实质上是一种以空间换时间的技术,它在实现的过程中,不得不存储产生过程中的各种状态,所以它的空间复杂度要大于其它的算法。选择动态规划算法是因为动态规划算法在空间上可以承受,而搜索算法在时间上却无法承受,所以我们舍空间而取时间。<o:p></o:p>

所以,能够用动态规划解决的问题还有一个显著特征:子问题的重叠性。这个性质并不是动态规划适用的必要条件,但是如果该性质无法满足,动态规划算法同其他算法相比就不具备优势。<o:p></o:p>

2、动态规划的基本思想<o:p></o:p>

前文主要介绍了动态规划的一些理论依据,我们将前文所说的具有明显的阶段划分和状态转移方程的动态规划称为标准动态规划,这种标准动态规划是在研究多阶段决策问题时推导出来的,具有严格的数学形式,适合用于理论上的分析。在实际应用中,许多问题的阶段划分并不明显,这时如果刻意地划分阶段法反而麻烦。一般来说,只要该问题可以划分成规模更小的子问题,并且原问题的最优解中包含了子问题的最优解(即满足最优子化原理),则可以考虑用动态规划解决。<o:p></o:p>

动态规划的实质是分治思想和解决冗余,因此,动态规划是一种将问题实例分解为更小的、相似的子问题,并存储子问题的解而避免计算重复的子问题,以解决最优化问题的算法策略。<o:p></o:p>

由此可知,动态规划法与分治法和贪心法类似,它们都是将问题实例归纳为更小的、相似的子问题,并通过求解子问题产生一个全局最优解。其中贪心法的当前选择可能要依赖已经作出的所有选择,但不依赖于有待于做出的选择和子问题。因此贪心法自顶向下,一步一步地作出贪心选择;而分治法中的各个子问题是独立的(即不包含公共的子子问题),因此一旦递归地求出各子问题的解后,便可自下而上地将子问题的解合并成问题的解。但不足的是,如果当前选择可能要依赖子问题的解时,则难以通过局部的贪心策略达到全局最优解;如果各子问题是不独立的,则分治法要做许多不必要的工作,重复地解公共的子问题。<o:p></o:p>

解决上述问题的办法是利用动态规划。该方法主要应用于最优化问题,这类问题会有多种可能的解,每个解都有一个值,而动态规划找出其中最优(最大或最小)值的解。若存在若干个取最优值的解的话,它只取其中的一个。在求解过程中,该方法也是通过求解局部子问题的解达到全局最优解,但与分治法和贪心法不同的是,动态规划允许这些子问题不独立,(亦即各子问题可包含公共的子子问题)也允许其通过自身子问题的解作出选择,该方法对每一个子问题只解一次,并将结果保存起来,避免每次碰到时都要重复计算。<o:p></o:p>

因此,动态规划法所针对的问题有一个显著的特征,即它所对应的子问题树中的子问题呈现大量的重复。动态规划法的关键就在于,对于重复出现的子问题,只在第一次遇到时加以求解,并把答案保存起来,让以后再遇到时直接引用,不必重新求解。<o:p></o:p>

3、动态规划算法的基本步骤<o:p></o:p>

设计一个标准的动态规划算法,通常可按以下几个步骤进行:<o:p></o:p>

1)划分阶段:按照问题的时间或空间特征,把问题分为若干个阶段。注意这若干个阶段一定要是有序的或者是可排序的(即无后向性),否则问题就无法用动态规划求解。<o:p></o:p>

2)选择状态:将问题发展到各个阶段时所处于的各种客观情况用不同的状态表示出来。当然,状态的选择要满足无后效性。<o:p></o:p>

3)确定决策并写出状态转移方程:之所以把这两步放在一起,是因为决策和状态转移有着天然的联系,状态转移就是根据上一阶段的状态和决策来导出本阶段的状态。所以,如果我们确定了决策,状态转移方程也就写出来了。但事实上,我们常常是反过来做,根据相邻两段的各状态之间的关系来确定决策。<o:p></o:p>

4)写出规划方程(包括边界条件):动态规划的基本方程是规划方程的通用形式化表达式。<o:p></o:p>

一般说来,只要阶段、状态、决策和状态转移确定了,这一步还是比较简单的。动态规划的主要难点在于理论上的设计,一旦设计完成,实现部分就会非常简单。根据动态规划的基本方程可以直接递归计算最优值,但是一般将其改为递推计算,实现的大体上的框架如下:<o:p></o:p>

标准动态规划的基本框架<o:p></o:p>

1. fn+1(xn+1)初始化;   {边界条件}<o:p></o:p>

for k:=n downto 1 do<o:p></o:p>

for 每一个xk∈Xk do<o:p></o:p>

for 每一个uk∈Uk(xk) do<o:p></o:p>

begin<o:p></o:p>

5.         fk(xk):=一个极值;           {∞或-∞}<o:p></o:p>

6.         xk+1:=Tk(xk,uk);             {状态转移方程}<o:p></o:p>

7.         t:=φ(fk+1(xk+1),vk(xk,uk));     {基本方程(9)}<o:p></o:p>

if tfk(xk)更优 then fk(xk):=t; {计算fk(xk)的最优值}<o:p></o:p>

end;<o:p></o:p>

9. t:=一个极值;                     {∞或-∞}<o:p></o:p>

for 每一个x1∈X1 do<o:p></o:p>

11.   if f1(x1)t更优 then t:=f1(x1);     {按照10式求出最优指标}<o:p></o:p>

12. 输出t;<o:p></o:p>

但是,实际应用当中经常不显式地按照上面步骤设计动态规划,而是按以下几个步骤进行:<o:p></o:p>

1)分析最优解的性质,并刻划其结构特征。<o:p></o:p>

2)递归地定义最优值。<o:p></o:p>

3)以自底向上的方式或自顶向下的记忆化方法(备忘录法)计算出最优值。<o:p></o:p>

4)根据计算最优值时得到的信息,构造一个最优解。<o:p></o:p>

步骤(1)~(3)是动态规划算法的基本步骤。在只需要求出最优值的情形,步骤(4)可以省略,若需要求出问题的一个最优解,则必须执行步骤(4)。此时,在步骤(3)中计算最优值时,通常需记录更多的信息,以便在步骤(4)中,根据所记录的信息,快速地构造出一个最优解。<o:p></o:p>

<o:p> </o:p>

【问题】   凸多边形的最优三角剖分问题<o:p></o:p>

问题描述:多边形是平面上一条分段线性的闭曲线。也就是说,多边形是由一系列首尾相接的直线段组成的。组成多边形的各直线段称为该多边形的边。多边形相接两条边的连接点称为多边形的顶点。若多边形的边之间除了连接顶点外没有别的公共点,则称该多边形为简单多边形。一个简单多边形将平面分为3个部分:被包围在多边形内的所有点构成了多边形的内部;多边形本身构成多边形的边界;而平面上其余的点构成了多边形的外部。当一个简单多边形及其内部构成一个闭凸集时,称该简单多边形为凸多边形。也就是说凸多边形边界上或内部的任意两点所连成的直线段上所有的点均在该凸多边形的内部或边界上。<o:p></o:p>

通常,用多边形顶点的逆时针序列来表示一个凸多边形,即P=<v0v1vn-1>表示具有n条边v0v1v1v2vn-1vn的一个凸多边形,其中,约定v0=vn <o:p></o:p>

vivj是多边形上不相邻的两个顶点,则线段vivj称为多边形的一条弦。弦将多边形分割成凸的两个子多边形<vivi+1vj><vjvj+1vi>。多边形的三角剖分是一个将多边形分割成互不重迭的三角形的弦的集合T。图1是一个凸多边形的两个不同的三角剖分。<o:p></o:p>

<o:p> </o:p>

(a)   (b)<o:p></o:p>

1 一个凸多边形的2个不同的三角剖分<o:p></o:p>

在凸多边形P的一个三角剖分T中,各弦互不相交且弦数已达到最大,即P的任一不在T中的弦必与T中某一弦相交。在一个有n个顶点的凸多边形的三角刮分中,恰好有n-3条弦和n-2个三角形。<o:p></o:p>

凸多边形最优三角剖分的问题是:给定一个凸多边形P=<v0v1vn-1>以及定义在由多边形的边和弦组成的三角形上的权函数ω。要求确定该凸多边形的一个三角剖分,使得该三角剖分对应的权即剖分中诸三角形上的权之和为最小。<o:p></o:p>

可以定义三角形上各种各样的权函数ω。例如:定义ω(△vivjvk)=| vivj |+| vivk |+| vkvj |,其中,| vivj |是点vivj的欧氏距离。相应于此权函数的最优三角剖分即为最小弦长三角剖分。<o:p></o:p>

1)最优子结构性质<o:p></o:p>

凸多边形的最优三角剖分问题有最优子结构性质。事实上,若凸(n+1)边形P=<v0v1 vn>的一个最优三角剖分T包含三角形v0vkvn1≤k≤n-1,则T的权为3个部分权的和,即三角形v0vkvn的权,子多边形<v0v1vk>的权和<vkvk+1vn>的权之和。可以断言由T所确定的这两个子多边形的三角剖分也是最优的,因为若有<v0v1vk><vkvk+1vn>的更小权的三角剖分,将会导致T

分享到:
评论

相关推荐

    常用算法设计方法常用算法设计方法

    算法设计是一件非常困难的工作,经常采用的算法设计技术主要有迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法等等。另外,为了更简洁的形式设计和描述算法,在算法设计时又常常采用递归技术,用...

    常用算法分析设计(分治、动态规划、分支限界、回溯、贪婪等等)

    详细介绍了常用的算法设计法,包括:分治、递归、动态规划、分支限界、回溯、贪婪、排列组合、图论算法等等

    常用算法设计方法(C语言)

    C语言的一些常用算法设计方法,包括迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法等等

    算法设计与分析(详细解析(含源代码))

    常用算法设计方法详细解析(含源代码) 算法是问题求解过程的精确描述,一个算法由有限条可完全机械地执行的、有确定结果的指令组成。指令正确地描述了要完成的任务和它们被执行的顺序。计算机按算法指令所描述的...

    常用算法设计方法.doc

    算法设计是一件非常困难的工作,经常采用的算法设计技术主要有迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法等等。另外,为了更简洁的形式设计和藐视算法,在算法设计时又常常采用递归技术,用...

    计算机专业常用算法设计方法

    算法:迭代法\穷举搜索法\递推法\动态规划法等

    常用算法设计方法+搜集

    算法设计是一件非常困难的工作,经常采用的算法设计技术主要有迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法等等。

    C 常用算法设计方法.docx

    算法设计是一件非常困难的工作,经常采用的算法设计技术主要有迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法等等

    常用算法设计方法(C语言).

    算法设计的各种方法都有,如迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法,且有例子

    Matlab常用算法大集合.zip

    Matlab常用算法大集合: Floyd算法.rar 免疫算法.rar 分治算法.rar 动态规划.rar 图论.rar 学习路线.png 搜索算法.rar 概率算法.rar 模拟退火算法.rar 灰色预测.rar 穷举法求解0-1整数规划的matlab程序.rar 类比法....

    算法设计与分析教学大纲

    在能力方面要求:通过本课程的学习,学生要掌握几种常用的算法设计策略,包括递归与分治策略、动态规划算法、贪心算法、回溯法、分支限界法概率算法、线性规划和网络流法和NP完全性理论与近似算法等,并会分析算法的...

    算法设计与分析课件

    第2章介绍常用数学工具,第3章从算法的观点介绍了NP完全理论,从第4章~第12章分别介绍了蛮力法、分治法、减治法、动态规划法、贪心法、回溯法、分支限界法、概率算法和近似算法等算法设计技术。课程中所配算法均给出...

    算法分析与设计 第三讲 分治法

    常用的算法设计技术有迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法和动态规划法等。另外,为了以更简洁的形式设计和描述算法,在设计算法时常采用递归技术,用递归描述算法。 本讲中,主要介绍分治法。

    算法设计与分析PPT(C语言完整版)

    4.5.2动态规划算法设计框架 4.5.3突出阶段性的动态规划应用 4.5.4突出递推的动态规划应用 4.6算法策略间的比较 4.6.1不同算法策略特点小结 4.6.2算法策略间的关联 4.6.3算法策略侧重的问题类型 习题 第5章图的搜索...

    计算机算法设计与分析

    其主要内容包括:算法及算法复杂性基本概念,算法描述,有效算法最常用的设计策略--递归和分治法,动态规划法的设计要点与适用性,贪心算法,回溯法和分支限界法,许多难解问题的高效算法--概率算法,以及NP完全理论...

    C语言版常用算法设计技术

    经常采用的算法设计技术主要有迭代法、穷举搜索法、递推法、贪婪法、回溯法、分治法、动态规划法等等。另外,为了更简洁的形式设计和藐视算法,在算法设计时又常常采用递归技术,用递归描述算法等。

    C++数据结构知识点与经典算法整理

    一、数据结构知识点总结整理 3 2.数据结构的定义: 4 3.数据结构的知识: 9 ...1.7 动态规划法 115 1.8 回溯法 119 1.9 分支定界法: 120 2.几个重要的算法程序 121 2.1 堆排序 121 2.2 归并排序 122

    算法分析与设计c语言实现

    本文讲解了如下常用的算法设计方法 1.1 迭代法  1.2 穷举搜索法  1.3 递推法  1.4 递归法  1.5 贪婪法  1.6 分治法  1.7 动态规划法  1.8 回溯法 并且包含用c语言实现的经典排序查找等算法

Global site tag (gtag.js) - Google Analytics