- 在线时间
- 1029 小时
- 最后登录
- 2017-4-30
- 注册时间
- 2014-1-21
- 听众数
- 213
- 收听数
- 2
- 能力
- 100 分
- 体力
- 15807 点
- 威望
- 98 点
- 阅读权限
- 150
- 积分
- 8571
- 相册
- 0
- 日志
- 0
- 记录
- 3
- 帖子
- 1549
- 主题
- 715
- 精华
- 2
- 分享
- 0
- 好友
- 542
TA的每日心情 | 开心 2017-4-28 17:18 |
|---|
签到天数: 415 天 [LV.9]以坛为家II
 群组: 乐考无忧考研公益讲座 群组: 2017美赛两天强训 群组: 模友会交流视频 群组: 群组: 国赛讨论 |
1.数论算法 # v3 J! h6 ?+ T. r) V* q# l
求两数的最大公约数 + G4 n" T* Y9 {' F9 E$ H8 n
function gcd(a,b:integer):integer;
' n, t, j5 M4 W- M ?% _1 dbegin l: c- x0 B7 G4 w
if b=0 then gcd:=a
& w. L" J/ Y# H4 y Xelse gcd:=gcd (b,a mod b); 6 i- Q& R' k+ Q4 r' J! c g
end ;
% B ~# b1 C @2 ]) V8 Q9 G$ o2 w0 Y' S. e7 {
求两数的最小公倍数 ; R; V. _ M' y! ^
function lcm(a,b:integer):integer; : I5 x5 d1 W& z0 d8 a+ T
begin 2 P' G6 ^; j# L/ i; J$ M
if a< b then swap(a,b); % b! C0 }6 @+ Q
lcm:=a;
4 z- V( x5 X2 {5 owhile lcm mod b >0 do inc(lcm,a);
$ h9 H4 F4 R% m* \end; 5 l; U3 U$ K; L
, D4 W$ `( @) V' m0 r7 Y
素数的求法
- r# q" \' A$ n4 `A.小范围内判断一个数是否为质数: * x1 e6 n% S f w# h# @
function prime (n: integer): Boolean;
4 m0 d( f; Z' p" c: L1 vvar I: integer;
' q1 O* T7 f8 D* G# c, bbegin ' m2 t3 O! A; G8 o
for I:=2 to trunc(sqrt(n)) do
: t5 a% e; k# H8 Pif n mod I=0 then
' e+ }6 a4 [' O; N: ubegin : ]6 r( @7 H& X N7 Y
prime:=false; exit;
1 D0 W' Q. f' x! n) w4 X) Nend;
6 J0 }* a" J' w/ M7 d, S" Eprime:=true; . C& h# q8 o8 ~/ q* |
end; % ~% E X% z$ p
+ r4 c. T2 Z( c* ~4 n
B.判断longint范围内的数是否为素数(包含求50000以内的素数表): : a/ O" a- B9 X) v
procedure getprime; 0 j0 J+ e: ~& U4 ~0 j
var 6 Y: l1 b7 B- A# c
i,j:longint;
" a3 R5 q( w; y* F0 u0 h2 yp:array[1..50000] of boolean; - }* ~# @3 q/ g. g3 P# {! t
begin
/ H5 f8 p5 A+ C3 ~- ^' Y2 |fillchar(p,sizeof(p),true);
/ Z+ k8 ~ A3 ]3 T$ Fp[1]:=false; 3 O7 ]0 A% y( ]5 z
i:=2; . i3 @; G$ Q) d+ Y4 R: w
while i< 50000 do
: e( R- p. w+ ]# v- T: Nbegin
; `& s- n9 r8 ^4 h {6 A6 Yif p then 2 u: I) K8 V, n& I- a+ m8 L. y
begin ) @7 E# |+ S" W' @/ |& v! _! O# B$ s( B
j:=i*2; $ j g1 e9 `4 s
while j< 50000 do . j+ j3 g, v# |- A3 f
begin 8 B* L7 t# l9 G) g7 R, ?: P& c, r
p[j]:=false;
; x' t- u I2 X' M; m- iinc(j,i); : e7 a! O$ W7 m) `2 P9 H8 V
end; _( E0 }; A8 b
end; 2 C8 \' m' g( {0 D$ C5 W1 w. ]0 W
inc(i);
* s5 N. {' g- A/ A+ @. s! Qend; ) _) }0 K+ X; G3 D
l:=0; + P/ T" A8 J% F& K4 m# o
for i:=1 to 50000 do
, q1 h, k& W. G4 S+ o( m- Y+ u. zif p then
; Q G8 {$ |' [4 {2 x" h) @begin
( y! c9 X3 ]' u- ?, ninc(l);4 d4 L- A' F9 S6 f: p( P
pr[l]:=i;
( W9 o* l1 c% ~6 Dend; M4 ]! d" {' m+ ?& B) t3 W: ]. a
end;{getprime}
9 C! n6 x, j: z& _1 ?function prime(x:longint):integer; , _4 c0 L$ ?' `% i
var i:integer;
' H) h$ ]( B# D. Xbegin - O6 R* U& o% k7 X! @1 D& G/ r3 c! I
prime:=false; $ S h: [% A6 D2 f) j# |. C
for i:=1 to l do
- `4 C) y- Y$ a3 zif pr >=x then break $ P2 R, D) L; }7 q, @
else if x mod pr=0 then exit;
! _9 z$ H* h6 Y- M5 O6 j3 Xprime:=true; 7 ^& W9 ~" O: q
end;{prime}
* D6 |/ H9 c C4 V" x- Q
$ ?$ s; E/ D- t% P6 Q, U4 W2.
5 u- I3 r7 }: E3 C+ D% e9 k* Q; F% u% U
3.
0 ~* q3 U. |6 p$ P+ C
" @1 s3 e; g v; r. z4.求最小生成树
" ?7 H" T8 e# A3 mA.Prim算法:
. R4 A6 _" {, q: W6 F( @3 uprocedure prim(v0:integer);
2 l8 a. X* ?. _4 U. G( Pvar
- h" Z: w+ T4 Z" Z$ }; vlowcost,closest:array[1..maxn] of integer;
; z3 z" K1 T: F8 N% [i,j,k,min:integer;
% J7 k: b& g9 W; T; ybegin
; w+ m9 k* g8 @8 g$ ~0 Rfor i:=1 to n do
' s$ y8 M/ l" k5 {* l3 W0 Tbegin 9 @( [5 }" G2 k. s) v
lowcost:=cost[v0,i];
3 g$ J" S# M W$ r' Cclosest:=v0;
& K1 g ~0 K" f6 ^( gend; / [( S" u# `. q. A& Y
for i:=1 to n-1 do
" T; h2 J& G- J, }7 B2 j: Ibegin ) f$ _* v$ m8 F- c4 a( G5 v3 _4 D
{寻找离生成树最近的未加入顶点k} 6 c, _+ ~9 W o9 y$ o
min:=maxlongint; 6 J) i) v2 i+ k6 R
for j:=1 to n do
' T0 |+ f0 A8 I, z; aif (lowcost[j]< min) and (lowcost[j]< >0) then
3 ~6 i$ c% M3 I+ P1 \; T( ~6 u4 Wbegin
5 A; q u) P/ H8 gmin:=lowcost[j]; * j! \2 N0 h3 o I
k:=j; ; t4 ]3 J7 ]0 a: W
end;
2 h# W+ q4 O0 z: rlowcost[k]:=0; {将顶点k加入生成树} % \6 _( k3 z8 m: G n
{生成树中增加一条新的边k到closest[k]} # o% L) b; y+ t6 p5 R& ?
{修正各点的lowcost和closest值} ) u A, z) _' o+ F) ?9 W
for j:=1 to n do
! E( {" L* Q( T& }if cost[k,j]< lwocost[j] then 4 l) H+ e% H7 T( l. O
begin
5 L! I# C5 z+ J7 K( q6 R8 Ulowcost[j]:=cost[k,j];
# [. Z( y% d% q8 l! Kclosest[j]:=k;
& D7 j# h5 L6 g0 A3 n/ Rend;
! O* C7 o ]9 B/ X0 F jend; . }. C9 U% e, H4 J
end;{prim} 9 H" _: }( S7 v3 ~
B.Kruskal算法:(贪心)
7 S1 L4 ]" C6 X1 d4 F* P按权值递增顺序删去图中的边,若不形成回路则将此边加入最小生成树。
, P+ S* e5 ~ x5 C3 \function find(v:integer):integer; {返回顶点v所在的集合}
c8 l) p+ V' j. K( o) M/ xvar i:integer; 1 u* x! O( Q* q, o
begin ( B1 e T0 @: i9 ]' h8 U' p9 x
i:=1; 1 h" Y; _5 G" g: R2 ]
while (i< =n) and (not v in vset) do inc(i);
, c+ R( |; X3 h5 jif i< =n then find:=i
, H7 O# ~$ Y5 k) ?) Y; F) y5 Oelse find:=0;
) B: c1 _* _$ I+ u. a3 @end; ) |* w& Y7 L1 ^5 |; n7 P
procedure kruskal;
# S4 v9 z* _/ u$ q, }var 9 X/ t- }; A4 O" e0 b2 G2 X
tot,i,j:integer;
. I( F& Z7 b& `1 F7 T m+ [) Wbegin : _! q, D; V+ \5 n' s# Q
for i:=1 to n do vset:=;{初始化定义n个集合,第I个集合包含一个元素I} " e: m' t; }7 R4 t. [# k
p:=n-1; q:=1; tot:=0; {p为尚待加入的边数,q为边集指针}
" N- ^9 @' x# j. f- T- u0 osort; : D8 V' g( z: m: y8 u
{对所有边按权值递增排序,存于e[I]中,e[I].v1与e[I].v2为边I所连接的两个顶点的序号,e[I].len为第I条边的长度}
6 V0 R6 Q! Y3 Owhile p >0 do
8 Q4 w# X% m' wbegin 5 A: n H7 x1 H% `
i:=find(e[q].v1);j:=find(e[q].v2); # W7 Y9 B W/ z/ D2 e- F
if i< >j then : Q9 @( o" L/ t. N9 A
begin
& i$ k. _ i, g8 W6 ginc(tot,e[q].len); 0 P: d4 T- X( R( ?. g# c+ V. N e! v
vset:=vset+vset[j];vset[j]:=[]; / }5 j3 J6 S0 T& K3 G: I) j4 w" E
dec(p);
+ {- L/ N: w) ^0 c7 D [end;
9 a2 p( ?* z: `8 linc(q); ( K) I$ b0 ~$ ]
end;
$ l. n# |$ C# Y/ Awriteln(tot);
; j: B( N. b1 L' d* C1 U" |- rend; ; [/ l9 m" r7 X" K
- _/ @( v6 C- B. c6 `2 x0 o7 d( M
5.最短路径 * i+ U8 d7 m, x1 n. `% h
A.标号法求解单源点最短路径:
. }7 s. C1 U# B t% z: M. }var $ C1 Q, q: j3 N
a:array[1..maxn,1..maxn] of integer;
, K1 W" ^$ @1 m: ~2 O: ]b:array[1..maxn] of integer; {b指顶点i到源点的最短路径} 9 y/ V& y. R1 w6 C( \! l
mark:array[1..maxn] of boolean; 9 M7 D/ t. J* [
( e5 @/ Z V6 J' A* j0 F4 Oprocedure bhf;
$ ^7 z" E' k/ y. T+ ?$ y$ {9 Nvar
+ x: q [+ ?- d7 J! S* I% Jbest,best_j:integer;
+ {8 j4 n! `. X) u- s3 t" dbegin 0 J2 p6 s! c( p8 t7 }: k3 C
fillchar(mark,sizeof(mark),false);
$ w# A, k3 u# }mark[1]:=true; b[1]:=0;{1为源点} ) A. A- O$ q# h1 t' \0 s# F
repeat
2 D' s; A9 d9 u3 b% Zbest:=0; $ B( N5 p6 E, b7 m6 W1 g. P j7 G3 W
for i:=1 to n do ' E* f5 K, v/ t' J g( y& K- X, c
If mark then {对每一个已计算出最短路径的点}
6 ^% E& K5 F2 D7 y- Sfor j:=1 to n do
7 T' j7 W, U3 M3 u0 h; x( lif (not mark[j]) and (a[i,j] >0) then
0 t8 h U) i4 q9 C; C& iif (best=0) or (b+a[i,j]< best) then " N! J. y% s+ l. g+ x5 @
begin
& Y. k& ^) P& L9 P* M* kbest:=b+a[i,j]; best_j:=j; 9 E4 g9 `* o1 w$ s" }3 d- C
end; 3 F E, W8 T# H- T6 L
if best >0 then
& B2 R7 H8 H; [. obegin ' u5 T# k) u" N$ m
b[best_j]:=best;mark[best_j]:=true;
M4 v- \& T& W3 A2 ]7 hend;
# l* |3 g# T, J9 Puntil best=0;
% z; l: M! [, m0 S* `end;{bhf} / o0 c0 u* p( X' v
. p# Z" b# D Y8 M5 `- M3 J) lB.Floyed算法求解所有顶点对之间的最短路径: 2 \6 O$ k1 B/ a# |. p
procedure floyed; ( v7 d8 U% S- \0 q$ a
begin
/ g/ \+ r0 X' p: w" B c6 u3 q: nfor I:=1 to n do
; c8 d- G9 a4 t" X; Z5 Y' {! C8 lfor j:=1 to n do
0 p$ e7 C' l1 i/ i( \0 H- zif a[I,j] >0 then p[I,j]:=I else p[I,j]:=0; " b# B0 N* P7 D: h2 V( z$ Y0 J8 |
{p[I,j]表示I到j的最短路径上j的前驱结点}
v5 c$ {( K( Q O: ~7 G) _for k:=1 to n do {枚举中间结点}
$ V, ?5 [! y* T7 h% V7 kfor i:=1 to n do / B3 j1 j2 N- A( N; B% a( g4 a
for j:=1 to n do
" J/ m9 j2 D; Y) ]6 M0 d- Z# Qif a[i,k]+a[j,k]< a[i,j] then
& X) R8 n! U' i/ v$ v6 A) x( abegin M' q4 E3 d9 | [- e+ U& `' G) X' g
a[i,j]:=a[i,k]+a[k,j]; $ k8 V8 x9 U' g# H$ }2 n) v
p[I,j]:=p[k,j]; 6 c5 H. x' Q% b5 i$ }0 j$ R* I
end; . r5 m" q- t5 |! w: m
end; 3 w2 m3 L _8 h
C. Dijkstra 算法:
. y0 w% t# a0 O% E类似标号法,本质为贪心算法。 6 y, W% c: d: e( Q9 w9 [ j
var % j6 u, }# P' K) [3 t% x
a:array[1..maxn,1..maxn] of integer; " W/ I- |. p+ z
b,pre:array[1..maxn] of integer; {pre指最短路径上I的前驱结点}
7 v ], h/ c& N2 j4 d; Amark:array[1..maxn] of boolean; 2 o4 e0 J' n$ K3 h- f& S
procedure dijkstra(v0:integer); w% D4 H7 r2 R, j: G
begin
% j, \7 f3 S2 [1 p5 Qfillchar(mark,sizeof(mark),false); \$ I! Z1 K7 J3 ^( Q0 _( K
for i:=1 to n do
2 g, h$ X: S! G" P- Dbegin 2 R4 k( |) I, p! Y5 h+ F( f( P& z
d:=a[v0,i];
. o! P2 e) |) `if d< >0 then pre:=v0 else pre:=0;
8 @/ h$ ~; w& uend;
5 E$ C& v& i8 b+ a4 Jmark[v0]:=true;
9 }) X$ ~+ N6 Xrepeat {每循环一次加入一个离1集合最近的结点并调整其他结点的参数}
! f; V. F6 [* g+ K0 gmin:=maxint; u:=0; {u记录离1集合最近的结点}
. L, A& H/ S5 J* a4 ]3 j4 P, `for i:=1 to n do 9 h/ {. F! H, K% `/ u4 b
if (not mark) and (d< min) then ' j, \& T8 W) ~
begin & K' q# l8 \- v& T% u: I' i) k
u:=i; min:=d;
8 q$ L2 x; H, [ O+ P2 bend;
( L) a6 B" |% _! G7 i( K, D7 ?if u< >0 then ) w% Z# y4 N% W5 @% e
begin
3 V9 R# ]9 ]. W8 U2 v, |8 nmark:=true;
6 O- B4 [3 Q; ofor i:=1 to n do % M! P1 P9 G9 u/ ]4 I0 d' P/ ]
if (not mark) and (a[u,i]+d< d) then 3 x4 F2 O+ r! ~; P2 ^0 X
begin
3 C. h$ L1 L0 ad:=a[u,i]+d;
% r) F. Q% f7 Dpre:=u; 6 I8 K1 n8 m. }+ F' G: D# E% o3 Z
end; 0 ^& A% J1 p7 S5 i) q; w
end;
6 n; ~: K0 z e2 l4 v8 luntil u=0;
2 B2 q1 ]' j* \5 Q8 aend;
, ]" V5 K- b: z7 ]# V6 KD.计算图的传递闭包
. B1 g( ^3 f B3 ]Procedure Longlink; : a1 i- G: p( E/ r1 u+ ]: i7 E
Var
5 V# R. E O5 W# u! y# OT:array[1..maxn,1..maxn] of boolean; " u9 q+ R- e. O2 l
Begin
% V; N' R3 ^% d. |0 G7 }8 T0 tFillchar(t,sizeof(t),false); P# G$ Z# N& U, S
For k:=1 to n do n W6 w* P" q7 C
For I:=1 to n do % {" g, i! f1 l" I S$ u9 I+ ?
For j:=1 to n do 7 q9 E: E% t: p9 ~( `7 O
T[I,j]:=t[I,j] or (t[I,k] and t[k,j]);
4 X0 V p1 n% VEnd;: N7 ~( _( y8 \& Q' E( p: K
7 Q7 v. e$ |9 L! [; v+ v+ w |
zan
|