获得本站免费赞助空间请点这里
返回列表 发帖

[Zend权威认证试题讲解]第二章-用PHP4进行面向对象编程

尽管PHP4的OOP性能不强,但它还是能够被用来构建可行的面向对象的代码结构——只要你知道对象模型的缺陷,并且小心的处理它们。
3 _* f& C# e4 P: APHP5在对象的处理方面做了很多改变,你或许会因此更倾向于完全忽略PHP4。但事实上,许多用OOP的程序员从很早以前就开始用老版本的PHP编写软件了。所以,大量的OOP代码早已存在,甚至在人们跳到PHP5上进行开发之前。' F" s8 ]9 m: l# h8 j4 {
本章不仅考察你对面向对象知识的总体掌握,还包括对PHP4特有的OOP实现机制的认识。
0 r# I) i7 S4 w% O  k# Y, a
9 a" K' N6 ]0 q问题7 h- `6 P  l8 r
/ G* X5 t4 h3 e$ O
1.对象的蓝图是什么?: y# C9 |/ ?. ?

- A7 z5 t# L* a答案:____________
5 D. T- @# B$ U* F, g& T4 b  t
' M, C' J' T9 E% h7 O+ t, ~
# ]- {* `4 Q/ N3 R. d" @2.以下代码执行后,数组$a->my_value中储存的值是什么?(三选)

  1. <?php
  2. class my_class
  3. {
  4.     var $my_value = array();
  5.     function my_class ($value)
  6.     {
  7.         $this->my_value[] = $value;
  8.     }
  9.     function set_value ($value)
  10.     {
  11.         $this->$my_value = $value;
  12.     }
  13. }
  14. $a = new my_class ('a');
  15. $a->my_value[] = 'b';
  16. $a->set_value ('c');
  17. $a->my_class('d');
  18. ?>
复制代码
A.c! h0 S4 x% O+ \& a" r% _1 I
B.b
0 R1 \- f! q, `- z1 E0 wC.a% s5 {9 G0 p& Z% O
D.d) p5 y: k# a6 c
E.e" A; u: i, L: c* Q6 n9 h
; L9 _/ x' L& B/ D7 w2 b
' @$ T% R/ L& u: r
3.如何让类中的某些方法无法在类的外部被访问?7 L2 x2 o; ~; R1 F* p, H
$ D, I( Q9 `# a/ m
A.把类声明为private
: ~/ \( a1 e- |* [- AB.把方法声明为private, @1 t7 A$ f( r: C8 F8 y
C.无法实现- c" q$ h. @- d; T2 y1 g, P
D.编写合适的重载方法(overloading method)3 t8 C' M1 }( K& E

7 Q& q- `, S5 Z, h4 H/ q$ e, i8 D
4.哪种OOP设计模式能让类在整个脚本里只实例化一次?
) w( \4 K, S  [) Z& Z5 F$ m3 o7 J/ T, }, D- ^: \
A.MVC模式: P' W7 l% k3 l! c! J# G) k9 t% J
B.抽象工厂模式(Abstract factory)
8 R  ~8 A5 C0 G1 ^$ O$ {9 UC.单件模式(Singleton)
& r; `( p7 C+ j4 Q0 R6 Z) }D.代理模式(Proxy)
. ]$ g0 q( L5 y! Y1 V( J% F! sE.状态模式(State)1 d& T3 Z* |/ A; w
! G- q; g- @* A) _: K% S
9 j! y9 [7 X% `- y
5.借助继承,我们可以创建其他类的派生类。那么在PHP中,子类最多可以继承几个父类?
# G; L5 ~5 A) D1 b1 n3 \3 N' z  E& f2 o- j% t( k7 x
A.1个
* |* z; H0 Z0 r5 n! _4 X6 ?* G' XB.2个+ m+ `+ S, Z4 c
C.取决于系统资源
; O* O$ Y2 g) W9 V4 wD.3个
% F1 S; v% B3 O$ mE.想要几个有几个
2 }" B, a% L" K2 u4 \8 S2 k5 U+ J+ c" @0 V6 _. Q8 t# c
4 K: Q6 ^& s! z7 F0 `) T; x
6.以下脚本近似的表示了一种在PHP4中无法实现的特性,请问这个特性叫什么?

  1. <?php
  2. class my_class
  3. {
  4.     function my_funct ($my_param)
  5.     {
  6.         user_error ("Please define me", E_ERROR);
  7.     }
  8.     function b()
  9.     {
  10.         return 10;
  11.     }
  12. }
  13. ?>
复制代码
A.多重继承
7 `( {. L$ `' O9 X6 |2 m) T) z7 WB.接口
7 h6 t2 g/ F6 o8 d' g$ n7 yC.抽象方法
6 @& S& r% Z! J; L1 H2 O1 b" i* F  Y1 XD.Private方法; W# h  }5 \) @+ z
E.函数重载(function overloading)( [( V  [2 U7 Y/ ]1 K

6 }: m% Y5 u  r- P% S9 z/ T) s. M0 k8 b
7.假设定义了一个testclass类,它的构造函数的函数名是什么?* q; y, B( B/ p8 b+ W1 w
% _" f1 D' G6 v4 g0 j. B
A.__construct& Y& P9 s+ }" _0 q! k
B.initialize
+ C5 o9 `& |1 ?8 U9 ~C.testclass
* h, O! h8 g- [' J# AD.__testclass
% N2 T$ e3 R0 o. @3 `. XE.只有PHP5才支持构造函数
% ]  F, W: t1 E& {- h
7 u" c- \/ R% T+ _4 s& D, ]3 R+ n. y
  H) i: c- h% K9 Z. s5 `8.一个类如何覆盖默认的序列化机制?
0 [" Y  t. O; Y4 c+ H' a7 \
8 F8 u5 P. e1 n% e8 h- {A.使用__shutdown和__startup方法
; `: U9 N5 ~" i7 HB.调用register_shutdown_function()函数$ a/ s. s0 p1 o1 u6 z& B& k3 j
C.使用__sleep()和__wakeup()方法
5 y: n! [2 R! W- }0 m# |0 a5 j  YD.无法覆盖默认序列化机制
* W% V' {- G4 p+ Z0 iE.使用ob_start()将类放入输出缓冲中3 w4 _- B- v0 f' o* q, z! Z

) m  Q7 N: V4 c; j3 S' i/ g3 h* A7 |4 }4 g# p  \
9.以下哪些面向对象的概念无法在PHP4中实现?
$ J" q% N# i* a3 p! |$ v7 Q% O9 u. n& Z3 E
@抽象类7 S! e, V7 j# R
@Final类4 Y3 d! Q: r! j, H6 m( k
@Public、private、protected(PPP)方法
7 B" {  a1 @% {. F@接口8 Q9 j$ `. T& p/ c, c

3 q# g2 ]  M0 ^+ v% U& z. K. ^A.抽象类1 e7 {6 H+ M0 V
B.PPP方法- x+ X: D2 z( I6 o; C
C.PPP方法和接口
  f" C1 t* Y' |8 C% }* c# HD.以上所有都不可用
4 {1 u7 i  r( t+ W) x) pE.以上所有都可用
6 Y6 a$ i4 e4 @9 `+ L1 d" H9 p4 E+ E$ \8 c8 _" i

, f; {' j8 h: b4 B* N7 q7 w10.如何在类的内部调用mymethod方法?/ a5 t7 y: O& M

- m& B! n3 z! Z2 cA.$self=>mymethod();
; A% g8 i' W# p/ F* M  m4 oB.$this->mymethod();: ^3 `# ]5 B' E2 C3 E, P
C.$current->mymethod();7 o5 w8 K. p& _3 `7 e7 O' x1 j8 s
D.$this::mymethod()
; m. N- X; R5 LE.以上都不对
9 ~# z/ R, }) D
! {9 A' P2 S$ E3 y& b
) ~$ U5 q% D; ~6 o11.以下脚本输出什么?

  1. <?php
  2. class my_class
  3. {
  4.     var $my_var;
  5.     function _my_class ($value)
  6.     {
  7.         $this->my_var = $value;
  8.     }
  9. }
  10. $a = new my_class (10);
  11. echo $a->my_var;
  12. ?>
复制代码
A.10
" ]; i  _, w3 V% m3 c/ ]0 RB.Null  x/ ~4 c( y* X# x; J2 x& t1 X
C.Empty
( L/ X# k; w$ V9 \2 D5 {D.什么都没有2 q) H9 v* H4 h9 x
E.一个错误
, x( a- o: t4 E4 h( c2 K
3 O# e- G: Q* W8 A6 R; D/ |0 Q8 V  [. ^$ O4 d
12.以下脚本输出什么?

  1. <?php
  2. class my_class
  3. {
  4.     var $value;
  5. }
  6. $a = new my_class;
  7. $a->my_value = 5;
  8. $b = $a;
  9. $b->my_value = 10;
  10. echo $a->my_value;
  11. ?>
复制代码
A.10  B, ], e- e1 J% K) X: O5 u5 l
B.5
! s% O6 i1 C8 [' E, @6 a1 e- f- vC.2
. Y6 y: H0 G+ @* h6 O1 tD.Null
' |3 q  f3 q3 ~2 B! IE.什么都没有3 ]. `( a! H$ ?5 Q1 N, t

# Q: \) o; n; Y! \* S' @- c/ \0 ?& D4 y( V0 h; ]; s0 i
13.以下脚本输出什么?

  1. <?php
  2. $global_obj = null;
  3. class my_class
  4. {
  5.     var $value;
  6.     function my_class()
  7.     {
  8.         global $global_obj;
  9.         $global_obj = &$this;
  10.     }
  11. }
  12. $a = new my_class;
  13. $a->my_value = 5;
  14. $global_obj->my_value = 10;
  15. echo $a->my_value;
  16. ?>
复制代码
A.5
1 j, V1 V0 M/ l& DB.102 E& p$ ]8 n" K
C.什么都没有
) G; w3 v; e3 q. mD.构造函数将报错
! a' `' @  G& v: M" L9 d& O% u3 e6 [E.5100 a' n' p6 p) q& o/ u
: d* d* v- p  G# j& R  b/ J
7 c/ G% l  M( c) A5 O
14.考虑如下一段代码,执行时,$eight_tenths->to_string方法返回的字符串是8/10而不是希望的4/5,为什么?

  1. <?php
  2. class fraction {
  3.     var $numerator;
  4.     var $denominator;
  5.     function fraction($n, $d) {
  6.         $this->set_numerator($n);
  7.         $this->set_denominator($d);
  8.     }
  9.     function set_numerator($num) {
  10.         $this->numerator = (int)$num;
  11.     }
  12.     function set_denominator($num) {
  13.         $this->denominator = (int)$num;
  14.     }
  15.     function to_string() {
  16.         return "{$this->numerator} / {$this->denominator}";
  17.     }
  18. }
  19. function gcd($a, $b) {
  20.     return ($b > 0) ? gcd($b, $a % $b) : $a;
  21. }
  22. function reduce_fraction($fraction) {
  23.     $gcd = gcd($fraction->numerator,
  24.     $fraction->denominator);
  25.     $fraction->numerator /= $gcd;
  26.     $fraction->denominator /= $gcd;
  27. }
  28. $eight_tenths = new fraction(8,10);
  29. /* Reduce the fraction */
  30. reduce_fraction($eight_tenths);
  31. var_dump($eight_tenths->to_string());
  32. ?>
复制代码
A.reduce_fraction函数必须返回一个值
4 I- z9 y7 E- z: W$ k$ GB.reduce_fraction函数必须接受一个整型值/ y! d$ u/ o- v2 @& _* |* f
C.gcd函数有问题* [& l, w! O6 S. T/ X
D.必须通过引用的方式传递$eight_tenths对象; P0 X" Q$ v: c, {+ z2 z4 x9 L
E.对象的实例不能传递给方法以外的其他结构。
' Q9 z! Y7 _& @3 j) i* _
1 n' z' \7 I# x7 c8 |
8 H! M! ]1 C( c' S4 Z15.以下代码是做什么的?

  1. <?php
  2. require_once("myclass.php");
  3. myclass::mymethod();
  4. ?>
复制代码
A.静态调用mymethod方法, c0 t' i1 _8 [/ P5 l* z. g. |
B.生成myclass的实例并调用mymethod方法
2 J# G9 V. A/ SC.产生一个语法错误
6 o7 E$ J3 x+ i- @$ XD.默认myclass类最后被创建出的实例并调用mymethod(), L' |. y& g; H, N6 ^
E.调用名为myclass::mymethod()的函数1 h0 R( v" j( L# M8 P

7 N; {, |4 v5 E' s2 E* x2 K/ C7 b& E) k; M2 S) T( e
16.PHP中有静态类变量吗?
3 e: X, h' R+ H' F# s( g( r5 u
6 k+ U% F" z$ u$ ~A.有
3 M) O1 x9 z& t# dB.没有
: ?  _  x4 D3 M2 o, G8 X6 h. V5 H6 H+ a- ~
$ t2 ?- o1 c* ]- ?
17.以下脚本输出什么?

  1. <?php
  2. class a
  3. {
  4.     function a ($x = 1)
  5.     {
  6.         $this->myvar = $x;
  7.     }
  8. }
  9. class b extends a
  10. {
  11.     var $myvar;
  12.     function b ($x = 2)
  13.     {
  14.         $this->myvar = $x;
  15.         parent::a();
  16.     }
  17. }
  18. $obj = new b;
  19. echo $obj->myvar;
  20. ?>
复制代码
A.1
4 ^2 |2 W$ L' i( F1 c# T% r. SB.2
+ z* v3 F/ H& T9 gC.一个错误,因为没有定义a::$myvar
, {& m9 l, n9 D* MD.一个警告,因为没有定义a::$myvar, k9 `& S+ k0 k- G! \. N
E.什么都没有9 H% u4 ^0 H" ^; z
/ ~% n! n% P  J  H  a
3 l2 i5 i1 |* x( Q- y
18.如何即时加载一个类?0 ]4 \: t1 w' ?# u6 x8 S. ?
! g8 z, j1 W  N' b+ ?
A.使用__autoload魔术函数
. w, Y4 N, Z: p  G, [# bB.把它们定义为forward类
) N* F, M4 H/ ?$ X" fC.实现一个特殊的错误处理手段
  _' G- x& |+ q. U- a6 X- `, Z* ID.不可能& d7 Y0 S+ [$ p0 k" I# ~
E.用有条件限制的include来包含它们
0 S; k0 c6 p/ f- Z* M: ?1 }: M$ r' b8 Q# b) C. N3 _

& Z7 e. I+ c) e9 f) ?) v19.__________提供了一个高性能的解决面向对象中重复出现的问题的方案?% \4 b1 i# V. N3 s9 g2 M% O
& R) H' C$ X" g7 d
    答案:__________' J7 _5 a# d& E) b
0 d$ L5 ^! j7 c& m  s+ A( O8 y

* `; O  b* V% _, @* J, c5 K- x+ f20.以下脚本输出什么?

  1. <?php
  2. class a
  3. {
  4.     function a()
  5.     {
  6.         echo 'Parent called';
  7.     }
  8. }
  9. class b
  10. {
  11.     function b()
  12.     {
  13.     }
  14. }
  15. $c = new b();
  16. ?>
复制代码
A.Parent called# T/ ^; Z" F& h8 O
B.一个错误
+ L6 n0 `% C, Y+ q/ D& K& V8 G7 Z8 jC.一个警告, x; c& A$ \0 D$ I8 r
D.什么都没有
- F/ E) h) ^9 n$ z* i3 s
% M. ~7 h6 H. P$ j6 p" C
& z+ _& y/ Z/ U% V) [6 k* x! d% \
7 ~# E: M" z( m# U& }3 N1 y答案速查# N9 G3 y5 M# _. H: v& K; e7 W
1.类" ~  m2 }" t' d2 Z8 M. Z
2.BCD
% N- @. G% }, L. q! [0 [) a3.C5 U: B3 h3 ]% J6 p9 m( f+ b
4.C% e+ l) N! Y+ [1 t  b7 P
5.A
) F9 C4 d6 L/ y  e7 h6.C
% G7 t( G2 ], e" g7 `7 D* X7.C
7 ?; F  A  K( n# M* u4 a( A- I( f. b8.C9 g$ J5 y$ V$ H/ F2 Q8 |- o9 c
9.D
  V4 b8 \" t" ]. B$ Y, \10.B
7 I) r# `+ U$ H7 V( O11.D
5 j3 I* a+ y( v8 K12.B
! [" |9 I3 `1 v. G6 P4 E  n13.A
' e  T8 y$ s& s8 E* I$ t/ U14.D
, ], l$ B: x) p0 c" n15.A
" X4 `3 ^; ~! q& k4 [1 U16.B) P; {% c! r) }4 F/ Q. ^
17.A
/ D' a1 a5 _* H" I$ }18.D5 C& p. b1 r! s+ ~% c% `3 }
19.设计模式8 B6 [7 E( v& q3 ^7 M
20.D
' N& I# y7 m1 M1 o- j8 B5 c1 W% }% B

2 O8 W  y. u2 `) R* r( f. Z$ ?4 F2 ]3 ?' e8 `
答案详解& c4 k9 l$ l( g! v( z) }
6 K* G0 T8 r3 R/ j# Y$ j4 m
1.类是对象的蓝图(对象是类的实例)。; L' K% U) t. D9 i/ t4 t# T

1 K) J; t* |' V! R& T6 ]  I: [7 |( t2.正确答案是B、C和D。set_value方法使用了错误的表达式$this->$my_value,因此该方法实际上是空的(这在PHP5里会导致一个错误,但在PHP4中不会。——译者注)。
/ ?1 l* t. k/ C. F: ~' w2 j) E3 k; Y( Y. x# x  g: f1 H
3.答案是C。PHP4中无法限制对类成员的访问,而在PHP5中则可以通过private关键字实现。
! ]/ x0 b, E" _* J0 v" u: P! i" \# }) A4 W. X$ z5 n" d
4.单件模式可以限制一个类被实例化的次数。6 @! c( C/ ~5 c0 y$ `6 w# m. ]5 x

- _0 n, U, ^# s" V7 m5.尽管其他编程语言允许多重继承,但在PHP的对象模型中却不可以。因此答案是A。  P. y; p! c/ n7 E

! {! @6 l# B1 U6.方框中的代码表现的是抽象方法(abstrace method)的实现。如果这个类继承自其他类,而my_funct方法在子类中被调用时没有覆盖,代码将抛出一个错误。虽然只是近似的实现了抽象方法,但在PHP4有限的对象模型中,这已经做得很好了。+ @: Y9 Z  v+ N! R2 p0 E, ^0 a# l

% O9 l  o* q6 r( G, f' P6 U7 p; N7.PHP5有统一的构造函数(__construct()),但在PHP4中,构造函数就是和类有相同名称的方法。对于名为testclass的类,它的构造函数就是testclass()。答案是C。
% x3 q1 P  P, b" C2 y2 }8 {/ _+ z, A) _
8.__sleep()和__wakeup()能被用来自定义对象的序列化过程。正确答案是C。
' x% k0 a% ], i% l4 }2 e9 A$ Q1 N
9.PHP4中没有题目选项里所列的任何一个概念。答案是D。
- v. K, r* K$ d
! _& [7 F; i1 z8 B. l  D6 G8 g) {10.PHP中,在类的内部访问其成员和方法,要用$this这个特殊变量。因此答案是B。
9 @# q' J2 i, _& ]- |# Q9 @. Y  _0 k6 I: k
11.正确答案是D。my_class::_my_class()不是合法的构造函数(方法名的开头多了个_),因此脚本不会输出任何东西。你可能觉得这题是在考眼力而不是知识,是的,我们就是这么打算。仔细想想你就会同意——绝大多数的bug都是由错误的拼写造成的。这题并不是在戏弄你,而是考验你的排错能力。
& T/ l+ X; Z: K! [; j8 ^8 k/ C# C7 a) Z  i
12.PHP4把对象视作标量进行处理,当$a赋给$b时,解释器创建对象的副本,因此对后一个对象的赋值不会影响到原先的对象。答案是B。但是要注意,PHP5里就不是这样处理的了(将会输出10)。
& X# I, R+ [4 H0 T
9 [8 s5 V  Z3 f: c13.一上来,构造函数my_class通过引用,将自身存储在了变量$global_obj中。你可能会因此觉得,当我们后来吧$global_obj->my_value的值变为10时,$a也会相应改变。不幸的是,new操作符返回的不是引用,而是对象的副本。脚本输出5,答案是A。. B3 D% o2 q- j' t4 c9 W: ]
. \. ?6 h/ M: J' B! z
14.PHP中,把对象传递给函数或者方法时,默认传递的是值。这意味着通过参数传递给函数的对象,其实是对象的副本。这点导致了在函数或方法里对对象进行改动时,不会影响函数外的原先的那个对象。
! m# }$ ^  t8 n$ A回到第14题中,这就说明对象$eight_tenths从来没有被reduce_fraction函数改动过,而$fraction对象(参数)则被改动了。如果要在函数内部改动对象,就必须以引用的方式传递参数:/ n7 g0 r' G: ]2 q. U/ W, L
    function reduce_fraction(&$fraction)0 e* P. V5 B) w+ D( E& q- \4 Q
答案是D。9 K  q  D0 ?& z! b
2 T! p7 N' `0 k' _/ l9 T: b
15.题中所示的语法是用来进行静态调用的。当方法被静态调用时,它们就像一个独立的函数,与任何类的实例无关。答案是A。; H# R# P  W; u+ A' U0 Y
- E9 t  ]& i' S8 X! i9 u: y$ r( A3 J
16.没有。PHP4只允许声明静态函数变量,没有静态类变量。5 v! E6 Z. p$ l% D
" c3 |& c& K. O8 K
17.答案是A。类b的属性$myvar将在b的父类——类a调用构造函数时被定义,此外,像PHP4中的普通变量一样,定义类变量时也不需要给它赋值。类b在其父类调用构造函数之前就给$myvar赋了值,所以不管之后如何赋值,输出都是1。' b; @" X7 A7 r. |% C: p

7 A& I2 I) {3 k  Z& T18.PHP4中无法即时装载类——它们必须在使用前就仔细声明好。PHP5中,可以使用__autoload魔术函数提醒解释器在找不到需要的类时尝试自动调用。因此答案是D。+ Y( i/ b$ [) q- b# n: A
! X1 T, P/ T6 b1 g, {7 J# ]
19.为软件设计和编程中的常规问题提供良好的解决方案,这显然是在说设计模式。
3 I6 Q0 T3 ^4 R6 K  U( Q1 @0 W# w  u+ U7 o. d9 n, u1 \+ P7 i+ n0 R
20.脚本什么都不输出(答案是D)。因为子类的构造函数不会自动调用父类的构造函数。

返回列表
【捌玖网络】已经运行: