<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>SEO блог и уеб програмиране &#187; оптимизация</title>
	<atom:link href="http://ganbox.com/blog/tag/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b0%d1%86%d0%b8%d1%8f/feed/" rel="self" type="application/rss+xml" />
	<link>http://ganbox.com/blog</link>
	<description>SEO практика: трикове при оптимизация и решаване на проблеми при уеб програмиране.</description>
	<lastBuildDate>Sun, 20 May 2012 19:28:54 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.3.1</generator>
<xhtml:meta xmlns:xhtml="http://www.w3.org/1999/xhtml" name="robots" content="noindex" />
		<item>
		<title>Оптимизиране на SQL заявки</title>
		<link>http://ganbox.com/blog/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-sql-%d0%b7%d0%b0%d1%8f%d0%b2%d0%ba%d0%b8/</link>
		<comments>http://ganbox.com/blog/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-sql-%d0%b7%d0%b0%d1%8f%d0%b2%d0%ba%d0%b8/#comments</comments>
		<pubDate>Fri, 09 Apr 2010 09:48:41 +0000</pubDate>
		<dc:creator>Георги Стефанов</dc:creator>
				<category><![CDATA[Оптимизация на сайт]]></category>
		<category><![CDATA[Уеб програмиране]]></category>
		<category><![CDATA[mySQL]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[SQL]]></category>
		<category><![CDATA[SQL заявки]]></category>
		<category><![CDATA[бавни SQL заявки]]></category>
		<category><![CDATA[бързодействие]]></category>
		<category><![CDATA[оптимизация]]></category>
		<category><![CDATA[оптимизация на SQL]]></category>
		<category><![CDATA[оптимизиране]]></category>
		<category><![CDATA[оптимизиране на SQL]]></category>
		<category><![CDATA[урок]]></category>

		<guid isPermaLink="false">http://ganbox.com/blog/?p=848</guid>
		<description><![CDATA[Основни положения при оптимизация на mySQL Обикновено проблемите с бавни SQL заявки се дължат на грешно проектирана база данни, липса или грешно използване на индекси и най-често неправилно написани SQL заявки. Статията разглежда няколко примера от практиката, които дават идеи за нов начин на писане на оптимизирани SQL заявки, при който не се прехвърлят много [...]]]></description>
			<content:encoded><![CDATA[<h2>Основни положения при оптимизация на mySQL</h2>
<p>Обикновено проблемите с <strong>бавни SQL заявки</strong> се дължат на грешно проектирана база данни, липса или грешно използване на индекси и най-често неправилно написани SQL заявки.</p>
<p>Статията разглежда няколко примера от практиката, които дават идеи за нов начин на писане на <strong>оптимизирани SQL заявки</strong>, при който не се прехвърлят много данни, за да се увеличи бързодействието.</p>
<h3>Мониторинг на SQL заявки</h3>
<p>На първо място препоръчвам на всеки използващ mySQL да ползва лог на бавните заявки. Активира се лесно и записва всяка заявка, която отнеме за изпълнението си повече от предварително зададено време (обикновено 10 сек.). Все пак не може да решиш проблем, ако не знаеш, че съществува <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_wink.gif' alt=';)' class='wp-smiley' /><br />
Използвай EXPLAIN пред SELECT за информация относно заявката &#8211; кой индекс се използва, колко реда се преглеждат и др.<br />
За мониторинг в реално време препоръчвам mytop &#8211; приложение написано на Perl, което показва всички активни заявки и колко време отнемат.</p>
<h3>Няколко бързи съвета за начинаещи</h3>
<p>1. Ползвай колона id с атрибути PRIMARY и auto_increment за всяка таблица, дори да ти се струва, че никога няма да ти трябва.<br />
2. Избягвай винаги &#8222;SELECT * FROM &#8230;&#8220; вместо това пиши &#8222;SELECT col1, col2, col3&#8230; FROM &#8230;&#8220; &#8211; изброявай нужните колони вместо да вземаш всички данни.<br />
3. Никога не ползвай &#8222;SELECT (*) FROM &#8230;&#8220;, а пиши &#8222;SELECT (id) FROM &#8230;&#8220;, където id е колона с уникални стойности.<br />
4. При създаването на таблицата създай индекси по всяка една колона, по която в бъдеще се очаква да сортираш данните или да извличаш данни като търсиш по стойност в тази колона.<br />
5. Почти винаги т.4 е недостатъчна &#8211; трудно е да създадеш всички необходими индекси предварително. При написването на всяка нова заявка проверявай какви са колоните, по които се търси и провери дали има индекси по тях &#8211; ако няма ги създай веднага!<br />
6. Ползвай сложни индекси. Ако имаш заявка, в която се търси по повече от една колона например съдържа &#8222;&#8230; WHERE user_id = 8 AND date &gt; &#8217;2010-01-01&#8242; &#8222;, в този случай ако имаш отделни индекси по user_id и по date ще бъде автоматично избран и ползван само единия. Ако много често се изпълнява такава заявка (или си я открил в лога с бавни заявки) ти трябва сложен индекс по двете колони. Очаквай отделна статия по въпроса за комбинираните индекси.<br />
7. Нормализиране на таблици. Тук няма да влизам в детайли, а само ще спомена, че ако в някоя колона имаш данни, които изглеждат като: &#8222;4,5,13,45,345&#8243; &#8211; няколко стойности разделени със запетая, това почти винаги е знак за грешно проектирана база данни.<br />
8. Когато свързваш две таблици трябва типа на колоните, по които се свързва да съвпада точно, например ако едната е от тип INT(8), а другата е от тип INT(11) може да се очакват проблеми.</p>
<h3>Аксиома на gan за оптимизация на SQL</h3>
<p>Стреми се да пишеш SQL заявките си така, все едно уеб сървъра и SQL сървъра са на различни машини &#8211; пренесените данни между двата трябва да са минимални!</p>
<h2>Откриване на позицията на ред след сортиране на таблица</h2>
<p>Нека да предположим, че имаме сайт за игра и има таблица за класиране UserTop с колони username и points, в която за всеки потребител се записва, колко точки е спечелил общо от всички игри. Колоната username съдържа уникални стойности. Пример за SQL таблица UserTop:</p>
<table>
<tbody>
<tr>
<th>username</th>
<th>points</th>
</tr>
<tr>
<td>someuser</td>
<td>150</td>
</tr>
<tr>
<td>otheruser</td>
<td>50</td>
</tr>
<tr>
<td>ganbox</td>
<td>250</td>
</tr>
<tr>
<td>randomuser</td>
<td>350</td>
</tr>
<tr>
<td>vox</td>
<td>250</td>
</tr>
</tbody>
</table>
<p>След приключване на всяка игра по потребителско име се намира реда за играча и точките му се увеличават с тези от последната игра. Нека да кажем, че в момента е логнат потребител ganbox и искаме да покажем на играча на коя позиция е в класацията. Затова трябва да сортираме таблицата по колона points в намаляващ ред и да видим на коя позиция се намира ganbox. Един начинаещ програмист ще направи следното:</p>
<p><span style="color: #800000;">Грешно решение:</span></p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SELECT * FROM UserTop ORDER BY points DESC &quot;</span><span style="color: #339933;">;</span><br />
<span style="color: #000088;">$results</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$db</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">GetAll</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$query</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span> <span style="color: #666666; font-style: italic;"># всички резултати в масив (ползва се ADODB библиотека за краткост)<br />
</span><span style="color: #000088;">$pos</span><span style="color: #339933;">=</span><span style="color: #cc66cc;">0</span><span style="color: #339933;">;</span><br />
<span style="color: #b1b100;">for</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$i</span><span style="color: #339933;">=</span><span style="color: #cc66cc;">0</span><span style="color: #339933;">;</span> <span style="color: #000088;">$i</span><br />
<span style="color: #b1b100;">if</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$result</span><span style="color: #009900;">&#91;</span><span style="color: #000088;">$i</span><span style="color: #009900;">&#93;</span><span style="color: #009900;">&#91;</span><span style="color: #0000ff;">'username'</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">==</span> <span style="color: #0000ff;">'ganbox'</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#123;</span><br />
<span style="color: #000088;">$pos</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$i</span><span style="color: #339933;">+</span><span style="color: #cc66cc;">1</span><span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span><br />
<span style="color: #009900;">&#125;</span></div></div>
<p>Тук има няколко грешки:<br />
1. Взема се цялото съдържание на таблица UserTop &#8211; може да бъде огромно количество данни.<br />
2. На всяка итерация на цикъла for се оценява големината на масива $results.<br />
3. След като е намерена и записана позицията в pos, цикъла не се прекъсва, а продължава да се върти безсмислено.<br />
Не можете да си представите, колко много хора пишат по този небрежен начин. Затова много сайтове за игри работят отчайващо бавно при използване от много хора.</p>
<p>Следва малко по-добро, но все още неправилно решение.<br />
<span style="color: #800000;">Друго грешно решение:</span></p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SELECT username FROM UserTop ORDER BY points DESC &quot;</span><span style="color: #339933;">;</span><br />
<span style="color: #000088;">$results</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$db</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">GetAll</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$query</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span> <span style="color: #666666; font-style: italic;"># всички резултати в масив<br />
</span><span style="color: #000088;">$pos</span><span style="color: #339933;">=</span><span style="color: #cc66cc;">0</span><span style="color: #339933;">;</span><br />
<span style="color: #000088;">$size</span><span style="color: #339933;">=</span><a href="http://www.php.net/count"><span style="color: #990000;">count</span></a><span style="color: #009900;">&#40;</span><span style="color: #000088;">$results</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
<span style="color: #b1b100;">for</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$i</span><span style="color: #339933;">=</span><span style="color: #cc66cc;">0</span><span style="color: #339933;">;</span> <span style="color: #000088;">$i</span><span style="color: #339933;">&lt;</span><span style="color: #000088;">$size</span><span style="color: #339933;">;</span> <span style="color: #000088;">$i</span><span style="color: #339933;">++</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#123;</span><br />
<span style="color: #b1b100;">if</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$result</span><span style="color: #009900;">&#91;</span><span style="color: #000088;">$i</span><span style="color: #009900;">&#93;</span><span style="color: #009900;">&#91;</span><span style="color: #0000ff;">'username'</span><span style="color: #009900;">&#93;</span> <span style="color: #339933;">==</span> <span style="color: #0000ff;">'ganbox'</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#123;</span><br />
<span style="color: #000088;">$pos</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$i</span><span style="color: #339933;">+</span><span style="color: #cc66cc;">1</span><span style="color: #339933;">;</span><br />
<span style="color: #b1b100;">break</span><span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span><br />
<span style="color: #009900;">&#125;</span></div></div>
<p>Подобренията се виждат:<br />
1. Избират се само данните от колона username &#8211; все още връща всички редове и данните са прекалено много.<br />
2. Оценката на големината на масива е изнесена извън цикъла.<br />
3. След откриване на позицията цикъла спира.<br />
По точки 2. и 3. сме ОК, но т. 1 е голям проблем. Представи си, че уеб сървъра и SQL сървъра са на отделни машини &#8211; всички данни от колона username трябва да минат по мрежата, ако таблицата съдържа много редове и се ползва от много хора, това ще доведе до значително забавяне. Така стигаме до идеята, че тази задача трябва да се <strong>оптимизира</strong>, като обработката се изнесе към SQL сървъра.</p>
<p>Почти правилно решение:</p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SELECT COUNT( username ) AS pos FROM UserTop WHERE points &gt;= (SELECT points FROM UserTop WHERE username='ganbox' ) &quot;</span><span style="color: #339933;">;</span><br />
<span style="color: #000088;">$pos</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$db</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">GetOne</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$query</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span> <span style="color: #666666; font-style: italic;"># един резултат</span></div></div>
<p>Първо се изпълнява заявката в скобите, която връща точките на текущия играч (низа &#8216;ganbox&#8217; трябва да се замести с потребителското име на логнатия играч). След това се вземат всички редове, които имат повече точки (включително тези които имат равен брой точки с играча) и се преброяват. Връща се едно единствено число &#8211; прекрасно!<br />
Има един малък проблем. Ако има играчи с брой точки равен на броя точки на текущия адрес, позицията няма да е вярна. Затова:</p>
<p><span style="color: #008000;">Правилно решение:</span></p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SELECT COUNT( username ) AS pos FROM UserTop WHERE points &gt; (SELECT points FROM UserTop WHERE username='ganbox' ) &quot;</span><span style="color: #339933;">;</span><br />
<span style="color: #000088;">$pos</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$db</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">GetOne</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$query</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span> <span style="color: #666666; font-style: italic;"># един резултат<br />
</span><span style="color: #000088;">$pos</span><span style="color: #339933;">++;</span> <span style="color: #666666; font-style: italic;"># това е търсената позиция</span></div></div>
<p>Знакът е &gt; защото в класация сортираме от по-голямо към по-малко.</p>
<p>.</p>
<p>Ако ти е станало интересно продължавай да четеш, ще усложним задачите с примери от практиката.<br />
Нека освен таблица UserTop, която е класация имаме UserLog, в която в края на всяка игра се прави запис, като се записва време на приключване на играта и кой потребител колко точки е спечелил през последната игра:</p>
<table>
<tbody>
<tr>
<th>timestamp</th>
<th>username</th>
<th>poits</th>
</tr>
<tr>
<td>2010-01-10 11:12:23</td>
<td>someuser</td>
<td>100</td>
</tr>
<tr>
<td>2010-01-11 10:24:33</td>
<td>otheruser</td>
<td>20</td>
</tr>
<tr>
<td>2010-01-12 15:14:13</td>
<td>ganbox</td>
<td>60</td>
</tr>
<tr>
<td>2010-01-15 19:15:55</td>
<td>randomuser</td>
<td>90</td>
</tr>
<tr>
<td>2010-02-15 11:52:54</td>
<td>vox</td>
<td>80</td>
</tr>
<tr>
<td>2010-04-08 22:04:44</td>
<td>someuser</td>
<td>50</td>
</tr>
<tr>
<td>2010-04-08 22:05:07</td>
<td>otheruser</td>
<td>30</td>
</tr>
<tr>
<td>2010-04-08 22:05:34</td>
<td>ganbox</td>
<td>190</td>
</tr>
<tr>
<td>2010-04-08 22:06:17</td>
<td>randomuser</td>
<td>260</td>
</tr>
<tr>
<td>2010-04-08 22:06:51</td>
<td>vox</td>
<td>140</td>
</tr>
<tr>
<td>2010-04-08 22:07:58</td>
<td>vox</td>
<td>30</td>
</tr>
</tbody>
</table>
<p>Задачата е да подредим потребителите в класацията UserTop така, че ако няколко души имат равен брой точки по-напред да излиза този, който е играл по-скоро. В примера vox и ganbox имат еднакъв брой 250 точки, но vox трябва да излиза по-напред, защото е играл последен.</p>
<h3>Сортиране на данни по два критерия в различни таблици</h3>
<p>Заявката, която ще изведе играчите сортирани първо по брой точки, а тези с еднакъв брой по времето на последната активност:</p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SELECT T.username, T.points FROM UserTop T LEFT JOIN UserLog L USING(username) GROUP BY username ORDER BY T.points DESC , L.timestamp DESC&quot;</span><span style="color: #339933;">;</span></div></div>
<p>Дотук добре. Сега освен страницата с тази класация искаме да показваме последните 10 действия &#8211; последните 10 играчи, които са играли. Там разбира се може да има повторения. Заявката е елементарна, но ни подготвя за следващата задача.</p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SELECT username, points FROM `UserLog` ORDER BY `UserLog`.`timestamp` &nbsp;DESC LIMIT 10&quot;</span><span style="color: #339933;">;</span></div></div>
<p>Забеляза ли, че не ползвам &#8222;SELECT * FROM&#8220;, а избирам тези които ми трябват (не го забравяй!).</p>
<p>Добре стигнахме до интересното. Следват два примера от практиката.</p>
<h4>Откриване на класирането на играч в края на игра</h4>
<p>Искаме в края на всяка игра освен да добавим запис в UserLog и да увеличим точките на играча в UserTop, да намерим новата позиция на играча в класацията и общия брой на точките му, за да може да му съобщим тези данни.</p>
<p><span style="color: #008000;">Правилно решение:</span></p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot; SET @num:=0; SELECT username, pos, points FROM (SELECT @num := @num + 1 AS pos, T.points, T.username FROM UserTop T LEFT JOIN UserLog L USING(username) GROUP BY T.username ORDER BY T.points DESC, L.timestamp DESC) as R WHERE username = 'ganbox' &quot;</span><span style="color: #339933;">;</span></div></div>
<p>Вижда се, че ganbox е на 3-то място с 250 точки, а vox на второ пак с 250, но е играл по-скоро.<br />
Заявката отново е <strong>супер оптимизирана</strong> <em>според аксиомата на gan</em> <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_smile.gif' alt=':)' class='wp-smiley' />  така, че да върне само нужните данни.<br />
Използва вътрешна SQL променлива @num, която първоначално се нулира. След това се изпълнява вътрешната заявка, която написана отделно:</p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot; SET @num := 0; SELECT @num := @num + 1 AS pos, T.points, T.username FROM UserTop T LEFT JOIN UserLog L USING(username) GROUP BY T.username ORDER BY T.points DESC, L.timestamp DESC&quot;</span> <span style="color: #339933;">;</span></div></div>
<p>ще върне всички играчи в класацията с добавена колона с име pos, в която е позицията на всеки играч. По този начин външната заявка има задачата само да филтрира данните за играча, който ни интересува.</p>
<h4>Откриване на запис отпаднал от последните 10 действия</h4>
<p>Искаме още в края на всяка игра, да намерим кой играч е отпаднал на 11-място в лога (т.е. името му не се вижда вече в последните 10) и само ако името му го няма в първите 10 резултата да предприемем действие &#8211; например да го поканим да играе отново.</p>
<p>Как да решим тази задача? Можем лесно да вземем играча на позиция 11, както и да изведем всички от първите 10 позиции, но как да проверим дали този от 11-та се съдържа в първите 10?<br />
Единия подход е с една заявка да вземем играча X от 11 позиция. След това с втора заявка да вземем потребителските имена на всички от първите 10, да ги заредим в PHP масив и да потърсим X в масива. Както вече се досети този подход не ми харесва, защото за 10 не е страшно, но ако са 100, а ако са 1000? Трябва ни отново <strong>оптимизирано решение</strong> с минимално прехвърляне на данни.</p>
<p><span style="color: #008000;">Правилно решение:</span></p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #000088;">$query</span> <span style="color: #339933;">=</span> <span style="color: #0000ff;">&quot;SET @L:=(SELECT username FROM UserLog ORDER BY timestamp DESC LIMIT 10,1); SELECT @L AS username,SUM(F) AS cnt FROM (SELECT IF(username=@L,1,0) AS F FROM UserLog ORDER BY timestamp DESC LIMIT 0,10) AS T&quot;</span><span style="color: #339933;">;</span></div></div>
<p>Тази заявка ще върне резултат:</p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border:1px solid #9F9F9F;width:435px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><a href="http://www.php.net/array"><span style="color: #990000;">Array</span></a><br />
<span style="color: #009900;">&#40;</span><br />
<span style="color: #009900;">&#91;</span>username<span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> <span style="color: #0000ff;">'someuser'</span><span style="color: #339933;">,</span><br />
<span style="color: #009900;">&#91;</span>cnt<span style="color: #009900;">&#93;</span> <span style="color: #339933;">=&gt;</span> &nbsp; &nbsp;<span style="color: #cc66cc;">1</span><br />
<span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span></div></div>
<p>Наистина ако сортираш UserLog от примера по време в обратен ред, someuser е на 11 позиция. Също така се среща на 6-та позиция и затова cnt е 1, като cnt показва не само дали се среща, но и колко пъти се среща в позициите от 1 до 10. По този начин предприемаш действие само ако cnt=0.</p>
<p>Това бяха само някои идеи за нов стил на писане на SQL заявки с цел <strong>оптимизиране на бързодействието</strong>. За пълнота ще спомена, че в много случаи отлични резултати могат да се постигнат при използване на <strong>кеширане на SQL заявки</strong>. Самото кеширане на заявки се програмира лесно. Трудното е да се определи в кой момент да се ползва кеш. Темата ще бъде разгледана в друга статия.</p>
<p>Всички SQL заявки са тествани и работят. Ще се радвам на коментари по темата.</p>
<div style="float:right;display:block" class="counterViews">Брой разглеждания на тази статия: <b>5257</b></div>

<p>Подобни статии:<ol><li><a href='http://ganbox.com/blog/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d0%b1%d1%8a%d1%80%d0%b7%d0%be%d0%b4%d0%b5%d0%b9%d1%81%d1%82%d0%b2%d0%b8%d0%b5%d1%82%d0%be/' rel='bookmark' title='Permanent Link: Оптимизиране на бързодействието на сайт'>Оптимизиране на бързодействието на сайт</a> <small>Пиша тази статия като продължение на статията PHP парсване на...</small></li>
<li><a href='http://ganbox.com/blog/%d0%b7%d0%b0%d1%80%d0%b5%d0%b6%d0%b4%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d0%b4%d0%b0%d0%bd%d0%bd%d0%b8-%d1%81-ajax-%d0%be%d1%82-%d0%be%d1%82%d0%b4%d0%b0%d0%bb%d0%b5%d1%87%d0%b5%d0%bd-%d1%81%d1%8a%d1%80/' rel='bookmark' title='Permanent Link: Зареждане на данни с Ajax от отдалечен сървър'>Зареждане на данни с Ajax от отдалечен сървър</a> <small>Задача за оптимизиране бързината на сайт Наскоро работих по сайта...</small></li>
<li><a href='http://ganbox.com/blog/seo-%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-oscommerce/' rel='bookmark' title='Permanent Link: SEO оптимизиране на osCommerce'>SEO оптимизиране на osCommerce</a> <small>Тази статия е за SEO оптимизиране на онлайн магазин от...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://ganbox.com/blog/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-sql-%d0%b7%d0%b0%d1%8f%d0%b2%d0%ba%d0%b8/feed/</wfw:commentRss>
		<slash:comments>13</slash:comments>
		</item>
		<item>
		<title>Зареждане на данни с Ajax от отдалечен сървър</title>
		<link>http://ganbox.com/blog/%d0%b7%d0%b0%d1%80%d0%b5%d0%b6%d0%b4%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d0%b4%d0%b0%d0%bd%d0%bd%d0%b8-%d1%81-ajax-%d0%be%d1%82-%d0%be%d1%82%d0%b4%d0%b0%d0%bb%d0%b5%d1%87%d0%b5%d0%bd-%d1%81%d1%8a%d1%80/</link>
		<comments>http://ganbox.com/blog/%d0%b7%d0%b0%d1%80%d0%b5%d0%b6%d0%b4%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d0%b4%d0%b0%d0%bd%d0%bd%d0%b8-%d1%81-ajax-%d0%be%d1%82-%d0%be%d1%82%d0%b4%d0%b0%d0%bb%d0%b5%d1%87%d0%b5%d0%bd-%d1%81%d1%8a%d1%80/#comments</comments>
		<pubDate>Mon, 15 Mar 2010 09:48:49 +0000</pubDate>
		<dc:creator>Георги Стефанов</dc:creator>
				<category><![CDATA[Уеб програмиране]]></category>
		<category><![CDATA[Ajax]]></category>
		<category><![CDATA[бързодействие]]></category>
		<category><![CDATA[оптимизация]]></category>
		<category><![CDATA[оптимизиране]]></category>
		<category><![CDATA[сигурност]]></category>
		<category><![CDATA[съвет]]></category>
		<category><![CDATA[урок]]></category>

		<guid isPermaLink="false">http://ganbox.com/blog/?p=737</guid>
		<description><![CDATA[Задача за оптимизиране бързината на сайт Наскоро работих по сайта Речник на думите в българския език. В този сайт при търсене по някоя дума в дясната колона се зарежда блок с подобни думи. Подобните думи са такива, които се получават от търсената дума с добавяне, премахване или подмяна на една или две букви. За откриването [...]]]></description>
			<content:encoded><![CDATA[<h3>Задача за оптимизиране бързината на сайт</h3>
<p>Наскоро работих по сайта <a href="http://rechnik.info/">Речник на думите в българския език. </a>В този сайт при търсене по някоя дума в дясната колона се зарежда блок с подобни думи. Подобните думи са такива, които се получават от търсената дума с добавяне, премахване или подмяна на една или две букви. За откриването на такива думи се използва алгоритъм с висока степен на сложност, който е бавен, защото претърсва цялата таблица с думи. Таблицата беше с размер около 100 000 реда и нарастваше и в резултат на това зареждането на страницата се бавеше до към 30 секунди &#8211; прекалено дълго време за уеб страница. Задачата ми беше да измисля начин за <a href="http://ganbox.com/seo-uslugi#оптимизиране-бързодействие">оптимизиране на бързодействието на сайта</a>.</p>
<h3>Решение</h3>
<p>Първото, което направих е да опитам да оптимизирам SQL заявката. Ползваният алгоритъм не подлежи на <strong>оптимизация</strong>, затова се наложи да прибягна до някои дребни хитрости. Например търсене на подобни думи в ограничен набор от думи с дължина +/-2 спрямо дължината на търсената дума. Създадох някои допълнителни индекси на SQL таблиците и успях да съкратя средното време на около 8 секунди. Да чакаш цялата страница 8 секунди също е прекалено досадно и затова направих блока с подобни думи да се зарежда с Ajax. По този начин страницата се показваше веднага и само блока се бавеше, което беше напълно допустимо. Вече си мислех, че съм приключил, когато дойде писмо от хостинг доставчика, че се ползва прекалено бавна заявка, която бави другите сайтове на този споделен хостинг и настояваха да се <strong>оптимизира</strong>. Така стигнах до идеята, че данните за подобните думи трябва да се зареждат от отдалечен сървър. За да стане възможно това, трябва отдалеченият сървър да има същата таблица с думи в базата си данни. Направих автоматичен скрипт, който периодично да копира таблиците от единия сървър на другия. По този начин освен всичко друго се получи и още един архив на данните. След това оставаше да направя Ajax да се обръща към новия сървър. Тук отново има проблем: от съображения за сигурност Ajax не може да се обръща към домейн различен от този, на който се изпълнява. За да заобиколя това ограничение аз използвам следната програмистка техника: Ajax се обръща към скрипта similarClient.php разположен локално на сървъра на rechnik.info, който прави заявка към скрипт с име words.php на отдалечения сървър. Там се търси за подобни думи в копието на базата данни и намерения резултат се изпраща към similarClient.php, който го връща като резултат към Ajax. Към цялата система се добавя и кеширане на заявките, локално върху отдалечения сървър &#8211; този подход спестява дисково пространство на хостинга за сметка на увеличен трафик. При първото търсене в речника по дадена дума резултата се бави няколко секунди, но при всяко следващо търсене по тази дума, резултата се взема от кеш и става почти моментално. Всичко това може да изглежда малко сложно, но можеш да видиш колко добре действа в сайта на <a href="http://rechnik.info">Речника</a>.</p>
<h3>Сигурност</h3>
<p>Скриптът разположен на отдалечения сървър, който получава дума и връща списък с подобни думи е уеб скрипт и се извиква през HTTP. Това означава, че всеки който знае URL адреса и какви параметри приема, може лесно да използва това за свои цели и да прави заявки към сървъра. Използвайки горния метод, аз скривам URL адреса и това повишава сигурността многократно. Въпреки това е възможно връзката да бъде подслушана или логовете на сървъра да станат достъпни публично. Това ще разкрие URL адреса и някой може да се възползва. Затова един съвет: никога не разчитай само на сложен и таен URL адрес на скрипт!</p>
<p>Аз използвам допълнителни механизми за защита: отдалеченият сървър приема заявки само от IP адреса на rechnik.info. Това увеличава сигурността, но за съжаление не е напълно достатъчно, защото е възможно да пристигнат заявки с фалшифициран IP адрес. Следващиата стъпка е добавяне на кодиран параметър в GET заявката. По този начин URL адреса на заявката прилича на следния:</p>
<pre>http://remote.domain.com/words.php?a=дума&amp;code=a4b25d814f32b2c852c3d71d3</pre>
<p>Низа подаван в променливата &#8222;code&#8220;, е кодирана от скрипта similarClient.php и само скрипта words.php знае как да го разкодира и да разбере дали е валиден. Резултат се връща, само ако стойността е валидна, в противен случай опита се записва и се съобщава на админ. За да може някой да хакне това, трябва да има FTP достъп до някой от двата php файла.Следващата стъпка е ползване на HTTPS вместо HTTP, но тук вече навлизаме в сферата на банковия софтуер <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_smile.gif' alt=':)' class='wp-smiley' /> </p>
<p>Сигурен съм, че много хора ще се запитат: &#8222;За какво е всичко това? Какво толкова защитаваш?&#8220;.</p>
<p>Нека да кажем, че се упражнявам <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_smile.gif' alt=':)' class='wp-smiley' />  Не обичам да оставям нещата на случайността и не правя компромиси със сигурността. Винаги изпипвам нещата и се подсигурявам двойно. Препоръчвам ти да се научиш да работиш по този начин.</p>
<h3>SEO</h3>
<p>Описаната система си има и недостатъци. От <a href="http://ganbox.com/seo">SEO</a> гледна точка връзките в блока Подобни думи са невидими за търсачките. За да може сайта да се индексира добре е нужен или Google sitemap, или допълнителен блок с връзки като този със &#8222;Случайни думи&#8220;. Още по-добра работа би свършил блок със &#8222;Съседни думи&#8220;, в който за дадена страница връзките винаги да са едни и същи.</p>
<div style="float:right;display:block" class="counterViews">Брой разглеждания на тази статия: <b>3489</b></div>

<p>Подобни статии:<ol><li><a href='http://ganbox.com/blog/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d0%b1%d1%8a%d1%80%d0%b7%d0%be%d0%b4%d0%b5%d0%b9%d1%81%d1%82%d0%b2%d0%b8%d0%b5%d1%82%d0%be/' rel='bookmark' title='Permanent Link: Оптимизиране на бързодействието на сайт'>Оптимизиране на бързодействието на сайт</a> <small>Пиша тази статия като продължение на статията PHP парсване на...</small></li>
<li><a href='http://ganbox.com/blog/%d0%be%d0%bf%d1%82%d0%b8%d0%bc%d0%b8%d0%b7%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-sql-%d0%b7%d0%b0%d1%8f%d0%b2%d0%ba%d0%b8/' rel='bookmark' title='Permanent Link: Оптимизиране на SQL заявки'>Оптимизиране на SQL заявки</a> <small>Основни положения при оптимизация на mySQL Обикновено проблемите с бавни...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://ganbox.com/blog/%d0%b7%d0%b0%d1%80%d0%b5%d0%b6%d0%b4%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d0%b4%d0%b0%d0%bd%d0%bd%d0%b8-%d1%81-ajax-%d0%be%d1%82-%d0%be%d1%82%d0%b4%d0%b0%d0%bb%d0%b5%d1%87%d0%b5%d0%bd-%d1%81%d1%8a%d1%80/feed/</wfw:commentRss>
		<slash:comments>9</slash:comments>
		</item>
		<item>
		<title>Линукс трикове</title>
		<link>http://ganbox.com/blog/%d0%bb%d0%b8%d0%bd%d1%83%d0%ba%d1%81-%d1%82%d1%80%d0%b8%d0%ba%d0%be%d0%b2%d0%b5/</link>
		<comments>http://ganbox.com/blog/%d0%bb%d0%b8%d0%bd%d1%83%d0%ba%d1%81-%d1%82%d1%80%d0%b8%d0%ba%d0%be%d0%b2%d0%b5/#comments</comments>
		<pubDate>Wed, 20 May 2009 19:48:34 +0000</pubDate>
		<dc:creator>gan</dc:creator>
				<category><![CDATA[SEO]]></category>
		<category><![CDATA[linux]]></category>
		<category><![CDATA[галерия]]></category>
		<category><![CDATA[заместване]]></category>
		<category><![CDATA[картинки]]></category>
		<category><![CDATA[команди]]></category>
		<category><![CDATA[линукс]]></category>
		<category><![CDATA[оптимизация]]></category>
		<category><![CDATA[разширение]]></category>
		<category><![CDATA[сайт]]></category>
		<category><![CDATA[трикове]]></category>
		<category><![CDATA[урок]]></category>
		<category><![CDATA[файл]]></category>
		<category><![CDATA[файлове]]></category>

		<guid isPermaLink="false">http://ganbox.com/blog/?p=88</guid>
		<description><![CDATA[Няколко скрипта за команден ред под Линукс, които ползвам често при seo оптимизация на сайт. Работят с много файлове наведнъж и спестяват много време. 1. Заместване разширението на файлове. for x in *.html; do mv &#8222;$x&#8220; &#8222;${x%.html}.php&#8220;; done - сменя разширението на всички файлове .html на .php в текущата директория 2. Заместване на една дума [...]]]></description>
			<content:encoded><![CDATA[<p>Няколко скрипта за команден ред под Линукс, които ползвам често при seo оптимизация на сайт. Работят с много файлове наведнъж и спестяват много време.</p>
<p>1. Заместване разширението на файлове.<br />
<strong>for x in *.html; do mv &#8222;$x&#8220; &#8222;${x%.html}.php&#8220;; done</strong><br />
- сменя разширението на всички файлове .html на .php в текущата директория</p>
<p>2. Заместване на една дума с друга дума в много файлове.<br />
<strong>sed -i &#8216;s/comment/коментар/g&#8217;  *.php</strong><br />
- замества всички срещания на думата &#8222;comment&#8220; с думата &#8222;коментар&#8220; във всички .php файлове на текущата директория.<br />
Внимание! Ако някъде в текста се съдържа думата &#8222;comments&#8220; тя ще стане на &#8222;коментарs&#8220;, затова е добра идея първо да се потърси с командата:<br />
<strong>grep comment *</strong><br />
- тази команда ще покаже не само в кои файлове, а и къде и как по-точно се среща търсената дума.</p>
<p>3. Създаване на малки картинки за галерия.<br />
<strong>for i in `ls *.jpg`; do  convert -resize 120&#215;120 $i ${i%.jpg}.gif; done</strong><br />
- от файлове *.jpg прави малки картинки *.gif<br />
За всеки един jpg файл от текущата директория, създава малка gif картинка, като по-дългата страна е 120 точки. Сега вече може да поставиш gif картинките в една страница и те да са връзки към по-големите jpg снимки. И не забравяй да сложиш правилните alt атрибути на тага img <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_wink.gif' alt=';)' class='wp-smiley' /><br />
Заб. Команда convert е част от библиотеката ImageMagick.</p>
<div style="float:right;display:block" class="counterViews">Брой разглеждания на тази статия: <b>2507</b><br /><span style="color:grey;font-size:0.8em">(след дата 20 декември 2009)</span></div>

<p>Подобни статии:<ol><li><a href='http://ganbox.com/blog/%d1%84%d0%bb%d0%b0%d1%88-%d0%b0%d0%bd%d0%b8%d0%bc%d0%b0%d1%86%d0%b8%d1%8f-%d0%b1%d0%b5%d0%b7-%d1%82%d0%b0%d0%b3-embed-%d0%b7%d0%b0-%d0%b2%d0%b0%d0%bb%d0%b8%d0%b4%d0%b5%d0%bd-xhtml/' rel='bookmark' title='Permanent Link: Флаш анимация без таг embed за валиден XHTML'>Флаш анимация без таг embed за валиден XHTML</a> <small>В този урок за SEO оптимизиране се показва как уеб...</small></li>
<li><a href='http://ganbox.com/blog/%d0%b3%d1%83%d0%b3%d1%8a%d0%bb-%d1%82%d1%80%d0%b8%d0%ba%d0%be%d0%b2%d0%b5-%d0%bf%d1%80%d0%b0%d0%b2%d0%b8%d0%bb%d0%bd%d0%be-%d0%b8%d0%bd%d0%b4%d0%b5%d0%ba%d1%81%d0%b8%d1%80%d0%b0%d0%bd%d0%b5-%d0%bd/' rel='bookmark' title='Permanent Link: Гугъл трикове: правилно индексиране на сайт'>Гугъл трикове: правилно индексиране на сайт</a> <small>Тази статия е за всички, които се интересуват от SEO...</small></li>
<li><a href='http://ganbox.com/blog/%d0%b7%d0%b0%d0%bc%d0%b5%d1%81%d1%82%d0%b2%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-%d1%82%d0%b0%d0%b3%d0%b0-marquee-%d0%b7%d0%b0-%d0%b2%d0%b0%d0%bb%d0%b8%d0%b4%d0%b5%d0%bd-xhtml/' rel='bookmark' title='Permanent Link: Заместване на тага marquee за валиден XHTML'>Заместване на тага marquee за валиден XHTML</a> <small>HTML тага marquee се използва за автоматично скролиране на текст....</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://ganbox.com/blog/%d0%bb%d0%b8%d0%bd%d1%83%d0%ba%d1%81-%d1%82%d1%80%d0%b8%d0%ba%d0%be%d0%b2%d0%b5/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>

<!-- Performance optimized by W3 Total Cache. Learn more: http://www.w3-edge.com/wordpress-plugins/

Minified using disk: basic
Page Caching using disk: enhanced
Database Caching 25/66 queries in 0.064 seconds using disk: basic

Served from: ganbox.com @ 2012-05-22 05:03:38 -->
