4Debian.Info - Портал русскогоязычного сообщества Debian Карта сайта Контакты

Меню

Репозитории Debian

Скачать Debian

Установка Debian - Видео



Авторизация:

Новые статьи

Статьи в RSS

Новые комментарии

Новые уроки

Уроки в RSS


Tcl
Tcl.svg
Семантика:

императивный, скриптовый

Тип исполнения:

интерпретатор

Появился в:

1988

Автор(ы):

Джон Оустерхаут

Релиз:

8.5.11 / 07 ноября 2011

Тестовая версия:

8.6b2 / 5 августа 2011

Типизация данных:

динамическая

Испытал влияние:

Shell, Lisp

Повлиял на:

Python

Tcl (от англ. Tool Command Language — «командный язык инструментов», читается «тикль» или «ти-си-эль») — скриптовый язык высокого уровня.

Tcl часто применяется совместно с графической библиотекой Tk (Tool Kit). Связку Tcl/Tk по-русски иногда называют «Так-тикль» (английский вариант — «тикль-ток»).

Области применения языка — быстрое прототипирование, создание графических интерфейсов для консольных программ (пакетов программ), встраивание в прикладные программы, тестирование. Также Tcl применяется в веб-разработке.

Tcl, наряду с Perl и Python, стал одним из трёх классических скриптовых языков общего назначения. Эта троица фигурирует не только в качестве списка свободных дистрибутивов, собираемых в ActiveState, но и, например, как языки, на которых (помимо диалекта PL/pgSQL) можно писать триггеры и хранимые процедуры популярного сервера БД PostgreSQL.

Tcl, как встраиваемый язык, нашел применение и в сфере САПР (CAD|CAM|CAE). Он, к примеру, используется как средство настройки баз данных, в постпроцессоре Unigraphics. Кроме того, он де-факто является языком автоматизации и интеграции во всех ведущих программных пакетах разработки микросхем (ПЛИС и ASIC).

Содержание
Особенности

В Tcl данными всех типов, включая код программы, можно манипулировать как строками. Это делает его языком с естественной поддержкой метапрограммирования. Эта парадигма программирования учитывалась в ходе разработки и эволюции языка.

Программа на Tcl состоит из команд, разделённых символами новой строки или точками с запятой. Каждая команда состоит из набора полей, разделённых пробелами. Первое поле должно быть именем команды, а необязательные остальные поля — передаваемые этой команде аргументы. Команда возвращает значение, иногда пустое. То есть, как и в Лиспе, в Tcl используется префиксная нотация.

Ключевых слов как таковых нет — понятие команды в Tcl аналогично понятию процедуры или функции распространённых языков программирования. Это относится и к управляющим конструкциям языка. В сочетании с элементарным синтаксисом это обеспечивает хорошую расширяемость языка, в том числе и библиотеками, написанными на других языках, таких как C/C++ или Java.

В Tcl также качественно реализована модель управления программой на основе событий. События могут генерироваться таймером, при появлении данных в канале, изменении значения переменной, при завершении какой-либо внешней программы, или просто при работе пользователя с интерфейсом Tk. Можно задавать свои события и управлять ими.

Как и большинство современных скриптовых языков Tcl содержит развитые средства работы с регулярными выражениями, работает с ассоциативными массивами и другими динамическими структурами данных.

В базовом Tcl нет поддержки ООП, однако существует множество расширений Tcl объектно-ориентированными механизмами, реализованных в виде подключаемых библиотек на Си, или самом Tcl. Вот самые распространённые из них:

расширение реализовано на особенности используемой модели
XOTcl C мощная объектная система с динамическим определением классов и методов, близкая к CLOS
incr Tcl C ООП в стиле, близком к C++, расширенном введением конфигураторов
Snit Tcl реализация объектной модели на основе делегирования, а не наследования
STOOOP Tcl ООП в стиле, максимально близком к C++, также введён механизм переключения классов

Пространства имён поддерживаются на уровне ядра языка.

В настоящее время бета-тестирование проходит новый ООП-фреймвок TclOO, который планируется включить в будущие версии ядра языка. Это компактная, но мощная объектно-ориентированная система, которую можно использовать как непосредственно, так и как базу для построения расширенных моделей ООП.

На Tcl легко программировать в функциональном стиле: он обеспечивает такие механизмы, как функции высшего порядка и абстракции функций, хотя Tcl редко используют таким образом. Вот пример того, как можно осуществить композицию двух функций:

bash:
  1. proc o {f g x} {$f [$g $x]}

В версии 8.5 в Tcl также введена команда apply, значительно облегчающая программирование в этом стиле.

Вот пример её использования из официальной документации:

bash:
  1. proc map {lambda list} {
  2. set result {}
  3. foreach item $list {
  4. lappend result [apply $lambda $item]
  5. }
  6. return $result
  7. }
  8.  
  9. map {x {return [string length $x]:$x}} {a bb ccc dddd}

результатом работы этого примера будет

1:a 2:bb 3:ccc 4:dddd
История

Язык разработан Джоном Оустерхаутом (John Ousterhout) в то время, когда он работал в университете Калифорнии в Беркли. Первоначальной целью было создание универсального языка для встраивания в консольные приложения. Вслед за этим Оустерхаут расширил свой язык средствами создания графических интерфейсов — так появился Tcl/Tk.

За 4 года развития в университетских стенах Тикль-сообщество выросло до нескольких сотен тысяч программистов. В 1993 году было разработано первое объектно-ориентированное расширение — incr Tcl.

В 1994 г. Оустерхаут становится ведущим инженером проекта SunScript. В течение следующих 4 лет, став одним из проектов корпорации Sun, Тикль не только остался свободно распространяемой системой, но и значительно усовершенствовался, стал кросс-платформенным.

Заметный удар по популярности языка нанёс Ричард Столлман, опубликовав в сентябре 1994 г. в ряде новостных групп рекомендацию «Почему вы не должны использовать Tcl» . В ней он на основании утверждений, что «Tcl разработан не как серьёзный язык программирования» и «Tcl имеет особый синтаксис, притягательный для хакеров из-за своей простоты. Но синтаксис Tcl кажется странным для большинства пользователей.», а также в связи с продвижением Emacs Lisp вынес вердикт: «проект GNU не собирается использовать Tcl в программном обеспечении GNU». Помимо субъективных претензий к дизайну языка, Столлман указал на объективный на тот момент недостаток — низкое быстродействие.

Тем не менее, рост количества разработчиков, активно использующих тикль, составил за этот период 600 %.

В 1998 г. в связи с Java-бумом и связанной с ним острой конкурентной борьбой корпорация уже не могла уделять тиклю прежнего внимания. Джон Остераут покидает Sun и создаёт (при поддержке Sun) компанию Scriptics, продолжая развитие технологии Tcl/Tk. Это была одна из первых коммерческих программных фирм, чья деятельность строилась вокруг бесплатного свободного ПО.

Цветовое решение логотипа, принятое в 90-х гг. XX века

В марте 1998 года Оустерхаут пишет ставшую классической статью «Сценарии: высокоуровневое программирование для XXI века», в которой назвал скриптовые языки языками системной интеграции, потому что они ориентированы прежде всего на работу с объектами операционной среды, а не с элементарными данными. Это позволяет без написания тысяч строк кода двумя-тремя командами «склеивать» такие объекты в одно целое.

В том же году он выиграл за разработку языка Tcl две крупных награды. Первая — это ACM Software System Award, вручаемая за «наиболее значимое программное обеспечение». Этой наградой до этого были отмечены такие основополагающие системы, как стек протоколов TCP/IP, первая электронная таблица, первая реляционная база данных, World Wide Web, Unix, PostScript и Smalltalk. Вторая — это ежегодная USENIX Software Tools User Group (STUG) Award, в знак признания выдающегося программного обеспечения.

В 1999 году выходит Tcl 8.0 — в этой реализация впервые появилась компиляция в байткод, что позволило в 6 раз поднять быстродействие. В следующую модификация Tcl 8.1 введена полноценная поддержка юникода, в ней впервые реализована многопоточность. В связи c этим релизом Остераут заявил:

« «В целом 8.1 переводит Tcl в новую категорию, поскольку теперь его можно использовать для серверных приложений. Си, Си++ и Java можно по-прежнему использовать для создания частей приложений, критических с точки зрения производительности, а Tcl — для интеграции и создания интерфейсных компонентов». »

В 2000 году фирма Scriptics была переименована в Ajuba Solutions, а затем куплена фирмой Interwoven, не заинтересованной в работе со свободным ПО. В связи с этим разработка Tcl была передана сообществу программистов. Так была сформирована Tcl Core Team.

На 9-ой конференции по Tcl/Tk, проходившей в Ванкувере, была представлена виртуальная файловая система StarKit, обеспечившая новый, удобный для обычного пользователя способ распространения Tcl программ.

Синтаксис Tcl
Скрипт 
команды, разделённые переводом строки или точкой с запятой (;).
Команда 
слова, разделённые пробелами.
bash:
  1. commandName argument1 argument2 … argumentN

Первое слово — имя команды, остальные — её аргументы. Любой аргумент может быть заменён другой командой, помещённой в квадратные скобки. Любые аргументы в фигурных скобках передаются команде «как есть», в виде единого аргумента.

Символы, имеющие особое значение
  • $  — вызывает подстановку значения переменной.
  • [] — вызывает подстановку результата выполнения команды внутри скобок.
  • "" — группирует аргументы в один с подстановкой значений переменных.
  • {} — группирует аргументы в один без подстановки значений переменных.
  • \  — экранирует следующий символ или вызывает подстановку управляющего символа.
  • #  — начало однострочного комментария, должен быть после неэкранированного символа перевода строки или «;» (точки с запятой).
Замечания о процессе разборки скрипта
  • Разбивка на слова, подстановка значений переменных и результатов выполнения команд производятся за один проход по команде.
  • Все символы, находящееся внутри скобок {} и кавычек "", рассматривается как одно слово.
  • Первое слово команды рассматривается как встроенная или определённая процедура.
Примеры команд
  • Присваивание обозначается словом set, а не знаком равенства.

Синтаксис:

bash:
  1. set переменная значение

Пример:

bash:
  1. set a 2
bash:
  1. set b 3
  • Комментарий обозначается символом #

Синтаксис:

bash:
  1. # текст комментария

Пример:

bash:
  1. # это комментарий
  • Вывод (печать) реализуется командой puts

Синтаксис:

bash:
  1. puts выражение

Пример:

bash:
  1. puts "Привет! Это я."
  2. puts 123
  3. puts $a
  4. puts "b = $b"
  • Цикл пока реализуется командой while, имеющей два аргумента. Первый — Tcl выражение (выражения фактически пишутся на миниатюрном языке, состоящем из математических операций и операций сравнения в стиле языка C. Второй аргумент — это скрипт, выполняемый при каждой итерации. Его, как правило, помещают в фигурные скобки, чтобы избежать немедленного выполнения.

Синтаксис:

bash:
  1. while { Выражение_Tcl_С_Логическим_Значением } {
  2. TclКоманда
  3. другаяTclКоманда
  4. … …
  5. }

Пример:

bash:
  1. while { $x < 10 } {
  2. puts $x
  3. incr x
  4. }
  • Условная команда

Синтаксис:

bash:
  1. if { Выражение_Tcl_С_Логическим_Значением } {
  2. КомандаTcl
  3. }

Пример:

bash:
  1. if {$x < 0} {
  2. set x 0
  3. }
  • Команда ветвления

Пример:

bash:
  1. switch -glob — $var {
  2. {see} -
  3. {saw} -
  4. {seen} { puts "Все это значит одно и то же!"}
  5. }
  • Команда может не иметь аргументов, например
bash:
  1. pwd

просто возвращает текущий рабочий каталог. А командой

bash:
  1. set wdir [pwd]

вы сохраните имя текущего каталога в переменной wdir.

  • Команда может возвращать список, например
bash:
  1. glob Образец

возвратит список имён файлов текущей директории, удовлетворяющих шаблону Образец.

Процедуры

Процедуры определяются следующим образом

bash:
  1. proc имяПроцедуры { списокАргументов } {
  2. … …
  3. }
Ассоциативные массивы

Следующий фрагмент кода создаст и инициализирует ассоциативный массив (называемый в других языках также карта, словарь, или хеш).

bash:
  1. set capital(France) Paris
  2. set capital(Italy) Rome
  3. set capital(Germany) Berlin
  4. set capital(Poland) Warsaw
  5. set capital(Russia) Moscow
  6. set capital(Spain) Madrid

Для запроса и вывода одного из значений ассоциативного массива используется

bash:
  1. puts $capital(Italy)

Список всех стран (ключей массива) можно получить командой

bash:
  1. array names capital

Результат не будет отсортирован, потому что массивы Tcl базируются на хеш-таблицах.

bash:
  1. Poland Spain Russia Germany Italy France

Сортировку можно выполнить командой

bash:
  1. lsort [array names capital]

Чтобы сослаться на массив (передать его как ссылку) используется имя содержащей его переменной:

bash:
  1. proc demo arrName {
  2. upvar 1 $arrName arr ;# tie the reference to a local name
  3. #do something with arr
  4. }

Для доступа к массиву по значению, используется array get или array set. Например, чтобы скопировать один массив в другой:

bash:
  1. array set Newarr [array get Oldarr]

В Tcl 8.5 введены массивы, передаваемые по значению — словари (dict).

Для преобразования скалярной переменной в массив и наоборот следует удалить существующую переменную при помощи команды unset.

Область видимости переменных

По умолчанию внутри процедуры видны только определённые в ней переменные. За пределами определения процедуры её переменные недоступны, иначе чем через механизм передачи параметров. Вы можете иметь переменные в процедуре с таким же именем, как и в основной программе, при этом значения этих переменных будут различаться, по сути это будут разные переменные. Чтобы изменить это поведение используются команды global или upvar.

Примеры

Дополнительные примеры с комментариями на русском вы можете найти на wiki-ресурсе Нижегородской Группы Пользователей Linux в разделе Programming Gems::Tcl/Tk.

Здравствуй, мир!

Программа Hello, world выглядит так

bash:
  1. puts "Здравствуй, мир!"

или так

bash:
  1. puts {Здравствуй, мир!}
Массивы Обращение к элементам массива

Вывод значения элемента массива OldArray(precision), имя которого (массива) задается новой переменной NewArray

bash:
  1. set NewArray OldArray
  2. puts [set ${NewArray}(precision)]
Сложение чисел массива

Метод (A) — Сложение с использованием цикла 'foreach'

bash:
  1. set numbers {1 2 3 4 5 6 7 8 9 10}
  2. set result 0
  3. foreach number $numbers {
  4. set result [expr {$result + $number}]
  5. }
  6. puts $result

Метод (B) — Гораздо более элегантный способ с использованием команды 'join'

bash:
  1. set numbers {1 2 3 4 5 6 7 8 9 10}
  2. puts [expr [join $numbers +]]
Список содержимого массива

В массиве tcl_platform, содержится информация о свойствах платформы, на которой запущен Tcl. Список имён свойств может быть получен с помощью

bash:
  1. array names tcl_platform

Следующий фрагмент выводит их вместе со значениями:

bash:
  1. foreach i [array names tcl_platform] {
  2. puts "$i = $tcl_platform($i)"
  3. }

Если свойства должны быть отсортированы

bash:
  1. foreach i [lsort [array names tcl_platform] ] {
  2. puts "$i = $tcl_platform($i)"
  3. }

Здесь демонстрируется вложенность команд. Они могут быть вложены на любую глубину. Подобного же результата можно достичь с помощью команды parray (появившейся в Tcl 8.5).

Пересечение множеств

Процедура filter возвращает те элементы списка, для которых script принимает значение TRUE:

bash:
  1. proc filter {list script} {
  2. set res {}
  3. foreach e $list {if {[uplevel 1 $script $e]} {lappend res $e}}
  4. return $res
  5. }

Команда uplevel 1 позволяет вычислить выражение в контексте, в котором вызывается слово filter.

Процедура in — это сокращённая запись для включения (инклюзии) множеств:

bash:
  1. proc in {list e} {
  2. expr {[lsearch -exact $list $e]>=0}
  3. }

Проверка:

bash:
  1. % filter {a b c} {in {b c d}}
  2. b c
Факториал
bash:
  1. proc ! x {expr {$x<2? 1: $x*[! [incr x -1]]}}

Здесь демонстрируется, что именем команды могут быть любые строки, а также известный по другим языкам оператор ?:.

Реализация, дистрибутивы, инструменты

Tcl — интерпретируемый язык, то есть программы на Tcl готовы к выполнению без компиляции и компоновки.

Интерпретатор Tcl портирован на большинство распространённых платформ. Он распространяется под свободной некопилефтной лицензией, позволяющей использовать его без ограничения в разработке проприетарных приложений, а также создавать проприетарные производные системы. Разработчики относят её к BSD подобным, хотя по сравнению с лицензией BSD лицензия Tcl накладывает меньшее количество запретов.

Реализация

В настоящее время работа над ядром языка Tcl ведётся группой разработчиков именуемой Tcl Core Team. Самые свежие версии Tcl-транслятора и основных библиотек доступны в репозитории на SourceForge.net, см. список.

Транслятор Tcl компилирует программу в промежуточное представление — байткод, а затем интерпретирует её. В отличие от, например Python — это промежуточное представление не сохраняется на диск, программа может хранится только в исходных текстах, возможно упакованных в StarKit модуль или с помощью какого-нибудь другого враппера. В зависимости от параметров компиляции интерпретатор Tcl может поддерживать или не поддерживать многопоточность.

Хотя в большинстве учебников по Tcl для начинающих говорится что-то вроде «в Tcl есть всего один тип переменных — строка», на самом деле трансляторы Tcl уже давно как устроены по-другому. Внутри Tcl использует систему динамических типов с автоматическим, прозрачным преобразованием к строковому виду.

Tcl входит практически во все дистрибутивы Linux. Для Microsoft Windows требуется отдельный установочный пакет.

В настоящее время текущая стабильная версия Tcl 8.4, её основные отличия от предыдущей — новые функции (оптимизация 64-битного кода, VFS, дополнения в плане интроспекции) и некоторое повышение быстродействия за счёт оптимизации работы с байткодом.

Новые возможности, добавленные в версии 8.1 (в частности, поддержка многопотокового выполнения, средства интернационализации и работа с символами Unicode), замедлили работу Tcl по сравнению с давшей скачок скорости восьмой версией примерно на 19 %. Разработчики 8.4 приложили все усилия для того, чтобы максимально повысить скорость выполнения программ, добиться, чтобы они работали не только не медленнее, но даже быстрее, чем это было при появлении версии 8.0. Эта цель была достигнута практически для всех средств Tcl и Tk — версия 8.4 в среднем в 8 раз быстрее чем версия 7.6. Текущая версия TCL — 8.5.

Альтернативные реализации и диалекты

Существует альтернативные реализации Tcl для популярных виртуальных машин общего назначения. Это написанный на Java интерпретатор — Jacl (последняя версия от 2008 года), и проект Eagle, реализующий большинство команд Tcl 8.4 для CLR (Доступны сборки для .NET 4.0 и 2.0). Ведутся эксперименты по трансляции Tcl программ в код виртуальной машины Parrot, разрабатываемой для шестой версии Perl.

Для программирования встраиваемых систем и встраивания в приложения разработан специальный диалект Tcl Jim. Это интерпретатор реализующий большое подмножество Tcl, расширенное дополнительными механизмами объектно-ориентированного и функционального программирования и занимающий в скомпилированном виде менее 100 килобайт.

Ещё один диалект Tcl — скриптовый язык мобильных устройств Hecl, реализованный на Java (J2ME). Что же касается платформы Android, поддержку Tcl на ней (наряду с другими скриптовыми языками) обеспечивает скриптовая прослойка SL4A (англ.).

Использование в веб-разработке

В web-программировании Tcl обычно используется вместе с веб-сервером AOLServer или одним из «облегчённых» веб-серверов, вроде lighttpd, ориентированным на FastCGI. Для интеграции Tcl с Apache существует модуль Rivet.

Существуют также такие проекты, как TclHttpd и его преемник — Wub, являющиеся полноценными Web-фреймворками. На базе нового объектного расширения языка TclOO разрабатывается веб-фреймвок Woof (Web Oriented Object Framework).

Помимо использования в качестве языка реализации веб-приложений, TCL может использоваться как скриптовый язык для расширения существующих приложений. К примеру на нём пишут модули для IRC бота Eggdrop.

Дистрибутивы

Все нижеперечисленные дистрибутивы распространяются свободно и собраны из компонентов, имеющих ту или иную свободную лицензию. Большинство из них — кроссплатформены (кроме WinTclTk и Tcl/Tk Aqua), хотя в Unix-системах Tcl/Tk обычно является частью дистрибутива ОС.

  • ActiveTcl
    Самый распространённый дистрибутив TCL. Подобно другим продуктам компании ActiveState включает в себя большой набор библиотек и расширений. В windows-версиях включает в себя средства интеграции с WSH. Какие либо инструментальные средства, облегчающие жизнь программиста, отсутствуют. Фирма продаёт пакет TclDevKit содержащий средства отладки и создания автономных программ (последнее на базе технологии StarKit) и Komodo — универсальное IDE для скриптовых языков, включающее средства визуального программирования Tk-форм.
  • etcl
    Еще одна сборка, от компании evolane. Богатый набор библиотек и платформ, включая мобильные платформы Windows Mobile 5, 6 а также linux под arm. Дистрибутив поставляется в виде одного исполняемого файла, с помощью которого можно собрать собственное приложение.
  • WinTclTk
    Более компактный, дистрибутив ориентированный на работу с XOTcl — объектно-ориентированным расширением Tcl. Список включённых в него библиотек несколько уже, чем в ActiveTcl, но «джентльменский набор» вы здесь найдёте. В этот дистрибутив входят Tloona IDE — среда разработки традиционного типа и XOTclIde — среда объектно-ориентированного программирования в стиле Smalltalk, обе — с отладчиками. Также на сайте разработчиков свободно доступен пакет для создания единого запускаемого файла TkWrapper основанный на FreeWrap — технологии, альтернативной StarKit, основанной на простом ZIP-сжатии. Последняя на сегодняшний день версия — 8.5.6.
  • TclKit
    Это Tcl/Tk c базовым набором библиотек в одном компактном выполняемом файле. Дистрибутив создан с использованием технологии StarKit. Содержит Библиотеку MetaKit позволяющую управлять содержанием этого файла.
  • dqkit
    Альтернативная сборка, известная как «TclKit на стероидах», имеется несколько вариантов, отличающихся составом библиотек, поддержкой многопоточности и т. д.
  • Tcl/Tk Aqua
    Дистрибутив для Mac OS. Существует три варианта: TclTkAquaBI — надмножество ActiveTCL, компактный дистрибутив TclTkAqua и минимальный набор TclTkAquaStandalone.
Достоинства и недостатки

Недостатком ранних версий Tcl/Tk, несомненно, было низкое быстродействие. Версии ниже 8-й использовали прямую интерпретацию исходного текста, однако использование байткода разрешило эту проблему. Сейчас по быстродействию Tcl не сильно отличается от PHP, хотя заметно уступает таким скриптовым языкам как, например, Python или Lua. Сложные или критические по времени выполнения участки кода рекомендуется выполнять в виде внешних модулей на C/C++.

Tcl/Tk, отличает непривычность синтаксиса для программистов на «традиционных» языках. Он ближе к стандартному синтаксису оболочки Unix. Этот синтаксис может быть произвольно изменён (как и в других «программируемых языках программирования», таких как Forth или Common Lisp). Это может быть преимуществом для программиста-индивидуала или небольшой группы программистов, но делает Tcl/Tk вряд ли пригодным для использования в качестве языка общего назначения при разработке в промышленном масштабе. Хотя наличие стандартизированных расширений снижает остроту этой проблемы. Tcl может использоваться и в крупных проектах в пределах своей «экологической ниши» (встроенные скрипты, «склеивание» компонент, WEB-программирование).

Хотя Tcl обладает всеми свойствами функционального языка, его интерпретатор до версии 8.6 не производил оптимизацию хвостовой рекурсии, что затрудняло использование чисто функционального стиля, уменьшало его эффективность. Впрочем, гибкость языка позволяет обойти эту проблему, реализовав подобие такой оптимизации на высоком уровне. Наконец, появившаяся в тестовой версии 8.6 команда tailcall позволяет без особых проблем использовать хвостовую рекурсию.

Наиболее тесная интеграция Tcl с графической библиотекой Tk, которая зачастую используется и в сочетании с другими языками программирования — несомненное преимущество Tcl. Для создания GUI оболочек к консольным программам или пакетам консольных программ Tcl может оказаться наилучшим выбором.

По количеству разработанных для Tcl библиотек расширений и инструментальных средств несколько уступает Питону, а тем более Перлу, но вряд ли вы столкнётесь с их нехваткой. Чего, правда, нельзя сказать о количестве документации и литературы по Tcl, особенно — русскоязычной.



Яндекс.Метрика
4Debian.Info
При полной или частичной перепечатке текстов статей и публикаций гиперссылка на www.4debian.info обязательна. | "4Debian.Info" | © 2018