<?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/%d1%83%d1%80%d0%be%d0%ba/feed/" rel="self" type="application/rss+xml" />
	<link>http://ganbox.com/blog</link>
	<description>SEO практика: трикове при оптимизация и решаване на проблеми при уеб програмиране.</description>
	<lastBuildDate>Tue, 07 Sep 2010 11:38:22 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.0.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>gan</dc:creator>
				<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 > &#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><span style="color: #339933;">&lt;</span>count<span style="color: #009900;">&#40;</span><span style="color: #000088;">$results</span><span style="color: #009900;">&#41;</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 />
&nbsp; <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 />
&nbsp; &nbsp; <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 />
&nbsp; <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 />
&nbsp; <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 />
&nbsp; &nbsp; <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 />
&nbsp; &nbsp; <span style="color: #b1b100;">break</span><span style="color: #339933;">;</span><br />
&nbsp; <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>Знакът е > защото в класация сортираме от по-голямо към по-малко.</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;1<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>1855</b></div>]]></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>8</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>gan</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>1445</b></div>]]></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>Премахване на .html от URL адреса с mod_rewrite</title>
		<link>http://ganbox.com/blog/%d0%bf%d1%80%d0%b5%d0%bc%d0%b0%d1%85%d0%b2%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-html-%d0%be%d1%82-url-%d0%b0%d0%b4%d1%80%d0%b5%d1%81%d0%b0-%d1%81-mod_rewrite/</link>
		<comments>http://ganbox.com/blog/%d0%bf%d1%80%d0%b5%d0%bc%d0%b0%d1%85%d0%b2%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-html-%d0%be%d1%82-url-%d0%b0%d0%b4%d1%80%d0%b5%d1%81%d0%b0-%d1%81-mod_rewrite/#comments</comments>
		<pubDate>Thu, 18 Feb 2010 19:02:59 +0000</pubDate>
		<dc:creator>gan</dc:creator>
				<category><![CDATA[Уеб програмиране]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[HTML]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[заместване на URL]]></category>
		<category><![CDATA[промяна на url]]></category>
		<category><![CDATA[урок]]></category>

		<guid isPermaLink="false">http://ganbox.com/blog/?p=761</guid>
		<description><![CDATA[Тази статия е продължение на статията Приложна магия с mod_rewrite в htaccess и отговаря на въпрос зададен като коментар: &#8222;очаквам след като изпиша примерно: http://www.domain.com/za_nas.html – в полето за адреси да се вижда само http://www.domain.com/za_nas&#8222;. Тъй като решението не е много лесно, защото има подслучаи, написах отговора като отделна статия. Задача за премахване на html [...]]]></description>
			<content:encoded><![CDATA[<p>Тази статия е продължение на статията <a href="/blog/приложна-магия-с-modrewrite/">Приложна магия с mod_rewrite в htaccess</a> и отговаря на въпрос зададен като коментар: <em>&#8222;очаквам след като изпиша примерно: <a rel="nofollow" href="http://www.domain.com/za_nas.html">http://www.domain.com/za_nas.html</a> – в полето за адреси да се вижда само <a rel="nofollow" href="http://www.domain.com/za_nas">http://www.domain.com/za_nas</a>&#8222;</em>.<br />
Тъй като решението не е много лесно, защото има подслучаи, написах отговора като отделна статия.</p>
<h3>Задача за премахване на html разширение на уеб страница в адресната лента на браузъра</h3>
<p>Скриване на разширението на файла.html. Когато в браузъра се напише http://domain/page.html да се отваря html страницата, но в адреса да се замества с http://domain/page</p>
<h3>Специални изисквания за заместване</h3>
<p>Ако в една и съща директория има файл /page.html и поддиректория /page/, то потребителя трябва да получава файла, когато напише /page и директорията, когато напише /page/.</p>
<p>Ако в директорията има файл /page.html и /page.pdf (или някакъв друг различен от html), то потребителя трябва да получава /page.html, когато напише /page.</p>
<h3>Примери за поведение на пренаписващия скрипт</h3>
<p>Използва се следния формат: <span style="color: #993300;">написва</span> -&gt; <span style="color: #ff6600;">зарежда се</span> -&gt; <span style="color: #008000;">вижда в адресната лента на браузъра</span></p>
<p><span style="color: #993300;">http://domain.com/page.html</span> -&gt;<span style="color: #ff6600;"> http://domain.com/page.html</span> -&gt; <span style="color: #008000;">http://domain.com/page</span></p>
<p><span style="color: #993300;">http://domain.com/index</span> -&gt; <span style="color: #ff6600;">http://domain.com/index.html</span> -&gt; <span style="color: #008000;">http://domain.com/</span></p>
<p><span style="color: #993300;">http://domain.com/index.html</span> -&gt; <span style="color: #ff6600;">http://domain.com/index.html</span> -&gt; <span style="color: #008000;">http://domain.com/</span></p>
<p><span style="color: #993300;">http://domain.com/index.txt</span> -&gt; <span style="color: #ff6600;">http://domain.com/index.txt</span> -&gt; <span style="color: #008000;">http://domain.com/index.txt</span></p>
<p>Ако в главната уеб директория има поддиректория /doc и файл /doc.html, а в директорията doc има два файла err.html и index.html</p>
<p><span style="color: #993300;">http://domain.com/doc.html</span> -&gt; <span style="color: #ff6600;">http://domain.com/doc.html</span> -&gt; <span style="color: #008000;">http://domain.com/doc</span></p>
<p><span style="color: #993300;">http://domain.com/doc</span> -&gt; <span style="color: #ff6600;">http://domain.com/doc.html</span> -&gt; <span style="color: #008000;">http://domain.com/doc</span></p>
<p><span style="color: #993300;">http://domain.com/doc/</span> -&gt; <span style="color: #ff6600;">http://domain.com/doc/index.html</span> -&gt; <span style="color: #008000;">http://domain.com/doc/</span></p>
<p><span style="color: #993300;">http://domain.com/doc/err</span> -&gt; <span style="color: #ff6600;">http://domain.com/doc/err.html</span> -&gt; <span style="color: #008000;">http://domain.com/doc/err</span></p>
<p><span style="color: #993300;">http://domain.com/doc/err.html</span> -&gt; <span style="color: #ff6600;">http://domain.com/doc/err.html</span> -&gt; <span style="color: #008000;">http://domain.com/doc/err</span></p>
<p>Ако в поддиректорията /doc няма index.html, то при написване на http://domain.com/doc/ ще получи грешка Forbidden.</p>
<h3>Решение</h3>
<p>Следва кода за файла .htaccess любезно написан, тестван и предоставен за безплатно ползване от <a href="http://ganbox.com">ganbox.com</a></p>
<div class="codecolorer-container php default" style="overflow:auto;white-space:nowrap;border: 1px solid #9F9F9F;width:435px;height:700px;"><div class="php codecolorer" style="padding:5px;font:normal 12px/1.4em Monaco, Lucida Console, monospace;white-space:nowrap"><span style="color: #666666; font-style: italic;"># Спира MultiViews (за да може /abc да го търси като /abc.ext преди /abc/)<br />
</span>Options <span style="color: #339933;">+</span>FollowSymLinks <span style="color: #339933;">-</span>MultiViews<br />
<br />
DirectoryIndex index<span style="color: #339933;">.</span>html<br />
<br />
<span style="color: #666666; font-style: italic;"># Спира mod_dir, за да не добавя наклонена черта в края.<br />
</span>DirectorySlash Off<br />
<br />
RewriteEngine On<br />
<br />
<span style="color: #666666; font-style: italic;"># Ако търси /index или /index.html<br />
</span>RewriteCond <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>THE_REQUEST<span style="color: #009900;">&#125;</span> \ <span style="color: #339933;">/</span><span style="color: #009900;">&#40;</span><span style="color: #339933;">.+/</span><span style="color: #009900;">&#41;</span>?index<span style="color: #009900;">&#40;</span>\<span style="color: #339933;">.</span>html<span style="color: #009900;">&#41;</span>?<span style="color: #009900;">&#40;</span>\?<span style="color: #339933;">.*</span><span style="color: #009900;">&#41;</span>?\  <span style="color: #009900;">&#91;</span>NC<span style="color: #009900;">&#93;</span><br />
<span style="color: #666666; font-style: italic;"># го изтрива и оставя само /<br />
</span>RewriteRule ^<span style="color: #009900;">&#40;</span><span style="color: #339933;">.+/</span><span style="color: #009900;">&#41;</span>?index<span style="color: #009900;">&#40;</span>\<span style="color: #339933;">.</span>html<span style="color: #009900;">&#41;</span>?$ <span style="color: #339933;">/%</span>1 <span style="color: #009900;">&#91;</span>R<span style="color: #339933;">=</span><span style="color: #cc66cc;">301</span><span style="color: #339933;">,</span>L<span style="color: #009900;">&#93;</span><br />
<br />
<span style="color: #666666; font-style: italic;"># Добавя наклонена черта в края на директория, ако няма файл с .html<br />
</span><span style="color: #666666; font-style: italic;"># Ако търси директория<br />
</span>RewriteCond <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>SCRIPT_FILENAME<span style="color: #009900;">&#125;</span><span style="color: #339933;">/</span> <span style="color: #339933;">-</span>d<br />
<span style="color: #666666; font-style: italic;"># и няма html файл с такова име<br />
</span>RewriteCond <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>SCRIPT_FILENAME<span style="color: #009900;">&#125;</span><span style="color: #339933;">.</span>html <span style="color: #339933;">!-</span>f<br />
<span style="color: #666666; font-style: italic;"># и няма наклонена черта, я добавя.<br />
</span>RewriteRule <span style="color: #009900;">&#91;</span>^<span style="color: #339933;">/</span><span style="color: #009900;">&#93;</span>$ <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>REQUEST_URI<span style="color: #009900;">&#125;</span><span style="color: #339933;">/</span> <span style="color: #009900;">&#91;</span>R<span style="color: #339933;">=</span><span style="color: #cc66cc;">301</span><span style="color: #339933;">,</span>L<span style="color: #009900;">&#93;</span><br />
<br />
<span style="color: #666666; font-style: italic;"># Изтрива HTML разширения<br />
</span><span style="color: #666666; font-style: italic;"># Ако това е заявка от браузър (а не от Apache/mod_rewrite)<br />
</span>RewriteCond <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>ENV<span style="color: #339933;">:</span>REDIRECT_STATUS<span style="color: #009900;">&#125;</span> ^$<br />
<span style="color: #666666; font-style: italic;"># и заявката има HTML разширение, го изтрива.<br />
</span>RewriteRule ^<span style="color: #009900;">&#40;</span><span style="color: #339933;">.+</span><span style="color: #009900;">&#41;</span>\<span style="color: #339933;">.</span>html$ <span style="color: #339933;">/</span>$<span style="color: #cc66cc;">1</span> <span style="color: #009900;">&#91;</span>R<span style="color: #339933;">=</span><span style="color: #cc66cc;">301</span><span style="color: #339933;">,</span>L<span style="color: #009900;">&#93;</span><br />
<br />
<span style="color: #666666; font-style: italic;"># Ако в заявката има .html разширение<br />
</span>RewriteCond <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>SCRIPT_FILENAME<span style="color: #009900;">&#125;</span><span style="color: #339933;">.</span>html <span style="color: #339933;">-</span>f<br />
<span style="color: #666666; font-style: italic;"># и няма наклонена черта, добавя .html<br />
</span>RewriteRule <span style="color: #009900;">&#91;</span>^<span style="color: #339933;">/</span><span style="color: #009900;">&#93;</span>$ <span style="color: #339933;">%</span><span style="color: #009900;">&#123;</span>REQUEST_URI<span style="color: #009900;">&#125;</span><span style="color: #339933;">.</span>html <span style="color: #009900;">&#91;</span>QSA<span style="color: #339933;">,</span>L<span style="color: #009900;">&#93;</span></div></div>
<div style="float:right;display:block" class="counterViews">Брой разглеждания на тази статия: <b>1962</b></div>]]></content:encoded>
			<wfw:commentRss>http://ganbox.com/blog/%d0%bf%d1%80%d0%b5%d0%bc%d0%b0%d1%85%d0%b2%d0%b0%d0%bd%d0%b5-%d0%bd%d0%b0-html-%d0%be%d1%82-url-%d0%b0%d0%b4%d1%80%d0%b5%d1%81%d0%b0-%d1%81-mod_rewrite/feed/</wfw:commentRss>
		<slash:comments>8</slash:comments>
		</item>
		<item>
		<title>Логване в сайт с OpenID през Google</title>
		<link>http://ganbox.com/blog/%d0%bb%d0%be%d0%b3%d0%b2%d0%b0%d0%bd%d0%b5-%d0%b2-%d1%81%d0%b0%d0%b9%d1%82-%d1%81-openid-%d0%bf%d1%80%d0%b5%d0%b7-google/</link>
		<comments>http://ganbox.com/blog/%d0%bb%d0%be%d0%b3%d0%b2%d0%b0%d0%bd%d0%b5-%d0%b2-%d1%81%d0%b0%d0%b9%d1%82-%d1%81-openid-%d0%bf%d1%80%d0%b5%d0%b7-google/#comments</comments>
		<pubDate>Thu, 28 Jan 2010 13:08:17 +0000</pubDate>
		<dc:creator>gan</dc:creator>
				<category><![CDATA[Сигурност]]></category>
		<category><![CDATA[Уеб програмиране]]></category>
		<category><![CDATA[OpenID]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[доставчик на OpenID]]></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=710</guid>
		<description><![CDATA[Ограничен достъп и логване с OpenID Някой от страниците на сайта ganbox.com реших да са достъпни само след логване, затова ми се наложи да си направя логин форма, през която потребителите да се логват и да получават достъп. Обикновено една такава система включва в себе си няколко функции: страница за регистриране, страница за логване, страница [...]]]></description>
			<content:encoded><![CDATA[<h3>Ограничен достъп и логване с OpenID</h3>
<p>Някой от страниците на сайта <a href="http://ganbox.com">ganbox.com</a> реших да са достъпни само след логване, затова ми се наложи да си направя логин форма, през която потребителите да се логват и да получават достъп. Обикновено една такава система включва в себе си няколко функции: страница за регистриране, страница за логване, страница за забравена парола, страница за промяна на парола или други лични данни, управление на сесии, база данни с данни за потребителите. Мислих няколко варианта и накрая реших, че без последните две няма как да мина, но нямам време да правя другите страници. Затова избрах решение с OpenID логване.</p>
<h3>Какво е OpenID?</h3>
<p>Съвсем накратко: <strong>с една парола достъп до много сайтове</strong>. Това е метод, който обединява протокол и системи, който ти дава възможност да се регистрираш в един сайт и след това да получаваш достъп до много други сайтове, които поддържат OpenID, само като се логваш в първия сайт.</p>
<h3>Как работи OpenID</h3>
<p>Регистрираш се в един сайт, който е <strong>доставчик на OpenID</strong> и там получаваш данни за достъп до акаунта, които обикновено са комбинация от имейл и парола или потребителско име и парола. От този момент можеш да се логваш във всеки един сайт, който поддържа този доставчик на OpenID.</p>
<p>Някои от сайтовете, като например сайта за споделяне на връзки dao.bg (а и много други подобни сайтове), изискват първо да си създадеш профил в сайта и едва след това може да присъединиш OpenID идентификатор в профила си, за да може да се логваш с OpenID.</p>
<p>Трик за сигурност: След като въведеш OpenID е препоръчително да смениш паролата си в този сайт с някаква невъзможно сложна и дълга парола.Тъй като няма повече да ползваш тази парола, от съображения за <strong>сигурност</strong> е по-добре да не може да бъде налучкана от хакери.</p>
<h3>Предимства на OpenID</h3>
<h4>1. Без измисляне на нови пароли. </h4>
<p>Не е нужно да помниш много пароли, а само една и по-рядко ще ти се случва да забравиш паролата си.</p>
<h4>2. Бърза регистрация. </h4>
<p>В повечето сайтове, в които искаш да влезеш няма да ти се налага да се регистрираш или ако има такава стъпка доста от данните ще бъдат попълнени, защото се изпращат от сайта доставчик на OpenID.</p>
<h4>3. На уеб програмистите спестява време.</h4>
<p>Отпада нуждата от програмиране на няколко уеб страници.</p>
<h4>4. По-висока степен на сигурност.</h4>
<p>Виж по-долу в параграфа Сигурност.</p>
<h3>Недостатъци на OpenID</h3>
<p>Хора които са слабо запознати с интернет технологиите и с OpenID могат да се уплашат, че за вход в твоя сайт трябва да въведат паролата си за Гугъл. Това се случи в ganbox.com, затова в логин страницата поставих блок Информация.</p>
<h3>Сигурност</h3>
<p>Всъщност метода на логване през OpenID е с <strong>висока степен на сигурност</strong>, защото въвеждаш паролата си в страница на Гугъл защитена с SSL и това, което Google изпраща към сайта, в който се логваш се контролира от теб. В случая на ganbox.com това е единствено имейл адреса, като при първото изпращане ти получаваш предупреждение за това и се съгласяваш с това действие.</p>
<p>Трябва да гледаш с добро око на сайтовете, които предлагат OpenID логин. Повечето хора използват една и съща парола за няколко сайта. Ако и ти правиш така, то при използване на OpenID избягваш следния риск (макар и минимален).  При регистриране в нов сайт в момента на въвеждане на паролата е възможно някой от администраторите да получи тази парола в чист текст. След това той може да злоупотреби и ще има достъп до всичките сайтове, в които използваш тази парола. При OpenID нямаш този проблем, защото в новия сайт не се изпраща парола.</p>
<h3>Логване с OpenID в страница на PHP</h3>
<p>В ganbox.com за начало като OpenID доставчик избрах Google, защото почти всеки има поща в gmail.com или ползва друга услуга на Google и има Google акаунт. В последствие може да се добавят още доставчици, като например Facebook и човек ще може да си избира с кой от двата да се логне.</p>
<p>В този урок ще покажа как да си направиш логване в твоя сайт през акаунт на <strong>Гугъл</strong>.</p>
<p>1. Изтегли си библиотеката <a href="http://ganbox.com/blog/wp-content/pub/class.openid.php_.zip">class.openid.php</a>, разархивирай файла и го постави в директория inc, която трябва да е в главната уеб директория.</p>
<p>2. В главната уеб директория създай два файла login.php и return.php</p>
<p>Основния код в login.php, който стартира процес по логване е:</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: #b1b100;">require_once</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;/inc/class.openid.php&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
<span style="color: #b1b100;">if</span><span style="color: #009900;">&#40;</span><span style="color: #339933;">!</span>user_logged<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#123;</span><br />
&nbsp; &nbsp; GoogleOpenId<span style="color: #339933;">::</span><span style="color: #004000;">login</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;http://ganbox.com&quot;</span><span style="color: #339933;">,</span><span style="color: #0000ff;">&quot;http://ganbox.com/return.php&quot;</span><span style="color: #339933;">,</span> <span style="color: #009900; font-weight: bold;">NULL</span><span style="color: #339933;">,</span> <span style="color: #009900; font-weight: bold;">true</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span><span style="color: #b1b100;">else</span><span style="color: #009900;">&#123;</span><br />
&nbsp; &nbsp; <a href="http://www.php.net/header"><span style="color: #990000;">header</span></a><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'Location: /'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>exit<span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span></div></div>
<p>Функция user_logged() проверява дали потребителя се е логнал.<br />
Последния параметър true на GoogleOpenId::login указва дали твоя сайт ще изиска имейл адрес.<br />
Този ред ще пренасочи браузъра към страница за логване в Google профил (ако вече не си логнат в Гугъл), след като се логнеш първият път ще ти се изведе съобщение, че сайта изисква имейл и трябва да се съгласиш да предоставиш тази информация, след което ще те редиректнат отново към http://ganbox.com/return.php</p>
<p>В return.php основния код е:</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: #b1b100;">require_once</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;/inc/class.openid.php&quot;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
<span style="color: #000088;">$response</span> <span style="color: #339933;">=</span> GoogleOpenId<span style="color: #339933;">::</span><span style="color: #004000;">getResponse</span><span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
<br />
<span style="color: #b1b100;">if</span><span style="color: #009900;">&#40;</span> <span style="color: #0000ff;">'id_res'</span> <span style="color: #339933;">==</span> <span style="color: #000088;">$response</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">getMode</span><span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#41;</span><span style="color: #009900;">&#123;</span> &nbsp; &nbsp;<span style="color: #666666; font-style: italic;"># успешно логване<br />
</span>&nbsp; &nbsp; <span style="color: #000088;">$email</span> <span style="color: #339933;">=</span> <span style="color: #000088;">$response</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">getEmail</span><span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span> <span style="color: #666666; font-style: italic;"># имейл<br />
</span>&nbsp; &nbsp; user_login<span style="color: #009900;">&#40;</span><span style="color: #000088;">$email</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
&nbsp; &nbsp; <a href="http://www.php.net/header"><span style="color: #990000;">header</span></a><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'Location:/'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>exit<span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span><span style="color: #b1b100;">else</span><span style="color: #009900;">&#123;</span><br />
&nbsp; &nbsp; <a href="http://www.php.net/die"><span style="color: #990000;">die</span></a><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'Неуспешен вход'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span></div></div>
<p>В този код функцията user_login() се грижи да създаде сесия и да отбележи, че потребителя е логнат. Ако до момента не е имало потребител с подадения имейл, автоматично го регистрира и логва.<br />
Това е всичко по оторизирането на потребителя &#8211; просто и бързо.</p>
<p>След това във всяка страница, в което искам да огранича достъпа викам функция  user_logged(), която проверява дали потребителя е логнат и ако не е логнат или ако сесията е изтекла се пренасочва към страница за логване.</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: #b1b100;">if</span><span style="color: #009900;">&#40;</span><span style="color: #339933;">!</span>user_logged<span style="color: #009900;">&#40;</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#41;</span><span style="color: #009900;">&#123;</span><br />
&nbsp; &nbsp; <a href="http://www.php.net/header"><span style="color: #990000;">header</span></a><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'Location: /login.php'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>exit<span style="color: #339933;">;</span><br />
<span style="color: #009900;">&#125;</span></div></div>
<p>Работещ пример има в сайта ganbox.com, като с ограничен достъп са страниците <a href="/pos">Класиране на сайт</a>, <a href="/seo-status">SEO състояние на сайт</a> и <a href="/resize">Промяна на снимка</a>.<br />
Това е накратко. Ако има интерес и имам време мога да дам пълните кодове. Пишете коментари! <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_smile.gif' alt=':)' class='wp-smiley' /> </p>
<div style="float:right;display:block" class="counterViews">Брой разглеждания на тази статия: <b>1717</b></div>]]></content:encoded>
			<wfw:commentRss>http://ganbox.com/blog/%d0%bb%d0%be%d0%b3%d0%b2%d0%b0%d0%bd%d0%b5-%d0%b2-%d1%81%d0%b0%d0%b9%d1%82-%d1%81-openid-%d0%bf%d1%80%d0%b5%d0%b7-google/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>PHP strlen() и текстове на кирилица</title>
		<link>http://ganbox.com/blog/php-strlen-%d0%b8-%d1%82%d0%b5%d0%ba%d1%81%d1%82%d0%be%d0%b2%d0%b5-%d0%bd%d0%b0-%d0%ba%d0%b8%d1%80%d0%b8%d0%bb%d0%b8%d1%86%d0%b0/</link>
		<comments>http://ganbox.com/blog/php-strlen-%d0%b8-%d1%82%d0%b5%d0%ba%d1%81%d1%82%d0%be%d0%b2%d0%b5-%d0%bd%d0%b0-%d0%ba%d0%b8%d1%80%d0%b8%d0%bb%d0%b8%d1%86%d0%b0/#comments</comments>
		<pubDate>Wed, 23 Dec 2009 18:27:14 +0000</pubDate>
		<dc:creator>gan</dc:creator>
				<category><![CDATA[SEO]]></category>
		<category><![CDATA[Уеб програмиране]]></category>
		<category><![CDATA[mbstring]]></category>
		<category><![CDATA[mb_strlen]]></category>
		<category><![CDATA[mb_substr]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[strlen]]></category>
		<category><![CDATA[substr]]></category>
		<category><![CDATA[XHTML]]></category>
		<category><![CDATA[съвет]]></category>
		<category><![CDATA[уеб форма]]></category>
		<category><![CDATA[урок]]></category>

		<guid isPermaLink="false">http://ganbox.com/blog/?p=676</guid>
		<description><![CDATA[Проблем Вече в няколко сайта уеб директории, забелязвам следния проблем: в уеб формата за качване на сайт полетата имат ограничение в броя на символите, което е нормално, но е проблем, когато пише, че може да въведеш 100 символа, а реално можеш да въведеш само 50. Още по-досадно и времеотнемащо е, ако софтуера създаващ уеб формата [...]]]></description>
			<content:encoded><![CDATA[<h3>Проблем</h3>
<p>Вече в няколко сайта уеб директории, забелязвам следния проблем: в уеб формата за качване на сайт полетата имат ограничение в броя на символите, което е нормално, но е проблем, когато пише, че може да въведеш 100 символа, а реално можеш да въведеш само 50. Още по-досадно и времеотнемащо е, ако софтуера създаващ уеб формата не се грижи да запомни всички въведени полета и при грешка в едно от полетата изтрива всички полета &#8211; за форми с много полета това е направо кошмар, защото трябва да въвеждаш всичко отначало.</p>
<p>Най-фрапиращ случай за това е сайта bestbgsites.com. Не е лесно да добавиш линк в този сайт. Гърми грешката: &#8222;Името на сайта е прекалено дълго! Може да бъде 85 символа&#8220; проблема е, че ако заглавието е на кирилица дължината е до 42 символа  Описанието пише, че може да бъде до 255 символа, но реално са 127.  Има и кратко описание до 30 символа (реално 15), което е трудно за измисляне. Ако сбъркаш някое от полетата всичките въведени данни се изтриват и трябва да се попълват повторно <img src='http://ganbox.com/blog/wp-includes/images/smilies/icon_sad.gif' alt=':(' class='wp-smiley' /> </p>
<p>Същия проблем има и сайта bglogs.com, като тук формата запомня полетата, но пък има ограничение в броя на грешните опити, след което пак трябва да започнеш на чисто. Полетата Описание и META Description пише, че имат лимит 500 символа и дори има Javascript, който намалява брояча на символите при всяко натискане на клавиш, когато въвеждаш в полето. Проблемът идва, когато изпратиш данните към сървъра, той ги проверява и ти показва, че си превишил доста лимита. В полето Заглавие е още по-трудно, защото лимита е замислен да бъде 100 символа, но на кирилица са само 50.</p>
<h3>Обяснение</h3>
<p>Тези проблеми се получават, защото представянето на всеки символ на кирилица при UTF8 е в 2 байта. Сайтовете ползват готов безплатен софтуер писан от англо-говорящи, които са свикнали при проверка държина на низ да използват PHP функцията strlen(),  която работи на ниво байтове и затова тази функция преброява низ на кирилица с дължина 50 символа, като низ с дължина 100 символа.</p>
<h3>Решение</h3>
<p>Решението е много лесно, но трябва да бъде извършено от собственика на сайта. Трябва функцията strlen() да се замени с функцията mb_strlen(), която се съобразява с енкодинга и правилно преброява низ независимо на какъв език е написан. Тази функция се осигурява от библиотеката mbstring (Multibyte String), която за щастие отдавна се поддържа от всички български хостинг доставчици.</p>
<h3>Отрязване на низ на кирилица</h3>
<p>Подобен проблем се получава, ако отрязваш низ на кирилица в енкодинг UTF8 например показване на извадка от новина. Ако искаш да покажеш първите 200 символа от новина и направиш</p>
<p>$text = substr($text, 0, 200);</p>
<p>като резултат ще имаш първите 100 символа, като е възможно последния символ да бъде &#8222;срязан по средата&#8220;, в смисъл, че от двата байта, с които е представен ще остане само първия и в резултат на това в страницата ще бъде показан странен символ, който е невалиден UTF8 символ и чупи страницата. В резултат на това страницата няма да може да бъде валидирана, като XHTML документ. Нямам лични наблюдения как се отразява това на <a title="seo оптимизация на бизнес сайт" href="/seo" target="_blank">SEO</a>, но предполагам, че е проблем за ботовете на много от търсачките.</p>
<p>За да решиш този проблем трябва просто да замениш функцията substr() с функцията mb_substr(), като разбира се библиотеката mbstring трябва да е предварително инсталирана.</p>
<h3>Заключение</h3>
<p>Изрично искам да отбележа, че нямам нищо против споменатите два сайта и целта на статията ми не е да ги злепостави, а да помогне на тях и на всички, които имат сайт, който ползва готов софтуер, в който не е помислено, че ще се въвежда нещо различно от латински символи.</p>
<div style="float:right;display:block" class="counterViews">Брой разглеждания на тази статия: <b>1587</b></div>]]></content:encoded>
			<wfw:commentRss>http://ganbox.com/blog/php-strlen-%d0%b8-%d1%82%d0%b5%d0%ba%d1%81%d1%82%d0%be%d0%b2%d0%b5-%d0%bd%d0%b0-%d0%ba%d0%b8%d1%80%d0%b8%d0%bb%d0%b8%d1%86%d0%b0/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
		</item>
	</channel>
</rss>
