Zend_Db_Adapter.xml 118 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138
  1. <sect1 id="zend.db.adapter">
  2. <title>Zend_Db_Adapter</title>
  3. <para>
  4. Zend_Db и его родственные классы предоставляют простой интерфейс к базам
  5. данных SQL в Zend Framework. Zend_Db_Adapter является базовым классом,
  6. который должен использоваться для подключения приложения PHP
  7. к СУРБД. Существуют различные классы адаптеров для наиболее часто
  8. используемых СУРБД.
  9. </para>
  10. <para>
  11. Адаптеры Zend_Db создают мост между общим интерфейсом и расширениями
  12. PHP от конкретных поставщиков для того, чтобы можно
  13. было единовременно писать приложения на PHP и затем переключаться между
  14. различными СУРБД с наименьшими усилиями.
  15. </para>
  16. <para>
  17. Интерфейс класса адаптера подобен интерфейсу расширения
  18. <ulink url="http://www.php.net/pdo">PHP Data Objects (PDO)</ulink>.
  19. Zend_Db предоставляет классы адаптеров к драйверам PDO для следующих
  20. популярных СУРБД:
  21. </para>
  22. <itemizedlist>
  23. <listitem>
  24. <para>
  25. IBM DB2 и Informix Dynamic Server (IDS), с использованием
  26. расширения <ulink url="http://www.php.net/pdo-ibm">pdo_ibm</ulink>
  27. </para>
  28. </listitem>
  29. <listitem>
  30. <para>
  31. MySQL, с использованием расширения <ulink url="http://www.php.net/pdo-mysql">pdo_mysql</ulink>
  32. </para>
  33. </listitem>
  34. <listitem>
  35. <para>
  36. Microsoft SQL Server, с использованием расширения <ulink url="http://www.php.net/pdo-mssql">pdo_mssql</ulink>
  37. </para>
  38. </listitem>
  39. <listitem>
  40. <para>
  41. Oracle, с использованием расширения <ulink url="http://www.php.net/pdo-oci">pdo_oci</ulink>
  42. </para>
  43. </listitem>
  44. <listitem>
  45. <para>
  46. PostgreSQL, с использованием расширения <ulink url="http://www.php.net/pdo-pgsql">pdo_pgsql</ulink>
  47. </para>
  48. </listitem>
  49. <listitem>
  50. <para>
  51. SQLite, с использованием расширения <ulink url="http://www.php.net/pdo-sqlite">pdo_sqlite</ulink>
  52. </para>
  53. </listitem>
  54. </itemizedlist>
  55. <para>
  56. Кроме этого, Zend_Db предоставляет классы адаптеров, использующие расширения PHP для следующих распространенных СУРБД:
  57. </para>
  58. <itemizedlist>
  59. <listitem>
  60. <para>
  61. MySQL, с использованием расширения <ulink url="http://www.php.net/mysqli">mysqli</ulink>
  62. </para>
  63. </listitem>
  64. <listitem>
  65. <para>
  66. Oracle, с использованием расширения <ulink url="http://www.php.net/oci8">oci8</ulink>
  67. </para>
  68. </listitem>
  69. <listitem>
  70. <para>
  71. IBM DB2, с использованием расширения <ulink url="http://www.php.net/ibm_db2">ibm_db2</ulink>
  72. </para>
  73. </listitem>
  74. <listitem>
  75. <para>
  76. Firebird/Interbase, с использованием расширения <ulink url="http://www.php.net/ibase">php_interbase</ulink>
  77. </para>
  78. </listitem>
  79. </itemizedlist>
  80. <note>
  81. <para>
  82. Все адаптеры Zend_Db используют расширения PHP. Вы должны иметь
  83. включенным соответствующее расширение в вашей среде PHP для
  84. использования адаптера Zend_Db. Например, если вы используете
  85. какой-либо из адаптеров PDO Zend_Db, то нужно включить как
  86. расширение PDO, так и драйвер PDO для используемой вами СУРБД.
  87. </para>
  88. </note>
  89. <sect2 id="zend.db.adapter.connecting">
  90. <title>Соединение с БД с использованием адаптера</title>
  91. <para>
  92. Этот раздел описывает, как создавать экземпляр адаптера БД. Это
  93. соответствует созданию соединения с сервером СУРБД из вашего
  94. приложения PHP.
  95. </para>
  96. <sect3 id="zend.db.adapter.connecting.constructor">
  97. <title>Использование конструктора адаптера Zend_Db</title>
  98. <para>
  99. Вы можете создавать экземпляр адаптера с помощью его
  100. конструктора. Конструктор адаптера принимает единственный
  101. аргумент, который является массивом параметров, использующихся
  102. для описания соединения.
  103. </para>
  104. <example id="zend.db.adapter.connecting.constructor.example">
  105. <title>Использование конструктора адаптера</title>
  106. <programlisting language="php"><![CDATA[
  107. $db = new Zend_Db_Adapter_Pdo_Mysql(array(
  108. 'host' => '127.0.0.1',
  109. 'username' => 'webuser',
  110. 'password' => 'xxxxxxxx',
  111. 'dbname' => 'test'
  112. ));
  113. ]]>
  114. </programlisting>
  115. </example>
  116. </sect3>
  117. <sect3 id="zend.db.adapter.connecting.factory">
  118. <title>Использование фабрики Zend_Db</title>
  119. <para>
  120. Вместо непосредственного использования конструктора
  121. адаптера можно создавать экземпляры адаптера, применяя
  122. статический метод <code>Zend_Db::factory()</code>. Этот метод
  123. динамически загружает файл класса адаптера, используя
  124. <link linkend="zend.loader.load.class">Zend_Loader::loadClass()</link>.
  125. </para>
  126. <para>
  127. Первым аргументом является строка с базовым именем класса
  128. адаптера. Например, строка 'Pdo_Mysql' соответствует классу
  129. Zend_Db_Adapter_Pdo_Mysql. Вторым аргументом является тот же
  130. массив параметров, который вы должны были бы передать
  131. конструктору адаптера.
  132. </para>
  133. <example id="zend.db.adapter.connecting.factory.example">
  134. <title>Использование метода-фабрики адаптеров</title>
  135. <programlisting language="php"><![CDATA[
  136. // Нам не нужно использовать следующее предложение, поскольку
  137. // файл Zend_Db_Adapter_Pdo_Mysql будет загружен через
  138. // метод-фабрику Zend_Db
  139. // require_once 'Zend/Db/Adapter/Pdo/Mysql.php';
  140. // Автоматически загружает класс Zend_Db_Adapter_Pdo_Mysql
  141. // и создает его экземпляр.
  142. $db = Zend_Db::factory('Pdo_Mysql', array(
  143. 'host' => '127.0.0.1',
  144. 'username' => 'webuser',
  145. 'password' => 'xxxxxxxx',
  146. 'dbname' => 'test'
  147. ));
  148. ]]>
  149. </programlisting>
  150. </example>
  151. <para>
  152. Если вы создали собственный класс, расширяющий
  153. Zend_Db_Adapter_Abstract, но не дали ему имя, начинающееся с
  154. префикса "Zend_Db_Adapter", то можете использовать метод
  155. <code>factory()</code> для загрузки своего адаптера, указав
  156. ведущую часть имени класса адаптера с помощью ключа
  157. 'adapterNamespace' в массиве параметров.
  158. </para>
  159. <example id="zend.db.adapter.connecting.factory.example2">
  160. <title>Использование метода-фабрики для пользовательского класса адаптера</title>
  161. <programlisting language="php"><![CDATA[
  162. // Нам не нужно загружать файл с классом адаптера,
  163. // поскольку он будет загружен через метод-фабрику Zend_Db
  164. // Автоматически загружает класс MyProject_Db_Adapter_Pdo_Mysql
  165. // и создает его экземпляр.
  166. $db = Zend_Db::factory('Pdo_Mysql', array(
  167. 'host' => '127.0.0.1',
  168. 'username' => 'webuser',
  169. 'password' => 'xxxxxxxx',
  170. 'dbname' => 'test',
  171. 'adapterNamespace' => 'MyProject_Db_Adapter'
  172. ));
  173. ]]>
  174. </programlisting>
  175. </example>
  176. </sect3>
  177. <sect3 id="zend.db.adapter.connecting.factory-config">
  178. <title>Использование Zend_Config с фабрикой Zend_Db</title>
  179. <para>
  180. Опционально вы можете заменить оба аргумента метода
  181. <code>factory()</code> объектом типа
  182. <link linkend="zend.config">Zend_Config</link>.
  183. </para>
  184. <para>
  185. Если первым аргументом является объект конфигурации, то
  186. ожидается, что он имеет свойство с именем <code>adapter</code>,
  187. содержащее строку с базовой частью имени класса адаптера.
  188. Опционально объект может содержать свойство с именем
  189. <code>params</code> и "подсвойствами", соответствующими
  190. параметрам адаптера.
  191. </para>
  192. <example id="zend.db.adapter.connecting.factory.example1">
  193. <title>Использование метода-фабрики адаптеров с объектом Zend_Config</title>
  194. <para>
  195. В примере ниже объект Zend_Config создан из массива. Вы
  196. можете также загружать данные из внешнего файла с помощью
  197. <link linkend="zend.config.adapters.ini">Zend_Config_Ini</link>
  198. или <link linkend="zend.config.adapters.xml">Zend_Config_Xml</link>.
  199. </para>
  200. <programlisting language="php"><![CDATA[
  201. $config = new Zend_Config(
  202. array(
  203. 'database' => array(
  204. 'adapter' => 'Mysqli',
  205. 'params' => array(
  206. 'dbname' => 'test',
  207. 'username' => 'webuser',
  208. 'password' => 'secret',
  209. )
  210. )
  211. )
  212. );
  213. $db = Zend_Db::factory($config->database);
  214. ]]>
  215. </programlisting>
  216. </example>
  217. <para>
  218. Второй аргумент метода <code>factory()</code> может быть
  219. ассоциативным массивом, содержащим элементы, которые
  220. соответствуют параметрам адаптера. Этот аргумент является
  221. опциональным. Если первым аргументом является объект типа
  222. Zend_Config, то предполагается, что он содержит все необходимые
  223. параметры, и второй аргумент игнорируется.
  224. </para>
  225. </sect3>
  226. <sect3 id="zend.db.adapter.connecting.parameters">
  227. <title>Параметры адаптера</title>
  228. <para>
  229. Список ниже описывает общие параметры, которые распознаются
  230. классами адаптеров Zend_Db.
  231. </para>
  232. <itemizedlist>
  233. <listitem>
  234. <para>
  235. <emphasis>host</emphasis>:
  236. строка, содержащая имя хоста или IP сервера БД. Если
  237. база данных размещается на том же хосте, что и
  238. приложение PHP, то вы можете использовать 'localhost'
  239. или '127.0.0.1'.
  240. </para>
  241. </listitem>
  242. <listitem>
  243. <para>
  244. <emphasis>username</emphasis>:
  245. идентификатор учетной записи для аутентификации подключения к серверу СУРБД.
  246. </para>
  247. </listitem>
  248. <listitem>
  249. <para>
  250. <emphasis>password</emphasis>:
  251. пароль учетной записи для аутентификации подключения к
  252. серверу СУРБД.
  253. </para>
  254. </listitem>
  255. <listitem>
  256. <para>
  257. <emphasis>dbname</emphasis>:
  258. имя экземпляра БД на сервере СУРБД.
  259. </para>
  260. </listitem>
  261. <listitem>
  262. <para>
  263. <emphasis>port</emphasis>:
  264. некоторые сервера СУРБД поддерживают сетевые соединения
  265. через указанный администратором порт. Данный параметр
  266. дает возможность задать порт, с которым приложение PHP
  267. будет устанавливать соединение, он должен
  268. соответствовать порту, установленному в сервере СУРБД.
  269. </para>
  270. </listitem>
  271. <listitem>
  272. <para>
  273. <emphasis>options</emphasis>:
  274. этот параметр является ассоциативным массивом опций,
  275. общих для всех классов Zend_Db_Adapter.
  276. </para>
  277. </listitem>
  278. <listitem>
  279. <para>
  280. <emphasis>driver_options</emphasis>:
  281. этот параметр является ассоциативным массивом
  282. дополнительных опций, специфических для данного
  283. расширения. Одним из типичных случаев использования
  284. этого параметра является установка атрибутов для
  285. драйвера PDO.
  286. </para>
  287. </listitem>
  288. <listitem>
  289. <para>
  290. <emphasis>adapterNamespace</emphasis>:
  291. имя начальной части имени класса для адаптера вместо
  292. 'Zend_Db_Adapter'. Используйте его, если нужно
  293. использовать метод <code>factory()</code> для загрузки
  294. "неZend'овского" класса адаптера БД.
  295. </para>
  296. </listitem>
  297. </itemizedlist>
  298. <example id="zend.db.adapter.connecting.parameters.example1">
  299. <title>Передача фабрике опции перевода регистра (case-folding)</title>
  300. <para>
  301. Вы можете установить эту опцию посредством константы
  302. <code>Zend_Db::CASE_FOLDING</code>. Она соответствует
  303. атрибуту <code>ATTR_CASE</code> в драйверах PDO и IBM DB2, и
  304. переводит строковые ключи в результатах запроса в требуемый
  305. регистр. Эта опция принимает значения
  306. <code>Zend_Db::CASE_NATURAL</code> (значение по умолчанию),
  307. <code>Zend_Db::CASE_UPPER</code> и
  308. <code>Zend_Db::CASE_LOWER</code>.
  309. </para>
  310. <programlisting language="php"><![CDATA[
  311. $options = array(
  312. Zend_Db::CASE_FOLDING => Zend_Db::CASE_UPPER
  313. );
  314. $params = array(
  315. 'host' => '127.0.0.1',
  316. 'username' => 'webuser',
  317. 'password' => 'xxxxxxxx',
  318. 'dbname' => 'test',
  319. 'options' => $options
  320. );
  321. $db = Zend_Db::factory('Db2', $params);
  322. ]]>
  323. </programlisting>
  324. </example>
  325. <example id="zend.db.adapter.connecting.parameters.example2">
  326. <title>Передача фабрике опции автоматического заключения в кавычки</title>
  327. <para>
  328. Вы можете задавать эту опцию через константу
  329. <code>Zend_Db::AUTO_QUOTE_IDENTIFIERS</code>. Если ее
  330. значение установлено в <constant>TRUE</constant> (по умолчанию), то
  331. идентификаторы, такие, как имена таблиц, имена столбцов и
  332. даже псевдонимы, разграничиваются во всем генерируемом
  333. объектом адаптера синтаксисе SQL. Это делает возможным
  334. использование идентификаторов, содержащих ключевые слова SQL
  335. и специальные символы. Если его значение равно
  336. <constant>FALSE</constant>, то автоматическое заключение в кавычки
  337. не производится. Если требуется заключение идентификаторов в
  338. кавычки, то оно должно производиться самостоятельно с
  339. использованием метода <code>quoteIdentifier()</code>.
  340. </para>
  341. <programlisting language="php"><![CDATA[
  342. $options = array(
  343. Zend_Db::AUTO_QUOTE_IDENTIFIERS => false
  344. );
  345. $params = array(
  346. 'host' => '127.0.0.1',
  347. 'username' => 'webuser',
  348. 'password' => 'xxxxxxxx',
  349. 'dbname' => 'test',
  350. 'options' => $options
  351. );
  352. $db = Zend_Db::factory('Pdo_Mysql', $params);
  353. ]]>
  354. </programlisting>
  355. </example>
  356. <example id="zend.db.adapter.connecting.parameters.example3">
  357. <title>Передача фабрике опций драйвера PDO</title>
  358. <programlisting language="php"><![CDATA[
  359. $pdoParams = array(
  360. PDO::MYSQL_ATTR_USE_BUFFERED_QUERY => true
  361. );
  362. $params = array(
  363. 'host' => '127.0.0.1',
  364. 'username' => 'webuser',
  365. 'password' => 'xxxxxxxx',
  366. 'dbname' => 'test',
  367. 'driver_options' => $pdoParams
  368. );
  369. $db = Zend_Db::factory('Pdo_Mysql', $params);
  370. echo $db->getConnection()
  371. ->getAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY);
  372. ]]>
  373. </programlisting>
  374. </example>
  375. </sect3>
  376. <sect3 id="zend.db.adapter.connecting.getconnection">
  377. <title>Управление отложенными соединениями</title>
  378. <para>
  379. Создание экземпляра класса адаптера не приведет к немедленному
  380. соединению с сервером СУРБД. Адаптер сохраняет параметры
  381. соединения и производит подключение, когда нужно произвести
  382. первый запрос к БД. Это значит, что само по себе создание
  383. объекта адаптера производится быстро и занимает мало ресурсов.
  384. Вы можете создавать экземпляр адаптера даже в том случае, если
  385. не уверены в том, что текущий запрос к вашему приложению
  386. требует каких-либо действий с БД.
  387. </para>
  388. <para>
  389. Если нужно принудительно создать соединение с СУРБД, то
  390. используйте метод <code>getConnection()</code>. Этот метод
  391. возвращает объект соединения в представлении соответствующего
  392. расширения PHP для баз данных. Например, если вы используете
  393. какой-либо класс адаптера для драйверов PDO, то
  394. <code>getConnection()</code> возвращает объект PDO после того,
  395. как он будет инициирован им в качестве "живого" соединения с
  396. определенной БД.
  397. </para>
  398. <para>
  399. Принудительное создание соединения может быть полезным, когда вы
  400. хотите отлавливать все исключения, которые бросаются из-за
  401. неправильных параметров доступа или других ошибок соединения с
  402. сервером СУРБД. Эти исключения не бросаются до тех пор, пока не
  403. создается соединение, поэтому можно упростить
  404. код приложения, обрабатывая исключения в одном
  405. месте вместо того, чтобы делать это каждый раз, когда
  406. производится первый запрос к БД.
  407. </para>
  408. <example id="zend.db.adapter.connecting.getconnection.example">
  409. <title>Обработка исключений при соединении</title>
  410. <programlisting language="php"><![CDATA[
  411. try {
  412. $db = Zend_Db::factory('Pdo_Mysql', $parameters);
  413. $db->getConnection();
  414. } catch (Zend_Db_Adapter_Exception $e) {
  415. // возможно, неправильные параметры соединения или СУРБД не запущена
  416. } catch (Zend_Exception $e) {
  417. // возможно, попытка загрузки требуемого класса адаптера потерпела неудачу
  418. }
  419. ]]>
  420. </programlisting>
  421. </example>
  422. </sect3>
  423. </sect2>
  424. <sect2 id="zend.db.adapter.example-database">
  425. <title>Пример базы данных</title>
  426. <para>
  427. В документации к классам Zend_Db мы использовали набор простых
  428. таблиц для того, чтобы проиллюстрировать использование классов и их
  429. методов. Эти таблицы должны были хранить информацию для отслеживания
  430. ошибок в проекте разработки ПО. База данных содержит четыре таблицы:
  431. </para>
  432. <itemizedlist>
  433. <listitem>
  434. <para>
  435. <emphasis>accounts</emphasis> (учетные записи)
  436. хранит информацию о всех пользователях системы отслеживания
  437. ошибок.
  438. </para>
  439. </listitem>
  440. <listitem>
  441. <para>
  442. <emphasis>products</emphasis> (продукты)
  443. хранит информацию обо всех программных продуктах, для
  444. которых могут отслеживаться ошибки.
  445. </para>
  446. </listitem>
  447. <listitem>
  448. <para>
  449. <emphasis>bugs</emphasis> (ошибки) хранит
  450. информацию об ошибках, включая текущее состояние ошибки,
  451. лицо, сообщившее об ошибке, лицо, которому назначено
  452. устранение ошибки и лицо, которому назначена проверка
  453. устранения ошибки.
  454. </para>
  455. </listitem>
  456. <listitem>
  457. <para>
  458. <emphasis>bugs_products</emphasis> хранит
  459. связи между ошибками и продуктами. Она реализует связь
  460. "многие-ко-многим", потому что одна ошибка может относиться
  461. к нескольким продуктам, и один продукт может иметь множество
  462. ошибок.
  463. </para>
  464. </listitem>
  465. </itemizedlist>
  466. <para>
  467. Следующий псевдокод для определения данных SQL описывает таблицы в
  468. этой базе данных. Это таблицы интенсивно используются в unit-тестах
  469. для Zend_Db.
  470. </para>
  471. <programlisting language="sql"><![CDATA[
  472. CREATE TABLE accounts (
  473. account_name VARCHAR(100) NOT NULL PRIMARY KEY
  474. );
  475. CREATE TABLE products (
  476. product_id INTEGER NOT NULL PRIMARY KEY,
  477. product_name VARCHAR(100)
  478. );
  479. CREATE TABLE bugs (
  480. bug_id INTEGER NOT NULL PRIMARY KEY,
  481. bug_description VARCHAR(100),
  482. bug_status VARCHAR(20),
  483. reported_by VARCHAR(100) REFERENCES accounts(account_name),
  484. assigned_to VARCHAR(100) REFERENCES accounts(account_name),
  485. verified_by VARCHAR(100) REFERENCES accounts(account_name)
  486. );
  487. CREATE TABLE bugs_products (
  488. bug_id INTEGER NOT NULL REFERENCES bugs,
  489. product_id INTEGER NOT NULL REFERENCES products,
  490. PRIMARY KEY (bug_id, product_id)
  491. );
  492. ]]>
  493. </programlisting>
  494. <para>
  495. Также обратите внимание, что таблица <code>bugs</code> содержит
  496. несколько внешних ключей, ссылающихся на таблицу
  497. <code>accounts</code>. Для одной ошибки эти внешние ключи могут
  498. ссылаться на разные строки в таблице <code>accounts</code>.
  499. </para>
  500. <para>
  501. Диаграмма ниже иллюстрирует физическую модель данных для этой базы
  502. данных.
  503. </para>
  504. <para>
  505. <inlinegraphic width="387" scale="100" align="center" valign="middle"
  506. fileref="figures/zend.db.adapter.example-database.png" format="PNG" />
  507. </para>
  508. </sect2>
  509. <sect2 id="zend.db.adapter.select">
  510. <title>Чтение результатов запроса</title>
  511. <para>
  512. Этот раздел описывает методы класса адаптера, с помощью которых вы
  513. можете производить запросы SELECT и извлекать их результаты.
  514. </para>
  515. <sect3 id="zend.db.adapter.select.fetchall">
  516. <title>Извлечение полного набора результатов</title>
  517. <para>
  518. Вы можете запустить запрос SELECT и извлечь его результаты за
  519. один шаг, используя метод <code>fetchAll()</code>.
  520. </para>
  521. <para>
  522. Первым аргументом этого метода должна быть строка, содержащая
  523. оператор SELECT. Также первым аргументом может быть объект
  524. класса <link linkend="zend.db.select">Zend_Db_Select</link>.
  525. Адаптер автоматически преобразует этот объект в строковое
  526. представление оператора SELECT.
  527. </para>
  528. <para>
  529. Вторым аргументом <code>fetchAll()</code> должен быть массив
  530. значений для подстановки вместо меток заполнения (placeholders)
  531. в операторе SQL.
  532. </para>
  533. <example id="zend.db.adapter.select.fetchall.example">
  534. <title>Использование fetchAll()</title>
  535. <programlisting language="php"><![CDATA[
  536. $sql = 'SELECT * FROM bugs WHERE bug_id = ?';
  537. $result = $db->fetchAll($sql, 2);
  538. ]]>
  539. </programlisting>
  540. </example>
  541. </sect3>
  542. <sect3 id="zend.db.adapter.select.fetch-mode">
  543. <title>Изменение режима извлечения</title>
  544. <para>
  545. По умолчанию <code>fetchAll()</code> возвращает массив строк,
  546. каждая из которых представлена ассоциативным
  547. массивом. Ключами ассоциативных массивов являются имена столбцов
  548. или псевдонимы столбцов, определенные в данном запросе на
  549. выборку.
  550. </para>
  551. <para>
  552. Вы можете задать другой стиль извлечения результатов, используя
  553. метод <code>setFetchMode()</code>. Поддерживаемые режимы
  554. идентифицируются константами:
  555. </para>
  556. <itemizedlist>
  557. <listitem>
  558. <para>
  559. <emphasis>Zend_Db::FETCH_ASSOC</emphasis>:
  560. возвращает данные в массиве ассоциативных массивов.
  561. Ключами массива являются имена столбцов в виде строк.
  562. Это режим извлечения, используемый по умолчанию в
  563. классах Zend_Db_Adapter.
  564. </para>
  565. <para>
  566. Обратите внимание, что если ваш список выборки содержит
  567. столбцы с одинаковыми именами, например, если они из
  568. разных таблиц в JOIN-е, то в ассоциативном массиве может
  569. быть только одна запись для этого имени. Если вы
  570. используете режим FETCH_ASSOC, то должны задавать
  571. псевдонимы столбцов в своем запросе SELECT для того,
  572. чтобы для всех столбцов были свои уникальные ключи.
  573. </para>
  574. <para>
  575. По умолчанию эти строки возвращаются так же, как если бы
  576. они были возвращены драйвером БД. Как правило, это
  577. синтаксис столбцов для данного сервера СУРБД. Вы можете
  578. задать регистр для этих строк, используя опцию.
  579. <code>Zend_Db::CASE_FOLDING</code>. Задавайте его во
  580. время инстанцирования адаптера. См.
  581. <xref linkend="zend.db.adapter.connecting.parameters.example1" />.
  582. </para>
  583. </listitem>
  584. <listitem>
  585. <para>
  586. <emphasis>Zend_Db::FETCH_NUM</emphasis>:
  587. возвращает данные в массиве массивов. Массив
  588. индексируется целочисленными значениями в соответствии с
  589. позицией данного поля в списке выборки запроса.
  590. </para>
  591. </listitem>
  592. <listitem>
  593. <para>
  594. <emphasis>Zend_Db::FETCH_BOTH</emphasis>:
  595. возвращает данные в массиве массивов. Ключами массива
  596. являются как строки, так и целочисленные значения. Число
  597. элементов в массиве получается в два раза больше, чем
  598. если бы использовались FETCH_ASSOC или FETCH_NUM.
  599. </para>
  600. </listitem>
  601. <listitem>
  602. <para>
  603. <emphasis>Zend_Db::FETCH_COLUMN</emphasis>:
  604. возвращает данные в массиве значений. Значение в каждом
  605. массиве является значением, возвращенным из одного
  606. столбца результата выборки. По умолчанию это первый
  607. столбец, индексированный нулем.
  608. </para>
  609. </listitem>
  610. <listitem>
  611. <para>
  612. <emphasis>Zend_Db::FETCH_OBJ</emphasis>:
  613. возвращает данные в массиве объектов. По умолчанию
  614. используется встроенный в PHP класс stdClass. Столбцы
  615. результата выборки доступны в качестве открытых свойств
  616. этого объекта.
  617. </para>
  618. </listitem>
  619. </itemizedlist>
  620. <example id="zend.db.adapter.select.fetch-mode.example">
  621. <title>Использование setFetchMode()</title>
  622. <programlisting language="php"><![CDATA[
  623. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  624. $result = $db->fetchAll('SELECT * FROM bugs WHERE bug_id = ?', 2);
  625. // $result является массивом объектов
  626. echo $result[0]->bug_description;
  627. ]]>
  628. </programlisting>
  629. </example>
  630. </sect3>
  631. <sect3 id="zend.db.adapter.select.fetchassoc">
  632. <title>Извлечение результатов выборки в виде ассоциативного массива</title>
  633. <para>
  634. Метод <code>fetchAssoc()</code> возвращает данные в массиве
  635. ассоциативных массивов безотносительно того, какое значение вы
  636. установили для режима извлечения.
  637. </para>
  638. <example id="zend.db.adapter.select.fetchassoc.example">
  639. <title>Использование fetchAssoc()</title>
  640. <programlisting language="php"><![CDATA[
  641. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  642. $result = $db->fetchAssoc('SELECT * FROM bugs WHERE bug_id = ?', 2);
  643. // $result является массивом ассоциативных массивов, независимо
  644. // от установленного режима извлечения
  645. echo $result[0]['bug_description'];
  646. ]]>
  647. </programlisting>
  648. </example>
  649. </sect3>
  650. <sect3 id="zend.db.adapter.select.fetchcol">
  651. <title>Извлечение единственного столбца из результатов выборки</title>
  652. <para>
  653. Метод <code>fetchCol()</code> возвращает данные в массиве
  654. значений безотносительно того, какое значение вы
  655. установили для режима извлечения. Он возвращает только первый
  656. столбец из возвращенных запросом. Все остальные столбцы,
  657. возвращенные запросом, не учитываются. Если вам нужно извлечь
  658. столбец, отличный от первого, то см.
  659. <xref linkend="zend.db.statement.fetching.fetchcolumn" />.
  660. </para>
  661. <example id="zend.db.adapter.select.fetchcol.example">
  662. <title>Использование fetchCol()</title>
  663. <programlisting language="php"><![CDATA[
  664. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  665. $result = $db->fetchCol(
  666. 'SELECT bug_description, bug_id FROM bugs WHERE bug_id = ?', 2);
  667. // содержит bug_description; bug_id не возвращается
  668. echo $result[0];
  669. ]]>
  670. </programlisting>
  671. </example>
  672. </sect3>
  673. <sect3 id="zend.db.adapter.select.fetchpairs">
  674. <title>Извлечение пар ключ-значение из результатов выборки</title>
  675. <para>
  676. Метод <code>fetchPairs()</code> возвращает данные в массиве пар
  677. ключ-значение,
  678. Ключ ассоциативного массива берется из первого столбца,
  679. возвращенного запросом SELECT. Значение берется из второго
  680. столбца, возвращенного запросом SELECT. Все остальные столбцы,
  681. возвращенные запросом, не учитываются.
  682. </para>
  683. <para>
  684. Вы должны строить запрос SELECT так, чтобы первый из
  685. возвращенных столбцов имел уникальные значения. Если в нем
  686. имеются повторяющиеся значения, то записи в ассоциативном
  687. массиве будут перезаписываться.
  688. </para>
  689. <example id="zend.db.adapter.select.fetchpairs.example">
  690. <title>Использование fetchPairs()</title>
  691. <programlisting language="php"><![CDATA[
  692. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  693. $result = $db->fetchPairs('SELECT bug_id, bug_status FROM bugs');
  694. echo $result[2];
  695. ]]>
  696. </programlisting>
  697. </example>
  698. </sect3>
  699. <sect3 id="zend.db.adapter.select.fetchrow">
  700. <title>Извлечение единственной строки из результатов выборки</title>
  701. <para>
  702. Метод <code>fetchRow()</code> возвращает данные с использованием
  703. текущего режима извлечения, но возвращает только первую строку
  704. из результатов выборки.
  705. </para>
  706. <example id="zend.db.adapter.select.fetchrow.example">
  707. <title>Использование fetchRow()</title>
  708. <programlisting language="php"><![CDATA[
  709. $db->setFetchMode(Zend_Db::FETCH_OBJ);
  710. $result = $db->fetchRow('SELECT * FROM bugs WHERE bug_id = 2');
  711. // обратите внимание, что $result - единственный объект, а не массив объектов
  712. echo $result->bug_description;
  713. ]]>
  714. </programlisting>
  715. </example>
  716. </sect3>
  717. <sect3 id="zend.db.adapter.select.fetchone">
  718. <title>Извлечение единственного скалярного значения из результатов выборки</title>
  719. <para>
  720. Метод <code>fetchOne()</code> является как бы комбинацией
  721. методов <code>fetchRow()</code> и <code>fetchCol()</code> - он
  722. возвращает значение первого столбца в первой строке из
  723. результатов выборки. Таким образом, он возвращает одно скалярное
  724. значение, а не массив или объект.
  725. </para>
  726. <example id="zend.db.adapter.select.fetchone.example">
  727. <title>Использование fetchOne()</title>
  728. <programlisting language="php"><![CDATA[
  729. $result = $db->fetchOne('SELECT bug_status FROM bugs WHERE bug_id = 2');
  730. // это единственное строковое значение
  731. echo $result;
  732. ]]>
  733. </programlisting>
  734. </example>
  735. </sect3>
  736. </sect2>
  737. <sect2 id="zend.db.adapter.write">
  738. <title>Изменение данных в БД</title>
  739. <para>
  740. Вы можете использовать класс адаптера для добавления новых данных
  741. или изменения существующих в своей базе данных. В данном разделе
  742. описываются методы для произведения этих операций.
  743. </para>
  744. <sect3 id="zend.db.adapter.write.insert">
  745. <title>Добавление данных</title>
  746. <para>
  747. Вы можете добавлять новые строки в таблицы в своей базе данных,
  748. используя метод <code>insert()</code>. Первым аргументом этого
  749. метода является строка с именем таблицы, а вторым аргументом -
  750. ассоциативный массив с именами столбцов и соответствующими им
  751. значениями.
  752. </para>
  753. <example id="zend.db.adapter.write.insert.example">
  754. <title>Добавление в таблицу</title>
  755. <programlisting language="php"><![CDATA[
  756. $data = array(
  757. 'created_on' => '2007-03-22',
  758. 'bug_description' => 'Something wrong',
  759. 'bug_status' => 'NEW'
  760. );
  761. $db->insert('bugs', $data);
  762. ]]>
  763. </programlisting>
  764. </example>
  765. <para>
  766. Те столбцы, которые не были включены в массив данных, не
  767. передаются базе данных. Таким образом, они следуют тем же
  768. правилам, что и SQL-оператор INSERT: если столбец имеет
  769. предложение DEFAULT, то он принимает это значение в созданной
  770. строке, иначе остается в состоянии NULL.
  771. </para>
  772. <para>
  773. По умолчанию значения в вашем массиве данных добавляются с
  774. использованием параметров. Это сокращает некоторые риски
  775. безопасности. Вам не нужно будет применять к значениям в массиве
  776. данных такие действия, как взятие в кавычки или экранирование.
  777. </para>
  778. <para>
  779. Иногда бывает необходимо, чтобы часть значений в массиве данных
  780. трактовалась как SQL-выражения, в этом случае они не должны
  781. заключаться в кавычки. По умолчанию все данные, переданные в
  782. виде строк, трактуются как строковые литералы. Для того, чтобы
  783. указать, что данное значение является SQL-выражением (а
  784. значит, не должно заключаться в кавычки), передавайте его в
  785. массиве данных в виде объекта типа Zend_Db_Expr вместо простой
  786. строки.
  787. </para>
  788. <example id="zend.db.adapter.write.insert.example2">
  789. <title>Добавление выражений в таблицу</title>
  790. <programlisting language="php"><![CDATA[
  791. $data = array(
  792. 'created_on' => new Zend_Db_Expr('CURDATE()'),
  793. 'bug_description' => 'Something wrong',
  794. 'bug_status' => 'NEW'
  795. );
  796. $db->insert('bugs', $data);
  797. ]]>
  798. </programlisting>
  799. </example>
  800. </sect3>
  801. <sect3 id="zend.db.adapter.write.lastinsertid">
  802. <title>Получение сгенерированного значения</title>
  803. <para>
  804. Некоторые СУРБД поддерживают автоинкремент первичных ключей.
  805. Таблица, описанная определенным образом, автоматически
  806. генерирует значение первичного ключа во время добавления новой
  807. строки.
  808. Возвращаемое методом <code>insert()</code> значение
  809. <emphasis>не</emphasis> является последним добавленным
  810. идентификатором, потому что таблица может не иметь
  811. автоинкрементных столбцов. Вместо этого возвращаемое
  812. значение является количеством затронутых строк (обычно 1).
  813. </para>
  814. <para>
  815. Если ваша таблица определена с автоинкрементным первичным
  816. ключом, то вы можете вызывать метод <code>lastInsertId()</code>
  817. после добавления. Этот метод возвращает последнее значение,
  818. сгенерированное в области видимости текущего соединения с БД.
  819. </para>
  820. <example id="zend.db.adapter.write.lastinsertid.example-1">
  821. <title>Использование lastInsertId() для автоинкрементного ключа</title>
  822. <programlisting language="php"><![CDATA[
  823. $db->insert('bugs', $data);
  824. // возвращает последнее значение, сгенерированное автоинкрементным столбцом
  825. $id = $db->lastInsertId();
  826. ]]>
  827. </programlisting>
  828. </example>
  829. <para>
  830. Некоторые СУРБД поддерживают объекты последовательностей
  831. (sequence object), которые генерируют уникальные значения для
  832. использования в качестве значений первичных ключей. Для
  833. поддержки последовательностей <code>lastInsertId()</code>
  834. принимает два необязательных строковых аргумента. Эти аргументы
  835. служат для передачи имен таблицы и столбца, при этом
  836. предполагается, что вы следуете соглашению, по которому имя
  837. последовательности состоит из имен таблицы и столбца, для
  838. которых эта последовательность генерирует значения, и суффикса
  839. "_seq". Это соглашение основано на используемом системой
  840. PostgreSQL при именовании последовательностей для столбцов
  841. SERIAL. Например, таблица "bugs" с первичным ключом "bug_id"
  842. должна использовать последовательность с именем
  843. "bugs_bug_id_seq".
  844. </para>
  845. <example id="zend.db.adapter.write.lastinsertid.example-2">
  846. <title>Использование lastInsertId() для последовательности</title>
  847. <programlisting language="php"><![CDATA[
  848. $db->insert('bugs', $data);
  849. // возвращает последнее значение, сгенерированное
  850. // последовательностью 'bugs_bug_id_seq'
  851. $id = $db->lastInsertId('bugs', 'bug_id');
  852. // альтернативно, возвращает последнее значение, сгенерированное
  853. // последовательностью 'bugs_seq'.
  854. $id = $db->lastInsertId('bugs');
  855. ]]>
  856. </programlisting>
  857. </example>
  858. <para>
  859. Если имя вашего объекта последовательности не следует этому
  860. соглашению по именованию, то используйте метод
  861. <code>lastSequenceId()</code>. Этот метод принимает один
  862. строковой аргумент, через который передается точное имя
  863. последовательности
  864. </para>
  865. <example id="zend.db.adapter.write.lastinsertid.example-3">
  866. <title>Использование lastSequenceId()</title>
  867. <programlisting language="php"><![CDATA[
  868. $db->insert('bugs', $data);
  869. // возвращает последнее значение, сгенерированное
  870. // последовательностью 'bugs_id_gen'.
  871. $id = $db->lastSequenceId('bugs_id_gen');
  872. ]]>
  873. </programlisting>
  874. </example>
  875. <para>
  876. Для тех СУРБД, которые не поддерживают последовательности,
  877. включая MySQL, Microsoft SQL Server и SQLite, аргументы метода
  878. lastInsertId() игнорируются, и возвращается самое последнее
  879. значение, сгенерированное для любой таблицы через оператор
  880. INSERT в течение данного соединения. Для этих типов СУРБД метод
  881. lastSequenceId() всегда будет возвращать <constant>NULL</constant>.
  882. </para>
  883. <note>
  884. <title>Почему не используется "SELECT MAX(id) FROM table"?</title>
  885. <para>
  886. Иногда этот запрос возвращает последнее значение первичного
  887. ключа, добавленное в таблицу. Однако этот способ небезопасен
  888. в условиях, когда несколько клиентов добавляют записи в базу
  889. данных. Может случиться (и должно происходить в конечном
  890. итоге) так, что другой клиент добавляет другую строку в
  891. короткий промежуток времени между добавлением строки,
  892. производимым вашим приложением-клиентом БД, и вашим запросом
  893. для получения значения MAX(id). Таким образом, это
  894. возвращаемое значение не будет соответствовать добавленной
  895. вами строке, вместо этого оно будет соответствовать строке,
  896. добавленной другим клиентом. Нет способа определить,
  897. когда это происходит.
  898. </para>
  899. <para>
  900. Использование высокого уровня изоляции транзакций, такого,
  901. как "repeatable read", может уменьшить этот риск, но
  902. некоторые СУРБД не поддерживают требуемую для этого изоляцию
  903. транзакций, либо намеренно используется более низкий уровень
  904. изоляции транзакций в приложении.
  905. </para>
  906. <para>
  907. Использование выражения наподобие "MAX(id)+1" для генерации
  908. нового значения первичного ключа тоже небезопасно, так как
  909. два клиента могут сделать этот запрос одновременно, и оба
  910. будут использовать одно и то же полученное значение для
  911. своей последующей операции INSERT.
  912. </para>
  913. <para>
  914. Все СУРБД предоставляют механизмы для генерации уникальных
  915. значений и возвращения последних сгенерированных значений.
  916. Эти механизмы работают вне области видимости транзакций,
  917. поэтому нет вероятности того, что оба клиента сгенерируют
  918. одно и то же значение, или что значение, сгенерированное
  919. другим клиентом, будет возвращено вашему клиенту как
  920. последнее сгенерированное им в его соединении.
  921. </para>
  922. </note>
  923. </sect3>
  924. <sect3 id="zend.db.adapter.write.update">
  925. <title>Обновление данных</title>
  926. <para>
  927. Вы можете обновлять строки в таблице БД, используя метод
  928. <code>update()</code> адаптера. Этот метод принимает три
  929. аргумента: первый является имением таблицы, второй -
  930. ассоциативным массивом столбцов, которые требуется изменить, и
  931. значений, которые требуется присвоить этим столбцам.
  932. </para>
  933. <para>
  934. Значения в массиве данных интерпретируются как строковые
  935. константы. Информацию об использовании выражений SQL в массиве
  936. данных см. в разделе
  937. <xref linkend="zend.db.adapter.write.insert" />.
  938. </para>
  939. <para>
  940. Третий аргумент является строкой, содержащей выражение SQL,
  941. которое используется в качестве условия, при выполнении которого
  942. строка должна изменяться. Значения и идентификаторы в этом
  943. аргументе не заключаются в кавычки и не экранируются. Вы
  944. ответственны за то, чтобы все динамическое содержимое было
  945. безопасным образом включено в эту строку. Информацию о методах,
  946. которые помогут вам в этом, см. в разделе
  947. <xref linkend="zend.db.adapter.quoting" />.
  948. </para>
  949. <para>
  950. Возвращаемое значение является числом строк, затронутых в
  951. операции обновления.
  952. </para>
  953. <example id="zend.db.adapter.write.update.example">
  954. <title>Обновление строк</title>
  955. <programlisting language="php"><![CDATA[
  956. $data = array(
  957. 'updated_on' => '2007-03-23',
  958. 'bug_status' => 'FIXED'
  959. );
  960. $n = $db->update('bugs', $data, 'bug_id = 2');
  961. ]]>
  962. </programlisting>
  963. </example>
  964. <para>
  965. Если вы опустите третий аргумент, то все строки в таблице БД
  966. будут обновлены со значениями, указанными в массиве данных.
  967. </para>
  968. <para>
  969. Если вы передадите массив строк в качестве третьего аргумента,
  970. то эти строки будут объединены как термы выражения, разделенные
  971. операторами <code>AND</code>.
  972. </para>
  973. <example id="zend.db.adapter.write.update.example-array">
  974. <title>Обновление строк с использованием массива выражений</title>
  975. <programlisting language="php"><![CDATA[
  976. $data = array(
  977. 'updated_on' => '2007-03-23',
  978. 'bug_status' => 'FIXED'
  979. );
  980. $where[] = "reported_by = 'goofy'";
  981. $where[] = "bug_status = 'OPEN'";
  982. $n = $db->update('bugs', $data, $where);
  983. // Результирующий SQL:
  984. // UPDATE "bugs" SET "update_on" = '2007-03-23', "bug_status" = 'FIXED'
  985. // WHERE ("reported_by" = 'goofy') AND ("bug_status" = 'OPEN')
  986. ]]>
  987. </programlisting>
  988. </example>
  989. </sect3>
  990. <sect3 id="zend.db.adapter.write.delete">
  991. <title>Удаление данных</title>
  992. <para>
  993. Вы можете удалять строки из таблицы БД, используя метод
  994. <code>delete()</code>. Этот метод принимает два аргумента,
  995. первый из них является строкой с именем таблицы.
  996. </para>
  997. <para>
  998. Второй аргумент является строкой, содержащей выражение SQL,
  999. который используется в качестве условия, при выполнении которого
  1000. строка удаляется. Значения и идентификаторы в этом аргументе не
  1001. заключаются в кавычки и не экранируются. Вы ответственны за то,
  1002. чтобы весь динамический контент был безопасным образом включен в
  1003. эту строку. Информацию о методах, которые помогут вам в этом,
  1004. см. в разделе <xref linkend="zend.db.adapter.quoting" />.
  1005. </para>
  1006. <para>
  1007. Возвращаемое значение является числом строк, задействованных в
  1008. операции удаления.
  1009. </para>
  1010. <example id="zend.db.adapter.write.delete.example">
  1011. <title>Удаление строк</title>
  1012. <programlisting language="php"><![CDATA[
  1013. $n = $db->delete('bugs', 'bug_id = 3');
  1014. ]]>
  1015. </programlisting>
  1016. </example>
  1017. <para>
  1018. Если вы опустите второй аргумент, то в результате
  1019. все строки в таблице БД будут удалены.
  1020. </para>
  1021. <para>
  1022. Если вы передадите массив строк в качестве второго аргумента, то
  1023. эти строки будут объединены как термы выражения, разделенные
  1024. операторами <code>AND</code>.
  1025. </para>
  1026. </sect3>
  1027. </sect2>
  1028. <sect2 id="zend.db.adapter.quoting">
  1029. <title>Заключение в кавычки значений и идентификаторов</title>
  1030. <para>
  1031. При построении запросов SQL часто требуется включить значения
  1032. переменных PHP в выражения SQL. Это несет в себе дополнительный
  1033. риск, потому что если значение в строке PHP содержит определенные
  1034. символы, такие, как символы кавычек, то в результате может
  1035. получиться недопустимый код SQL. Например, обратите внимание на
  1036. несоответствие кавычек в следующем запросе:
  1037. <programlisting language="php"><![CDATA[
  1038. $name = "O'Reilly";
  1039. $sql = "SELECT * FROM bugs WHERE reported_by = '$name'";
  1040. echo $sql;
  1041. // SELECT * FROM bugs WHERE reported_by = 'O'Reilly'
  1042. ]]>
  1043. </programlisting>
  1044. </para>
  1045. <para>
  1046. Еще серьезнее риск того, что такие ошибки в коде могут быть
  1047. целенаправленно использованы тем, кто пытается получить управление
  1048. вашим веб-приложением. Если он может указать значение переменной
  1049. PHP, используя параметры HTTP или другой механизм, то может
  1050. заставить ваши SQL-запросы выполнять действия, для которых они не
  1051. предназначены - например, возвращение данных, на чтение которых лицо
  1052. не имеет прав. Это серьезное и широко распространенное нарушение
  1053. безопасности приложения, известное под названием "SQL-инъекции"
  1054. (см. <ulink url="http://ru.wikipedia.org/wiki/%D0%98%D0%BD%D1%8A%D0%B5%D0%BA%D1%86%D0%B8%D1%8F_SQL">http://ru.wikipedia.org/wiki/Инъекция_SQL</ulink>).
  1055. </para>
  1056. <para>
  1057. Класс адаптера Zend_Db предоставляет удобные функции для того, чтобы
  1058. уменьшить уязвимость приложения к SQL-инъекциям. Решение
  1059. состоит в том, чтобы экранировать специальные символы, такие, как
  1060. кавычки в значениях PHP, до того, как они будут включены в строки
  1061. запросов SQL. Это защищает как от случайных, так и от
  1062. целенаправленных манипуляций строками SQL через переменные PHP,
  1063. содержащие специальные символы.
  1064. </para>
  1065. <sect3 id="zend.db.adapter.quoting.quote">
  1066. <title>Использование <code>quote()</code></title>
  1067. <para>
  1068. Метод <code>quote()</code> принимает единственный аргумент -
  1069. скалярное строковое значение. Он возвращает значение с
  1070. специальными символами, экранированными соответствующим образом
  1071. для используемой вами СУРБД, и окруженным ограничителями
  1072. строковых значений. Стандартным ограничителем строковых значений
  1073. в SQL является одинарная кавычка (<code>'</code>).
  1074. </para>
  1075. <example id="zend.db.adapter.quoting.quote.example">
  1076. <title>Использование quote()</title>
  1077. <programlisting language="php"><![CDATA[
  1078. $name = $db->quote("O'Reilly");
  1079. echo $name;
  1080. // 'O\'Reilly'
  1081. $sql = "SELECT * FROM bugs WHERE reported_by = $name";
  1082. echo $sql;
  1083. // SELECT * FROM bugs WHERE reported_by = 'O\'Reilly'
  1084. ]]>
  1085. </programlisting>
  1086. </example>
  1087. <para>
  1088. Обратите внимание, что возвращаемое методом <code>quote()</code>
  1089. значение включает в себя окружающие кавычки. Этим
  1090. метод отличается от некоторых функций, которые экранируют
  1091. специальные символы, но не добавляют кавычки, например,
  1092. <ulink url="http://www.php.net/mysqli_real_escape_string">mysql_real_escape_string()</ulink>.
  1093. </para>
  1094. <para>
  1095. Данные могут требовать или не требовать заключения в кавычки в
  1096. зависимости от того, в каком контексте типа данных SQL они
  1097. используются. Например, в некоторых СУРБД целочисленное
  1098. значение не должно заключаться в кавычки, если оно
  1099. сравнивается со столбцом или выражением целочисленного типа.
  1100. Другими словами, следующий запрос является ошибочным в некоторых
  1101. реализациях SQL, если столбец <code>intColumn</code> имеет
  1102. целочисленный тип данных <code>INTEGER</code>.
  1103. <programlisting language="php"><![CDATA[
  1104. SELECT * FROM atable WHERE intColumn = '123'
  1105. ]]>
  1106. </programlisting>
  1107. </para>
  1108. <para>
  1109. Вы можете использовать необязательный второй аргумент метода
  1110. <code>quote()</code> для избирательного заключения в кавычки
  1111. тех типов данных SQL, которые вы укажете.
  1112. </para>
  1113. <example id="zend.db.adapter.quoting.quote.example-2">
  1114. <title>Использование quote() с указанием типа SQL</title>
  1115. <programlisting language="php"><![CDATA[
  1116. $value = '1234';
  1117. $sql = 'SELECT * FROM atable WHERE intColumn = '
  1118. . $db->quote($value, 'INTEGER');
  1119. ]]>
  1120. </programlisting>
  1121. </example>
  1122. <para>
  1123. Каждый класс Zend_Db_Adapter имеет закодированные имена типов
  1124. данных SQL для соответствующих СУРБД. Вы можете также
  1125. использовать константы <code>Zend_Db::INT_TYPE</code>,
  1126. <code>Zend_Db::BIGINT_TYPE</code> и
  1127. <code>Zend_Db::FLOAT_TYPE</code> для написания еще более
  1128. независимого от типа используемой СУРБД кода.
  1129. </para>
  1130. <para>
  1131. Zend_Db_Table автоматически указывает типы SQL для метода
  1132. <code>quote()</code> при генерации SQL-запросов, ссылающихся на
  1133. ключевые столбцы таблицы.
  1134. </para>
  1135. </sect3>
  1136. <sect3 id="zend.db.adapter.quoting.quote-into">
  1137. <title>Использование <code>quoteInto()</code></title>
  1138. <para>
  1139. Наиболее типичным случаем использования операции заключения в
  1140. кавычки является добавление переменной PHP в выражение или
  1141. оператор SQL. Вы можете использовать метод
  1142. <code>quoteInto()</code> для того, чтобы выполнить это за один
  1143. шаг. Этот метод принимает два аргумента: первый аргумент
  1144. является строкой, содержащей символ метки заполнения
  1145. (<code>?</code>), а второй аргумент - значением или
  1146. переменной PHP, которая должна быть подставлена вместо этой
  1147. метки заполнения.
  1148. </para>
  1149. <para>
  1150. Символ метки заполнения одинаковый в многих СУРБД для
  1151. позиционных параметров, но метод <code>quoteInto()</code> только
  1152. эмулирует параметры запроса. Этот метод просто добавляет
  1153. значение в строку, экранируя специальные символы и заключая его
  1154. в кавычки. В случае настоящих параметров запроса сохраняется
  1155. разделение между строкой SQL и параметрами, поскольку строка
  1156. запроса анализируется сервером СУРБД.
  1157. </para>
  1158. <example id="zend.db.adapter.quoting.quote-into.example">
  1159. <title>Использование quoteInto()</title>
  1160. <programlisting language="php"><![CDATA[
  1161. $sql = $db->quoteInto("SELECT * FROM bugs WHERE reported_by = ?", "O'Reilly");
  1162. echo $sql;
  1163. // SELECT * FROM bugs WHERE reported_by = 'O\'Reilly'
  1164. ]]>
  1165. </programlisting>
  1166. </example>
  1167. <para>
  1168. Вы можете использовать опциональный третий параметр метода
  1169. <code>quoteInto()</code> для указания типа данных SQL. Числовые
  1170. типы данных не заключаются в кавычки, остальные заключаются.
  1171. </para>
  1172. <example id="zend.db.adapter.quoting.quote-into.example-2">
  1173. <title>Использование quoteInto() с указанием типа SQL</title>
  1174. <programlisting language="php"><![CDATA[
  1175. $sql = $db->quoteInto("SELECT * FROM bugs WHERE bug_id = ?", '1234', 'INTEGER');
  1176. echo $sql;
  1177. // SELECT * FROM bugs WHERE reported_by = 1234
  1178. ]]>
  1179. </programlisting>
  1180. </example>
  1181. </sect3>
  1182. <sect3 id="zend.db.adapter.quoting.quote-identifier">
  1183. <title>Использование <code>quoteIdentifier()</code></title>
  1184. <para>
  1185. Значения являются не единственной частью синтаксиса SQL, которая
  1186. может изменяться. Если вы используете переменные PHP для имен
  1187. таблиц, столбцов и других идентификаторов в своих операторах
  1188. SQL, то эти строки тоже следует заключать в кавычки. По
  1189. умолчанию идентификаторы в SQL следуют тем же правилам
  1190. синтаксиса, что есть в PHP и других языках программирования.
  1191. Например, идентификаторы не должны содержать пробелы,
  1192. определенные знаки препинания, специальные символы или
  1193. международные символы. Также в синтаксисе SQL зарезервированы
  1194. некоторые слова, и они не должны использоваться в качестве
  1195. идентификаторов.
  1196. </para>
  1197. <para>
  1198. Тем не менее, в SQL есть возможность, которая называется
  1199. <emphasis>идентификаторы с ограничителями</emphasis> (delimited
  1200. identifiers), она дает б<emphasis>о</emphasis>льшие возможности
  1201. выбора идентификаторов. Если вы заключите идентификатор SQL в
  1202. кавычки требуемого типа, то можете использовать те
  1203. идентификаторы, которые были бы недопустимыми без кавычек.
  1204. Идентификаторы с ограничителями могут содержать пробелы, знаки
  1205. препинания и международные символы. Вы можете также использовать
  1206. зарезервированные слова SQL, если заключите их в ограничители
  1207. идентификаторов.
  1208. </para>
  1209. <para>
  1210. <code>quoteIdentifier()</code> работает так же, как
  1211. <code>quote()</code>, но он применяет символы ограничителей
  1212. идентификаторов к строке в соответствии с типом используемой
  1213. СУРБД. Например, стандартный SQL использует двойные кавычки
  1214. (<code>"</code>) в качестве ограничителей идентификаторов и
  1215. большинство типов СУРБД использует именно их. MySQL по умолчанию
  1216. использует обратные кавычки (<code>`</code>). Метод
  1217. <code>quoteIdentifier()</code> также экранирует специальные
  1218. символы в строковом аргументе.
  1219. </para>
  1220. <example id="zend.db.adapter.quoting.quote-identifier.example">
  1221. <title>Использование quoteIdentifier()</title>
  1222. <programlisting language="php"><![CDATA[
  1223. // мы можем иметь имя таблицы,
  1224. // которое является зарезервированным в SQL словом
  1225. $tableName = $db->quoteIdentifier("order");
  1226. $sql = "SELECT * FROM $tableName";
  1227. echo $sql
  1228. // SELECT * FROM "order"
  1229. ]]>
  1230. </programlisting>
  1231. </example>
  1232. <para>
  1233. Идентификаторы с ограничителями в SQL являются чувствительными к
  1234. регистру, в отличие от не заключенных в кавычки. Поэтому, если
  1235. вы используете идентификаторы с ограничителями, то должны
  1236. использовать в точности то же написание идентификаторов, как и в
  1237. схеме БД, включая регистр букв.
  1238. </para>
  1239. <para>
  1240. В большинстве случаев, когда SQL генерируется в классах Zend_Db,
  1241. все идентификаторы по умолчанию автоматически заключаются в
  1242. ограничители. Вы можете изменить это поведение с помощью опции
  1243. <code>Zend_Db::AUTO_QUOTE_IDENTIFIERS</code>. Указывайте ее при
  1244. инстанцировании объекта адаптера. См.
  1245. <xref linkend="zend.db.adapter.connecting.parameters.example2" />.
  1246. </para>
  1247. </sect3>
  1248. </sect2>
  1249. <sect2 id="zend.db.adapter.transactions">
  1250. <title>Управление транзакциями</title>
  1251. <para>
  1252. Базы данных описывают транзакции как логические единицы работы,
  1253. которые могут фиксироваться или откатываться как одно изменение,
  1254. даже если они затрагивают несколько таблиц. Все запросы к БД
  1255. выполняются в контексте транзакций, даже если драйвер баз данных
  1256. работает с ними неявным образом. Это называется режимом
  1257. <emphasis>автоматической фиксации</emphasis>, в котором драйвера БД
  1258. создают транзакции для каждого выполняемого SQL-оператора. По
  1259. умолчанию все классы адаптеров Zend_Db функционируют в режиме
  1260. автоматической фиксации.
  1261. </para>
  1262. <para>
  1263. Вы можете также задавать начало и конец транзакции, и таким образом
  1264. контролировать число SQL-запросов в группе, которая фиксируется (или
  1265. откатывается) как одна операция.
  1266. Используйте метод <code>beginTransaction()</code> для инициирования
  1267. транзакции. Последующие SQL-операторы будут выполняться в контексте
  1268. этой транзакции до тех пор, пока вы не завершите ее явным образом.
  1269. </para>
  1270. <para>
  1271. Для завершения транзакции используйте методы <code>commit()</code>
  1272. или <code>rollBack()</code>. Метод <code>commit()</code> помечает
  1273. изменения, произведенные в течение данной транзакции, как
  1274. зафиксированные, это означает, что результаты этих изменений будут
  1275. видны в запросах, выполняемых в других транзакциях.
  1276. </para>
  1277. <para>
  1278. Метод <code>rollBack()</code> делает обратное - он не учитывает
  1279. изменения, произведенные в течение транзакции. Изменения будут
  1280. эффективно отменены, и состояние данных вернется к тому, в котором
  1281. они были до того, как была начата транзакция. Тем не менее, откат
  1282. транзакции не повлияет на изменения, произведенные другими
  1283. транзакциями, запущенными в это же время.
  1284. </para>
  1285. <para>
  1286. После того, как вы завершите транзакцию,
  1287. <code>Zend_Db_Adapter</code> вернется в режим автоматической
  1288. фиксации до того, как вы не вызовете <code>beginTransaction()</code> снова.
  1289. </para>
  1290. <example id="zend.db.adapter.transactions.example">
  1291. <title>Управление транзакциями для обеспечения согласованности данных</title>
  1292. <programlisting language="php"><![CDATA[
  1293. // Старт транзакции явным образом
  1294. $db->beginTransaction();
  1295. try {
  1296. // Попытка произвести один или несколько запросов
  1297. $db->query(...);
  1298. $db->query(...);
  1299. $db->query(...);
  1300. // Если все запросы были произведены успешно, то транзакция фиксируется,
  1301. // и все изменения фиксируются одновременно
  1302. $db->commit();
  1303. } catch (Exception $e) {
  1304. // Если какой-либо из этих запросов прошел неудачно, то вся транзакция
  1305. // откатывается, при этом все изменения отменяются, даже те, которые были
  1306. // произведены успешно.
  1307. // Таким образом, все изменения либо фиксируются, либо не фиксируется вместе.
  1308. $db->rollBack();
  1309. echo $e->getMessage();
  1310. }
  1311. ]]>
  1312. </programlisting>
  1313. </example>
  1314. </sect2>
  1315. <sect2 id="zend.db.adapter.list-describe">
  1316. <title>Листинг и описание таблиц</title>
  1317. <para>
  1318. Метод <code>listTables()</code> возвращает массив имен всех
  1319. таблиц в текущей базе данных.
  1320. </para>
  1321. <para>
  1322. Метод <code>describeTable()</code> возвращает ассоциативный массив
  1323. метаданных таблицы. Указывайте имя таблицы в качестве первого
  1324. аргумента этого метода. Второй аргумент является опциональным, и
  1325. обозначает схему, в которой существует эта таблица.
  1326. </para>
  1327. <para>
  1328. Ключами возвращаемого ассоциативного массива являются имена столбцов
  1329. таблицы. Значения, соответствующие этим столбцам, также являются
  1330. ассоциативными массивами со следующими ключами и значениями:
  1331. </para>
  1332. <table frame="all" cellpadding="5" id="zend.db.adapter.list-describe.metadata">
  1333. <title>Поля метаданных, возвращаемых методом describeTable()</title>
  1334. <tgroup cols="3" align="left" colsep="1" rowsep="1">
  1335. <thead>
  1336. <row>
  1337. <entry>Ключ</entry>
  1338. <entry>Тип</entry>
  1339. <entry>Описание</entry>
  1340. </row>
  1341. </thead>
  1342. <tbody>
  1343. <row>
  1344. <entry>SCHEMA_NAME</entry>
  1345. <entry>(string)</entry>
  1346. <entry>Имя схемы БД, в которой находится эта таблица.</entry>
  1347. </row>
  1348. <row>
  1349. <entry>TABLE_NAME</entry>
  1350. <entry>(string)</entry>
  1351. <entry>Имя таблицы, которой принадлежит данный столбец.</entry>
  1352. </row>
  1353. <row>
  1354. <entry>COLUMN_NAME</entry>
  1355. <entry>(string)</entry>
  1356. <entry>Имя столбца</entry>
  1357. </row>
  1358. <row>
  1359. <entry>COLUMN_POSITION</entry>
  1360. <entry>(integer)</entry>
  1361. <entry>Порядковый номер столбца в таблице.</entry>
  1362. </row>
  1363. <row>
  1364. <entry>DATA_TYPE</entry>
  1365. <entry>(string)</entry>
  1366. <entry>Имя типа данных столбца, используемое в данной СУРБД</entry>
  1367. </row>
  1368. <row>
  1369. <entry>DEFAULT</entry>
  1370. <entry>(string)</entry>
  1371. <entry>Значение по умолчанию, если есть.</entry>
  1372. </row>
  1373. <row>
  1374. <entry>NULLABLE</entry>
  1375. <entry>(boolean)</entry>
  1376. <entry>TRUE, если столбец допускает значение NULL, иначе FALSE.</entry>
  1377. </row>
  1378. <row>
  1379. <entry>LENGTH</entry>
  1380. <entry>(integer)</entry>
  1381. <entry>Длина или значение столбца, сообщаемое СУРБД.</entry>
  1382. </row>
  1383. <row>
  1384. <entry>SCALE</entry>
  1385. <entry>(integer)</entry>
  1386. <entry>Масштаб для типа данных NUMERIC или DECIMAL.</entry>
  1387. </row>
  1388. <row>
  1389. <entry>PRECISION</entry>
  1390. <entry>(integer)</entry>
  1391. <entry>Точность для типа данных NUMERIC или DECIMAL.</entry>
  1392. </row>
  1393. <row>
  1394. <entry>UNSIGNED</entry>
  1395. <entry>(boolean)</entry>
  1396. <entry>TRUE, если целочисленный тип объявлен как UNSIGNED (беззнаковое число).</entry>
  1397. </row>
  1398. <row>
  1399. <entry>PRIMARY</entry>
  1400. <entry>(boolean)</entry>
  1401. <entry>TRUE, если столбец является частью первичного ключа этой таблицы.</entry>
  1402. </row>
  1403. <row>
  1404. <entry>PRIMARY_POSITION</entry>
  1405. <entry>(integer)</entry>
  1406. <entry>Порядковый номер (начинается с 1) данного столбца в первичном ключе.</entry>
  1407. </row>
  1408. <row>
  1409. <entry>IDENTITY</entry>
  1410. <entry>(boolean)</entry>
  1411. <entry>TRUE, если данный столбец использует автоматически генерируемые значения.</entry>
  1412. </row>
  1413. </tbody>
  1414. </tgroup>
  1415. </table>
  1416. <para>
  1417. Если таблица, соответствующая заданным имени таблицы и имени схемы
  1418. (опционально), не существует, то <code>describeTable()</code>
  1419. возвращает пустой массив.
  1420. </para>
  1421. <note>
  1422. <title>Как поле метаданных IDENTITY соотносится с типом СУРБД</title>
  1423. <para>
  1424. Поле метаданных IDENTITY было выбрано в качестве
  1425. "идиоматического" термина для представления связи с
  1426. суррогатными ключами. Это поле обычно знакомо под следующими
  1427. именами:
  1428. </para>
  1429. <itemizedlist>
  1430. <listitem>
  1431. <para>
  1432. <code>IDENTITY</code> - DB2, MSSQL
  1433. </para>
  1434. </listitem>
  1435. <listitem>
  1436. <para>
  1437. <code>AUTO_INCREMENT</code> - MySQL
  1438. </para>
  1439. </listitem>
  1440. <listitem>
  1441. <para>
  1442. <code>SERIAL</code> - PostgreSQL
  1443. </para>
  1444. </listitem>
  1445. <listitem>
  1446. <para>
  1447. <code>SEQUENCE</code> - Oracle
  1448. </para>
  1449. </listitem>
  1450. </itemizedlist>
  1451. </note>
  1452. </sect2>
  1453. <sect2 id="zend.db.adapter.closing">
  1454. <title>Закрытие соединения</title>
  1455. <para>
  1456. Обычно нет необходимости в том, чтобы закрывать соединение с БД.
  1457. PHP автоматически очищает все ресурсы в конце запроса. Расширения
  1458. PHP для баз данных спроектированы таким образом, чтобы они закрывали
  1459. соединение, когда удаляется ссылка на объект ресурса.
  1460. </para>
  1461. <para>
  1462. Тем не менее, если у вас есть скрипт PHP длительного времени
  1463. выполнения, который инициирует множество соединений с БД, то может
  1464. потребоваться закрывать соединения, чтобы избежать снижения
  1465. производительности сервера СУРБД. Вы можете использовать метод
  1466. адаптера <code>closeConnection()</code> для явного закрытия лежащего
  1467. в основе соединения с БД.
  1468. </para>
  1469. <example id="zend.db.adapter.closing.example">
  1470. <title>Закрытие соединения с БД</title>
  1471. <programlisting language="php"><![CDATA[
  1472. $db->closeConnection();
  1473. ]]>
  1474. </programlisting>
  1475. </example>
  1476. <note>
  1477. <title>Поддерживает ли Zend_Db постоянные соединения?</title>
  1478. <para>
  1479. Использование постоянных соединений не поддерживается или
  1480. рекомендуется в Zend_Db.
  1481. </para>
  1482. <para>
  1483. Использование постоянных соединений может привести к избытку
  1484. неиспользуемых соединений на сервере СУРБД, что приносит больше
  1485. проблем, чем дает выигрыша в производительности, достигаемого
  1486. путем уменьшения накладных расходов на установку соединений.
  1487. </para>
  1488. <para>
  1489. Соединения с БД имеют свое состояние, т.е. некоторые объекты на
  1490. сервере СУРБД существуют в области видимости сессии. Примером
  1491. являются блокировки, пользовательские переменные, временные
  1492. таблицы и информация о последних выполненных запросах, такая,
  1493. как количество затронутых строк и последнее сгенерированное
  1494. значение. Если вы используете постоянные соединения, то ваше
  1495. приложение может получать неверные или привилегированные данные,
  1496. созданные в предыдущем PHP-запросе.
  1497. </para>
  1498. </note>
  1499. </sect2>
  1500. <sect2 id="zend.db.adapter.other-statements">
  1501. <title>Запуск других операторов БД</title>
  1502. <para>
  1503. Может потребоваться получить прямой доступ к
  1504. объекту соединения в том виде, в котором он предоставляется
  1505. расширением PHP для баз данных. Некоторые из этих расширений могут
  1506. предоставлять функционал, который не поддерживается методами
  1507. Zend_Db_Adapter_Abstract.
  1508. </para>
  1509. <para>
  1510. Например, все операторы SQL, запускаемые через Zend_Db,
  1511. подготавливаются перед выполнением. Однако некоторый функционал баз
  1512. данных несовместим с подготовленными операторами. Операторы DDL,
  1513. такие, как CREATE и ALTER, не могут подготавливаться в MySQL. Также
  1514. операторы SQL не дают выигрыша от
  1515. <ulink url="http://dev.mysql.com/doc/refman/5.1/en/query-cache-how.html">кэширования запросов MySQL</ulink> в версиях MySQL до 5.1.17.
  1516. </para>
  1517. <para>
  1518. Большинство расширений PHP для баз данных предоставляет метод для
  1519. выполнения операторов SQL без их подготовки. Например, в PDO таким
  1520. методом является <code>exec()</code>. Вы можете обратиться напрямую
  1521. к объекту соединения в расширении PHP, используя getConnection().
  1522. </para>
  1523. <example id="zend.db.adapter.other-statements.example">
  1524. <title>Запуск неподготовленного оператора в адаптере PDO</title>
  1525. <programlisting language="php"><![CDATA[
  1526. $result = $db->getConnection()->exec('DROP TABLE bugs');
  1527. ]]>
  1528. </programlisting>
  1529. </example>
  1530. <para>
  1531. Так же вы можете получить доступ к другим методам или свойствам,
  1532. специфическим для данного расширения. Тем не менее, следует
  1533. учитывать, что, делая это, вы можете ограничить ваше приложение
  1534. интерфейсом, предоставляемым расширением для определенной
  1535. СУРБД.
  1536. </para>
  1537. <para>
  1538. В будущих версиях Zend_Db будет возможность добавить точки входа
  1539. методов для функционала, который является общим для поддерживаемых
  1540. расширений PHP. Это не нарушит обратную совместимость.
  1541. </para>
  1542. </sect2>
  1543. <sect2 id="zend.db.adapter.adapter-notes">
  1544. <title>Примечания к отдельным адаптерам</title>
  1545. <para>
  1546. В данный разделе описываются различия между классами адаптеров, о
  1547. которых следует знать.
  1548. </para>
  1549. <sect3 id="zend.db.adapter.adapter-notes.ibm-db2">
  1550. <title>IBM DB2</title>
  1551. <itemizedlist>
  1552. <listitem>
  1553. <para>
  1554. Для установки этого адаптера через метод factory() используйте строку 'Db2'.
  1555. </para>
  1556. </listitem>
  1557. <listitem>
  1558. <para>
  1559. Этот адаптер использует PHP-расширение ibm_db2.
  1560. </para>
  1561. </listitem>
  1562. <listitem>
  1563. <para>
  1564. IBM DB2 поддерживает как последовательности, так и
  1565. автоинкрементные ключи. Поэтому аргументы для
  1566. <code>lastInsertId()</code> являются опциональными. Если
  1567. вы не передадите аргументы, то адаптер вернет последнее
  1568. значение, сгенерированное для автоинкрементного ключа.
  1569. Если вы передадите аргументы, то адаптер вернет
  1570. последнее значение, сгенерированное последовательностью,
  1571. имя которой удовлетворяет соглашению
  1572. '<emphasis>таблица</emphasis>_<emphasis>имя</emphasis>_seq'.
  1573. </para>
  1574. </listitem>
  1575. </itemizedlist>
  1576. </sect3>
  1577. <sect3 id="zend.db.adapter.adapter-notes.mysqli">
  1578. <title>MySQLi</title>
  1579. <itemizedlist>
  1580. <listitem>
  1581. <para>
  1582. Для установки этого адаптера через метод factory() используйте строку 'Mysqli'.
  1583. </para>
  1584. </listitem>
  1585. <listitem>
  1586. <para>
  1587. Этот адаптер использует PHP-расширение mysqli.
  1588. </para>
  1589. </listitem>
  1590. <listitem>
  1591. <para>
  1592. MySQL не поддерживает последовательности, поэтому
  1593. <code>lastInsertId()</code> игнорирует переданные
  1594. аргументы и всегда возвращает последнее значение,
  1595. сгенерированное для автоинкрементного ключа. Метод
  1596. <code>lastSequenceId()</code> возвращает
  1597. <constant>NULL</constant>.
  1598. </para>
  1599. </listitem>
  1600. </itemizedlist>
  1601. </sect3>
  1602. <sect3 id="zend.db.adapter.adapter-notes.oracle">
  1603. <title>Oracle</title>
  1604. <itemizedlist>
  1605. <listitem>
  1606. <para>
  1607. Для установки этого адаптера через метод factory() используйте строку 'Oracle'.
  1608. </para>
  1609. </listitem>
  1610. <listitem>
  1611. <para>
  1612. Этот адаптер использует PHP-расширение oci8.
  1613. </para>
  1614. </listitem>
  1615. <listitem>
  1616. <para>
  1617. Oracle не поддерживает автоинкрементные ключи, поэтому
  1618. вы должны указывать имя последовательности для
  1619. <code>lastInsertId()</code> или
  1620. <code>lastSequenceId()</code>.
  1621. </para>
  1622. </listitem>
  1623. <listitem>
  1624. <para>
  1625. Расширение Oracle не поддерживает позиционные параметры.
  1626. Вы должны использовать именованные параметры.
  1627. </para>
  1628. </listitem>
  1629. <listitem>
  1630. <para>
  1631. На данный момент опция
  1632. <code>Zend_Db::CASE_FOLDING</code> не поддерживается
  1633. адаптером Oracle. Для того, чтобы применять эту опцию с
  1634. Oracle, вам нужно использовать адаптер PDO OCI.
  1635. </para>
  1636. </listitem>
  1637. </itemizedlist>
  1638. </sect3>
  1639. <sect3 id="zend.db.adapter.adapter-notes.pdo-ibm">
  1640. <title>PDO для IBM DB2 и Informix Dynamic Server (IDS)</title>
  1641. <itemizedlist>
  1642. <listitem>
  1643. <para>
  1644. Для установки этого адаптера через метод factory() используйте строку 'Pdo_Ibm'.
  1645. </para>
  1646. </listitem>
  1647. <listitem>
  1648. <para>
  1649. Этот адаптер использует PHP-расширения pdo и pdo_ibm.
  1650. </para>
  1651. </listitem>
  1652. <listitem>
  1653. <para>
  1654. Вы должны использовать расширение PDO_IBM версии не ниже
  1655. 1.2.2. Если вы используете более раннюю версию этого
  1656. расширения, то должны обновить расширение PDO_IBM из
  1657. PECL.
  1658. </para>
  1659. </listitem>
  1660. </itemizedlist>
  1661. </sect3>
  1662. <sect3 id="zend.db.adapter.adapter-notes.pdo-mssql">
  1663. <title>PDO Microsoft SQL Server</title>
  1664. <itemizedlist>
  1665. <listitem>
  1666. <para>
  1667. Для установки этого адаптера через метод factory() используйте строку 'Pdo_Mssql'.
  1668. </para>
  1669. </listitem>
  1670. <listitem>
  1671. <para>
  1672. Этот адаптер использует PHP-расширения pdo и pdo_mssql.
  1673. </para>
  1674. </listitem>
  1675. <listitem>
  1676. <para>
  1677. Microsoft SQL Server не поддерживает последовательности,
  1678. поэтому <code>lastInsertId()</code> игнорирует
  1679. переданные аргументы и всегда возвращает последнее
  1680. значение, сгенерированное для автоинкрементного ключа.
  1681. Метод <code>lastSequenceId()</code> возвращает
  1682. <constant>NULL</constant>.
  1683. </para>
  1684. </listitem>
  1685. <listitem>
  1686. <para>
  1687. Zend_Db_Adapter_Pdo_Mssql устанавливает
  1688. <code>QUOTED_IDENTIFIER ON</code> сразу после соединения
  1689. с сервером баз данных. Это заставляет драйвер
  1690. использовать стандартные символы-ограничители
  1691. идентификаторов (<code>"</code>) вместо квадратных
  1692. скобок, которые SQL Server использует в качестве
  1693. ограничителей идентификаторов.
  1694. </para>
  1695. </listitem>
  1696. <listitem>
  1697. <para>
  1698. Вы можете указывать <code>pdoType</code> в качестве
  1699. ключа в массиве опций. Возможными значениями могут быть
  1700. "mssql" (по умолчанию),
  1701. "dblib", "freetds" или "sybase". Эта опция влияет на
  1702. префикс DSN, который используется адаптером, когда
  1703. строится строка DSN. "Freetds" и "sybase" подразумевают
  1704. префикс "sybase:", который используется для набора
  1705. библиотек <ulink url="http://www.freetds.org/">FreeTDS</ulink>. Более
  1706. подробную информацию о префиксах, используемых в этих
  1707. драйверах, читайте на
  1708. <ulink url="http://www.php.net/manual/en/ref.pdo-dblib.connection.php">
  1709. http://www.php.net/manual/en/ref.pdo-dblib.connection.php</ulink>.
  1710. </para>
  1711. </listitem>
  1712. </itemizedlist>
  1713. </sect3>
  1714. <sect3 id="zend.db.adapter.adapter-notes.pdo-mysql">
  1715. <title>PDO MySQL</title>
  1716. <itemizedlist>
  1717. <listitem>
  1718. <para>
  1719. Для установки этого адаптера через метод factory() используйте строку 'Pdo_Mysql'.
  1720. </para>
  1721. </listitem>
  1722. <listitem>
  1723. <para>
  1724. Этот адаптер использует PHP-расширения pdo и pdo_mysql.
  1725. </para>
  1726. </listitem>
  1727. <listitem>
  1728. <para>
  1729. MySQL не поддерживает последовательности,
  1730. поэтому <code>lastInsertId()</code> игнорирует
  1731. переданные аргументы и всегда возвращает последнее
  1732. значение, сгенерированное для автоинкрементного ключа.
  1733. Метод <code>lastSequenceId()</code> возвращает
  1734. <constant>NULL</constant>.
  1735. </para>
  1736. </listitem>
  1737. </itemizedlist>
  1738. </sect3>
  1739. <sect3 id="zend.db.adapter.adapter-notes.pdo-oci">
  1740. <title>PDO Oracle</title>
  1741. <itemizedlist>
  1742. <listitem>
  1743. <para>
  1744. Для установки этого адаптера через метод factory() используйте строку 'Pdo_Oci'.
  1745. </para>
  1746. </listitem>
  1747. <listitem>
  1748. <para>
  1749. Этот адаптер использует PHP-расширения pdo и pdo_oci.
  1750. </para>
  1751. </listitem>
  1752. <listitem>
  1753. <para>
  1754. Oracle не поддерживает автоинкрементные ключи, поэтому
  1755. вы должны указывать имя последовательности для
  1756. <code>lastInsertId()</code> или
  1757. <code>lastSequenceId()</code>.
  1758. </para>
  1759. </listitem>
  1760. </itemizedlist>
  1761. </sect3>
  1762. <sect3 id="zend.db.adapter.adapter-notes.pdo-pgsql">
  1763. <title>PDO PostgreSQL</title>
  1764. <itemizedlist>
  1765. <listitem>
  1766. <para>
  1767. Для установки этого адаптера через метод factory() используйте строку 'Pdo_Pgsql'.
  1768. </para>
  1769. </listitem>
  1770. <listitem>
  1771. <para>
  1772. Этот адаптер использует PHP-расширения pdo и pdo_pgsql.
  1773. </para>
  1774. </listitem>
  1775. <listitem>
  1776. <para>
  1777. PostgreSQL поддерживает как последовательности, так и
  1778. автоинкрементные ключи. Поэтому аргументы для
  1779. <code>lastInsertId()</code> являются опциональными. Если
  1780. вы не передадите аргументы, то адаптер вернет последнее
  1781. значение, сгенерированное для автоинкрементного ключа.
  1782. Если вы передадите аргументы, то адаптер вернет
  1783. последнее значение, сгенерированное последовательностью,
  1784. имя которой удовлетворяет соглашению
  1785. '<emphasis>таблица</emphasis>_<emphasis>имя</emphasis>_seq'.
  1786. </para>
  1787. </listitem>
  1788. </itemizedlist>
  1789. </sect3>
  1790. <sect3 id="zend.db.adapter.adapter-notes.pdo-sqlite">
  1791. <title>PDO SQLite</title>
  1792. <itemizedlist>
  1793. <listitem>
  1794. <para>
  1795. Для установки этого адаптера через метод factory() используйте строку 'Pdo_Sqlite'.
  1796. </para>
  1797. </listitem>
  1798. <listitem>
  1799. <para>
  1800. Этот адаптер использует PHP-расширения pdo и pdo_sqlite.
  1801. </para>
  1802. </listitem>
  1803. <listitem>
  1804. <para>
  1805. SQLite не поддерживает последовательности,
  1806. поэтому <code>lastInsertId()</code> игнорирует
  1807. переданные аргументы и всегда возвращает последнее
  1808. значение, сгенерированное для автоинкрементного ключа.
  1809. Метод <code>lastSequenceId()</code> возвращает
  1810. <constant>NULL</constant>.
  1811. </para>
  1812. </listitem>
  1813. <listitem>
  1814. <para>
  1815. Для того, чтобы соединится с базой данных SQLite2,
  1816. указывайте <code>'sqlite2'=>true</code> в массиве
  1817. параметров при создании экземпляра адаптера Pdo_Sqlite.
  1818. </para>
  1819. </listitem>
  1820. <listitem>
  1821. <para>
  1822. Для соединения с базой данных SQLite в памяти указывайте
  1823. <code>'dbname'=>':memory:'</code> в массиве параметров
  1824. при создании экземпляра адаптера Pdo_Sqlite.
  1825. </para>
  1826. </listitem>
  1827. <listitem>
  1828. <para>
  1829. Старые версии драйвера SQLite для PHP могут не
  1830. поддерживать команды PRAGMA, необходимые для обеспечения
  1831. использования коротких имен столбцов в результатах. Если
  1832. имеются проблемы с тем, что результаты возвращаются с
  1833. ключами в виде "tablename.columnname", когда производится запрос с объединением таблиц, то следует обновить PHP до текущей версии.
  1834. </para>
  1835. </listitem>
  1836. </itemizedlist>
  1837. </sect3>
  1838. <sect3 id="zend.db.adapter.adapter-notes.firebird">
  1839. <title>Firebird/Interbase</title>
  1840. <itemizedlist>
  1841. <listitem>
  1842. <para>
  1843. Этот адаптер использует PHP-расширение php_interbase.
  1844. </para>
  1845. </listitem>
  1846. <listitem>
  1847. <para>
  1848. Firebird/interbase не поддерживает автоинкрементные
  1849. ключи, поэтому вы должны указывать имя
  1850. последовательности для
  1851. <code>lastInsertId()</code> или
  1852. <code>lastSequenceId()</code>.
  1853. </para>
  1854. </listitem>
  1855. <listitem>
  1856. <para>
  1857. На данный момент опция
  1858. <code>Zend_Db::CASE_FOLDING</code> не поддерживается
  1859. адаптером Firebird/interbase. Не заключенные в кавычки
  1860. идентификаторы автоматически возвращаются в верхнем
  1861. регистре..
  1862. </para>
  1863. </listitem>
  1864. </itemizedlist>
  1865. </sect3>
  1866. </sect2>
  1867. </sect1>
  1868. <!--
  1869. vim:se ts=4 sw=4 et:
  1870. -->