Author Topic: Массивы  (Read 73988 times)

0 Members and 1 Guest are viewing this topic.

Johnny

  • Создатель
  • Герой форума
  • *
  • Posts: 593
    • View Profile
Массивы
« on: October 08, 2013, 12:44:28 PM »
последняя редакция: 21.03.2014

Вводная часть

В версии 4.8 были введены массивы. Для начала классическое школьное определение. Массив (англ. Array) – это набор однотипных данных, расположенных в памяти друг за другом. При этом, благодаря этому «друг за другом», каждый элемент массива имеет свой порядковый номер (индекс, смещение). По сути, этот номер определяет смещение элемента массива в памяти относительно его (массива) начала, но вам не стоит вникать во все эти сложности, потому что как обычно, все нюансы «сложного программирования» Clickermann обрабатывает сам.

Итак массив, как и переменная, имеет собственное имя. Правила для имен массивов такие же как и для переменных. При этом, после указания имени массива, в квадратных скобках задается индекс элемента для получения его значения (или соотв. присвоения).

Рассмотрим пример

Code: (clickermann) [Select]
$arr[0] = 1
$arr[1] = 2
$arr[2] = 3
$arr[3] = 4
$arr[4] = 5

Здесь мы разместили в массиве $arr пять элементов. Обратите внимание что индексация массива начинается (как и в С++) с нуля.
Теперь, к примеру, если мы сделаем следующую операцию

Code: (clickermann) [Select]
Print( $arr[1] )
То в лог выведется «2». По аналогии с переменными, элементы массива могут становиться аргументами функций и процедур, участниками условий и вычислений. Вы можете спросить в чем же тогда удобство? Ну, например, вышеозначенный массив имеет пять элементов, которые мы могли бы заменить переменными $arr1($arr2, …). Тогда их вывод имел бы вид

Code: (clickermann) [Select]
Print($arr1)
Print($arr2)
Print($arr3)
Print($arr4)
Print($arr5)

Уже смотрится не очень. А если бы переменных было штук пятьдесят?
Однако в случае с массивом все намного проще. Мы просто прогоним весь массив в цикле

Code: (clickermann) [Select]
For($i, $i<5)
  Print( $arr[ $i ] )
End_cyc

Да, индексный параметр в квадратных скобках так же поддерживает вычисления и переменные, что существенно облегчает нам жизнь. Три строчки кода против пяти для нашего примера. И те же три строчки против пятидесяти были бы если бы у нас было, как я говорил, пятьдесят переменных.


Объявление и доступ

Что будет если выскочить за пределы массива? В том же С++ программа при работе выдала бы ошибку доступа к виртуальной памяти. Однако в нашем случае это просто вернуло бы «0» (не уверен, что всегда).

Массивы в Clickermann являются динамическими. Поскольку, как вы уже знаете, Clickermann лишен необходимости объявлять заранее переменные (и вообще что либо… хотя не гарантирую такую халяву в будущем… шов уже трещит), то это же правило распространяется и на массивы.

Вы можете например задать массив на семь элементов таким образом

Code: (clickermann) [Select]
$arr[6] = 5
При этом элементы массива 0,1,2,3,4,5 будут содержать т.н. мусор, то есть данные, лежащие в оперативной памяти до выполнения нашего кода. Это происходит потому что память массиву присваивается, но никакие значения в нее пока не кладутся и там хранится старая мусорная информация. Данный массив так же можно растянуть еще до десяти элементов

Code: (clickermann) [Select]
$arr[8] = 32
$arr[9] = 43

Однако это работает только в одну сторону. Чтобы обрезать массив потребуется вызвать спец функции. Поговорим о них.

Функции работы с массивом

Для начала я напомню, что массив имеет собственное имя (та часть, что без квадратных скобок). Некоторые функции работы с массивом имеют в качестве аргумента именно имя массива, а не какой-то его конкретный элемент.

Функция arrsize($arr)

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

Code: (clickermann) [Select]
$arr[0] = 12
$arr[5] = 123

Print( arrsize($arr) ) // вернет «6»

К слову сказать, если размерность массива 1, то это по сути переменная. И с ним можно работать как с переменной (запись «$arr[0]» в данном случае будет эквивалента «$arr»)

Процедура arrpush($arr, value)

Данная процедура добавляет новый элемент в конец массива. Каждый ее вызов соответственно увеличивает размер массива на 1. Вот пример

Code: (clickermann) [Select]
Arrpush($arr, 1)
Arrpush($arr, 2)
Arrpush($arr, 3)

Print( $arr[1] ) // вернет «2»
Print( arrsize($arr) ) // вернет «3»

Таким образом, можно особо и не возиться с индексами чтобы заполнить массив

Функция arrpop($arr)

Данная функция возвращает последний элемент массива, при этом удаляя его из массива, уменьшая, таким образом, его размерность на 1. Да, именно так можно обрезать (или целиком вообще вычистить) массив. Стоит понимать, что эта функция выдает элементы в обратном порядке (т.к. после удаления последним элементом становится предыдущий).

Code: (clickermann) [Select]
Arrpush($arr, 1)
Arrpush($arr, 2)

Print( arrpop($arr) ) // «2»
Print( arrpop($arr) ) // «1»

Но самое главное, что с помощью этих двух функций – arrpush и arrpop – можно без труда организовать самый натуральный стэк. Это тема отдельной беседы, потому что стэк (и регистры) фундаментальные понятия микропрограммирования. Нам же нужно помнить, что стэк, это стопка. Если положить то только сверху. Если взять, то только сверху. Если нужен предпоследний элемент, извольте сначала «снять с него» последний. Можете почитать Вики, если интересно, http://ru.wikipedia.org/wiki/%D0%A1%D1%82%D0%B5%D0%BA

Процедуры scanpicture и scanpxl

Ну и наконец две самые вкусные функции, призванные, наконец, если не заменить, то дополнить два самых старейших костыля if_pixel_in и if_picture_in. Они конечно работают и на них построено огромное количество скриптов, однако у них есть один существенный недостаток – они находят только один элемент. Точнее первый. Например, как только наличие картинки на экране будет установлено, дальнейший анализ прекращается. Продвинутые пользователи конечно выходили из ситуации, играясь с координатами поиска, но это все не то (хотя безусловно полезно для общего развития).

Теперь разберем рассмотрим следующий скрипт

Code: (clickermann) [Select]
getscreen

scanpicture($var, 0,0, 1250,959, "pict.bmp")

while (arrsize($var) > 0)
  $y = arrpop($var)
  $x = arrpop($var)
  lclick($x, $y)
end_cyc

Этот короткий скрипт реализует очень большой геморрой. Он ищет в заданной области все экземпляры картинки pict.bmp и последовательно кликает на каждую из них. Разберем его по порядку. Прежде всего обратите внимание на новую процедуру scanpicture. Ее синтаксис схож с if_picture_in за тем исключением, что первым параметром идет массив. В этот массив процедура последовательно добавляет (через внутренний вызов arrpush) координаты каждого найденного экземпляра искомого изображения. Таким образом после выполнения scanpicture, входной массив $var будет содержать N пар координат x,y, где N количество найденных картинок.

Полученный массив можно представить схематически так:
[x1, y1, x2, y2, … , xN, yN] где x,y – пара координат

Далее необходимо всю эту кучу пробежать. Удобнее всего, как известно, перебирать массивы в цикле. Организуем цикл while с условием arrsize($var) > 0. Это будет оптимально по тому что позволит сразу проскочить тело цикла, если ничего не найдено. Стоит заметить что arrsize вернет значение N * 2, потому что эта функция возвращает длину массива и ее не волнует наше представление данных (в данном случае имеется ввиду что i – это координата X, i+1 – координата Y).
Далее мы в теле цикла вызываем arrpop($var) таким образом выдергивая последнюю координату Y и уменьшая длину цикла на 1. Следующий вызов arrpop вернет уже координату X последней пары (про работу стэка я уже рассказывал выше, вот это типичный пример). Таким образом в координатах $x, $y у нас лежат координаты последней найденной картинки. А поскольку arrpop уже удалил эту пару из массива, то сейчас там в самом конце лежат уже координаты предыдущей картинки. Таким образом извлекаются все координаты найденных картинок. И когда будет извлечена последняя пара, arrsize вернет 0 и цикл закончится.

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

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

Стоит добавить, что массивы, как и переменные хранят свои значения. Поэтому последовательный вызов двух scanpicture (например) без извлечения элементов через arrpop приведет к тому что у вас будет двойная очередь координат. Поэтому если вам для работы необходим «чистый» массив, то не забывайте высвобождать память, путем вызова undefine($var). Тут все как для переменных.

Такие дела. Я как мог подробно разжевал основные моменты работы с массивами в общем и тонкости их использования непосредственно в Clickermann. Надеюсь что данный материал поможет вам легко и быстро освоить новые способы программирования.
« Last Edit: November 02, 2014, 04:01:22 PM by Vint »

mmutant

  • Зашел в гости
  • *
  • Posts: 2
    • View Profile
Re: Массивы
« Reply #1 on: October 12, 2013, 10:26:46 AM »
Quote
удалось прикрутить массивы в незначительный ущерб имеющейся в 4.7 гибкости (да, да, да, старые скрипты могут слететь)
А можно подробней про "ущерб"?

Oraven

  • Супермодератор
  • Герой форума
  • *
  • Posts: 3685
  • Котэ
    • View Profile
Re: Массивы
« Reply #2 on: October 12, 2013, 12:44:14 PM »
Например, если переменная содержит "число", и в нее попытаться записать "строку" то выскочит ошибка. В 4.7 ошибок не было.

$a = 1
$a = "тест"

Johnny

  • Создатель
  • Герой форума
  • *
  • Posts: 593
    • View Profile
Re: Массивы
« Reply #3 on: October 13, 2013, 09:05:10 AM »
ключевой момент в том, какого типа значение присваивается первый раз.
если инициализировать строкой, то все будет работать как в 4.7
то есть наоборот

$a = "тест"
$a = 1
print($a + 2)

в принципе, кстати, этот момент я могу исправить и вернуть как было в 4.7. но на сей раз придется пожертвовать оперативной памятью :D
а вообще, если развивать проект дальше до уровня чего-то маломальски похожего на язык программирования, то придется отказаться от такой халявы и вводить жесткие типы данных. как минимум поделить на строковый и численный (его в свою очередь можно поделить на дробный и целочисленный). без этого мы лишаемся возможности более плотно взаимодействовать с системой.
« Last Edit: October 13, 2013, 09:15:45 AM by Johnny »

Johnny

  • Создатель
  • Герой форума
  • *
  • Posts: 593
    • View Profile
Re: Массивы
« Reply #4 on: October 13, 2013, 09:16:44 AM »
ключевой момент в том, какого типа значение присваивается первый раз.
если инициализировать строкой, то все будет работать как в 4.7
то есть наоборот

$a = "тест"
$a = 1
print($a + 2)

в принципе, кстати, этот момент я могу исправить и вернуть как было в 4.7. но на сей раз придется пожертвовать оперативной памятью :D
а вообще, если развивать проект дальше до уровня чего-то маломальски похожего на язык программирования, то придется отказаться от такой халявы и вводить жесткие типы данных. как минимум поделить на строковый и численный (его в свою очередь можно поделить на дробный и целочисленный). без этого мы лишаемся возможности более плотно взаимодействовать с системой, в которой все типы данных четко регламентированы.

ЦИТРИН

  • Супермодератор
  • Оплот сообщества
  • *
  • Posts: 484
  • Марафон опавших листьев, коктейль из дыма и мыслей
    • View Profile
Re: Массивы
« Reply #5 on: October 26, 2013, 01:22:41 AM »
Спасибо за массивы)
Все отлично отрабатывается.
Весьма удобно.
Сокращает работу над скриптом в сложных задачах.
Бабочки полёт будит тихую поляну в солнечном свету...

ЦИТРИН

  • Супермодератор
  • Оплот сообщества
  • *
  • Posts: 484
  • Марафон опавших листьев, коктейль из дыма и мыслей
    • View Profile
Re: Массивы
« Reply #6 on: November 05, 2013, 05:17:18 PM »
Есть вопросы.
Возникла задача  автоматизировать процесс  кликанья в метки "лайк"  по блогу.
На странице блога примерно  2-5 однотипных изображений лайков ("+").
Скрипт
Code: (clickermann) [Select]
WAITMS(500)
DEFINE($NUM2,0)
DEFINE($NUM1,0)
start:
GETSCREEN
WAITMS(500)
SCANPICTURE($ARR_SCAN,347,135,719,1037, "plus1.bmp", -1, 95)
$GGG = ARRSIZE($ARR_SCAN)
IF($GGG = 0) // если больше нет картинок  "+"
   WAITMS(500)
   LCLICK(1673,1003)
   INC($NUM,1)
   INC($NUM2,1)
   KEYPRESS(#pagedown ) // перелистываем страницу
   WAITMS(500)
   IF($NUM > 3) // если  нет картинок посл 3 сканов  страницы идем вверх и повторяем  скрипт
      LCLICK(1673,1003)
      WAITMS(500)
      KEYPRESS(#home)
      WAITMS(500)
      UNDEFINE($NUM) // обнуляем  NUM
      IF($NUM2 > 9) //если  нет картинок после 9 сканов  страницы завершаем  скрипт
         HALT
      END_IF
      goto( start)
   END_IF
END_IF
$ARR_NUM = 0
WHILE ($ARR_NUM < ARRSIZE($ARR_SCAN))
   $X = $ARR_SCAN [$ARR_NUM]
   INC($ARR_NUM,1)
   $Y = $ARR_SCAN [$ARR_NUM]
   LCLICK($X, $Y)   // перемещаем курсор
   WAITMS(400)
END_CYC
UNDEFINE($ARR_SCAN)
goto( start)
Так вот .  Команда    KEYPRESS(#pagedown ) // перелистываем страницу
дает двойное перелистывание  а не одиночное.
 Команды
Code: (clickermann) [Select]
....
IF($NUM2 > 9) //если  нет картинок после 9 сканов  страницы завершаем  скрипт
         HALT
      END_IF
      goto( start)
   END_IF
END_IF
....
вместо    goto( start)  приводят к открытию новой  чистой вкладки в браузере Хром.
 Что я неправильно  сделал?
« Last Edit: November 05, 2013, 05:21:17 PM by ЦИТРИН »
Бабочки полёт будит тихую поляну в солнечном свету...

Johnny

  • Создатель
  • Герой форума
  • *
  • Posts: 593
    • View Profile
Re: Массивы
« Reply #7 on: November 05, 2013, 05:47:51 PM »
сдается мне ты какую то функциональную клавишу зажал. это вовсе не про массивы...

Vint

  • Супермодератор
  • Герой форума
  • *
  • Posts: 3935
  • Лечу куда хочу. cman 4.13.014x32, 4.14.003 W10
    • View Profile
Re: Массивы
« Reply #8 on: November 09, 2013, 08:43:07 PM »
Тут наверняка не в клавишах дело.
 Что-то с подложкой сайта. Она меняется постоянно + картинки слегка полупрозрачными сделаны.
Сумма пикселей всегда разная.
 Простую задачу не могу решить
Есть поле.
На поле может быть до 100  разных картинок.
Расположены они  строго по рядам ( межрядное 40 пикселей)  и строкам( межстрочное тоже 40 ).
В каждом ряду  может  быть только по 20 картинок
Строк всего  может  быть только 5.
Нужно последовательно  тыцнуть примерно в центр  каждой картинки.
Условие простое--->  Если картинок 24,  то нужно нажать ровно 24 раза.
 Постоянные ошибки.
Наверно нужно без массива искать выход.
Помогите корифеи !)
Тебе нужно нажать только на яйца с определённым фоном?
Открыл. Никаких проблем нет. Подложка нигде не пролазит. Да, на яйца что-то влияет, думаю рендеринг, а на цветной фон яйца ничего. Цвет вообще чистый на определённом удалении от яйца, нужно проверять его там и кликать со смещением в центр.
Вот фотошопная палочка с допуском 0...

И да, ты прав массивы здесь нафиг не нужны. Нужен только PXL() и всё. Шаг стабильный, рамка не размазана. 2 вложенных цикла перебирающих вертикальный и горизонтальный шаг, внутри сравнение PXL() и действие. Всё.

Дополнение! Горизонтальный шаг 39, а не 40.
« Last Edit: November 09, 2013, 08:56:59 PM by Vint »


Vint

  • Супермодератор
  • Герой форума
  • *
  • Posts: 3935
  • Лечу куда хочу. cman 4.13.014x32, 4.14.003 W10
    • View Profile
Re: Массивы
« Reply #9 on: November 09, 2013, 09:14:30 PM »
Беда в том ,что яйца часто меняются на иные картинки.
Не угадаешь.
Тупо тыкать  попробовал, но там  и число меняется.
100 раз  все протыкивать  и соотносить  каждый скрипт ( примерно 5000 строк )  своей переменной?
Суммарный скрипт выйдет на 25 мегабайт)))
Наверно есть простое решение.
Заранее известно , что будет  допустим 36, или 95  картинок.
Я эту переменную получаю.
Как протыкать  36 или 95  раз , каждый раз, новую  картинку, в зависимости от этой переменной?
 PXL()  не получается.
Если честно... ничего не понял.

Тебе нужно тыкать на все подряд? На скрине всего 7 цветов фона. Других не бывает?
И давай тему массивы не замусоривать.
« Last Edit: November 09, 2013, 09:16:05 PM by Vint »


Vint

  • Супермодератор
  • Герой форума
  • *
  • Posts: 3935
  • Лечу куда хочу. cman 4.13.014x32, 4.14.003 W10
    • View Profile
Re: Массивы
« Reply #10 on: November 19, 2013, 03:56:32 PM »
При использовании ARRPUSH, странно слетают дробные числа.
Хотел округлить, но не тут то было. В версии 4.8 сломался ROUND с отрицательным параметром, не могу округлить числа.
Предполагаю это общая проблема.
HELP!

Code: (clickermann) [Select]
ARRPUSH($Arr_Cena, 1120.69)
ARRPUSH($Arr_Cena, 1269.25)
ARRPUSH($Arr_Cena, 1425.4)
ARRPUSH($Arr_Cena, 1563.7)
ARRPUSH($Arr_Cena, 1704.68)
ARRPUSH($Arr_Cena, 1855.03)
ARRPUSH($Arr_Cena, 2004.48)
ARRPUSH($Arr_Cena, 2307.41)
ARRPUSH($Arr_Cena, 2611.22)
ARRPUSH($Arr_Cena, 2977.94)

$i = 0
for ($i, $i<ARRSIZE($Arr_Cena))
   LOGWRITE (" ", ROUND($Arr_Cena[$i-1], -2))
end_cyc
HALT

16:57:01  1120.6900000000001
16:57:01  1269.25
16:57:01  1425.4000000000001
16:57:01  1563.7000000000001
16:57:01  1704.6800000000001
16:57:01  1855.03
16:57:01  2004.48
16:57:01  2307.4099999999999
16:57:01  2611.2199999999998
16:57:01  2977.9400000000001


Torari

  • Зашел в гости
  • *
  • Posts: 18
    • View Profile
Re: Массивы
« Reply #11 on: March 26, 2014, 11:56:23 AM »
А можно положить один массив в другой массив? Если это сделать можно, то было бы здорово посмотреть на пару простых примеров. Для общего повышения образованности, так сказать. Создать массив№1, который лежит в массиве№2. Положить данные в массив№1, который в массиве№2. Взять данные или вывести данные в логи, которые лежат в массиве№1, который лежит в массиве№2.

Oraven

  • Супермодератор
  • Герой форума
  • *
  • Posts: 3685
  • Котэ
    • View Profile
Re: Массивы
« Reply #12 on: March 26, 2014, 12:37:01 PM »
А можно положить один массив в другой массив? Если это сделать можно, то было бы здорово посмотреть на пару простых примеров. Для общего повышения образованности, так сказать. Создать массив№1, который лежит в массиве№2. Положить данные в массив№1, который в массиве№2. Взять данные или вывести данные в логи, которые лежат в массиве№1, который лежит в массиве№2.

Шта  :o

В обычной переменной хранится одно значение или строка, а в массиве, как в таблице, много значений или строк в столбик. Каждая "строка" имеет свой номер от 0 и далее.
« Last Edit: March 26, 2014, 12:39:56 PM by Oraven »

Torari

  • Зашел в гости
  • *
  • Posts: 18
    • View Profile
Re: Массивы
« Reply #13 on: March 26, 2014, 01:05:03 PM »
Quote
а в массиве, как в таблице, много значений или строк в столбик.
Ну, а в любой таблице, в ячейку можно вставить другую таблицу =) можно в ворде поэкспериментировать  ;).

Такие массивы называются "массивы массивов" или "многомерные массивы". Как-то так или как-то по другому. Кажется в школе была тема про них или в вузе, не помню уже подробностей. В любом случае, если кликермэн это поддерживает, то мне нужны примеры синтаксиса... иначе, я сам не додумаюсь).

Крис Тинка

  • Активный участник
  • ***
  • Posts: 171
    • View Profile
Re: Массивы
« Reply #14 on: March 26, 2014, 01:46:11 PM »
В кликерманне линейные массивы. Можно сделать псевдо-многомерные массивы. Например псевдо-двумерный массив(таблица 100х100) ячейка 45х53
Code: (clickermann) [Select]
$a = 45
$b = 53
$var[100*$a + $b] = 254
print ($var[4553])
halt
« Last Edit: March 26, 2014, 01:49:08 PM by Крис Тинка »