forum.boolean.name

forum.boolean.name (http://forum.boolean.name/index.php)
-   Уроки (http://forum.boolean.name/forumdisplay.php?f=145)
-   -   Сопрограммы (Coroutine) в Unity (http://forum.boolean.name/showthread.php?t=13315)

pax 10.09.2010 22:46

Сопрограммы (Coroutine) в Unity
 
Решил написать маленький тутор по использованию корутин.

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

PHP код:

using UnityEngine;
using System.Collections;

public class 
CorutineTutor MonoBehaviour
{
    public 
Texture2D texture;

    private 
Vector3 _position

    public 
void Start()
    {
        
// запускаем корутину 
        
StartCoroutine(Coroutine());
    }

    public 
IEnumerator Coroutine()
    {
        
// центр экрана
        
Vector3 centerScreen = new Vector3(Screen.width/2fScreen.height/2f0);

        
// бесконечный цикл
        
while (true)
        {
            
float time Time.realtimeSinceStartup;

            
// перемещение по "кренделю" :)
            
_position centerScreen 
                new 
Vector3(200 Mathf.Sin(time), 200 Mathf.Cos(time), 0) * Mathf.Sin(time 3);

            
// остановка выполнения функции на 10 миллисекунд
            
yield return new WaitForSeconds(0.01f);
        }
    }


    public 
void OnGUI()
    {
        
// рисуем текстуру
        
GUI.DrawTexture(
            new 
Rect(
                
_position.texture.width 2f
                
_position.texture.height 2f
                
texture.width
                
texture.height
                
), texture);
    }


демо работы скрипта

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

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

Итак последовательно:
1. При старте скрипта запускается корутина (функция с названием Coroutine).
2. Эта функция выполняется до строчки yield return new WaitForSeconds(0.01f); которая прерывает ее работу на 10 миллисекунд.
3. Прерывание не влияет на остальные функции (процесс не останавливается и действие происходит в том же потоке)
4. По истечении времени корутина продолжает бесконечный цикл пока снова не доходит до строки с задержкой.
5. Функция OnGUI рисует текстуру на экране.


Возможные прерывания выполнения кода:
1. yield return null; - прерывает выполнение корутины до следующего кадра.
2. yield break; завершает выполнение корутины.
3. yield return new WaitForSeconds( количество секунд ); - прерывает на время.
4. yield return new WaitForEndOfFrame(); - прерывает выполнение до конца кадра.
5. yield return new WaitForFixedUpdate(); - прерывает выполнение до кадра, в котором обновляется физика.

Специфическое использование:
PHP код:

// Скриншот  "Friday's" in Times Square
string url "http://images.earthcam.com/ec_metros/ourcams/fridays.jpg";
public 
IEnumerator Start () 
{
     
// Запуск скачивания картинки по приведенной url
    
WWW www  = new WWW (url);
    
    
// ожидание конца скачивания
    
yield return www;
    
    
// применение загруженной текстуры к материалу
    
renderer.material.mainTexture www.texture


Чтобы использовать корутины в C# необходимо объявлять эти функции с типом IEnumerator. Для JavaScript этого не нужно.


Еще один пример как можно реализовать то же самое, что и выше в функции Coroutine но без бесконечного цикла:
PHP код:

public IEnumerator Coroutine()
    {
        
// центр экрана
        
Vector3 centerScreen = new Vector3(Screen.width/2fScreen.height/2f0);

        
float time Time.realtimeSinceStartup;

        
// перемещение по "кренделю" :)
        
_position centerScreen 
                new 
Vector3(200 Mathf.Sin(time), 200 Mathf.Cos(time), 0) * Mathf.Sin(time 3);

        
// остановка выполнения функции на 10 миллисекунд
        
yield return new WaitForSeconds(0.01f);

         
// запускаем корутину снова
        
StartCoroutine(Coroutine());
    } 


Кода можно применять: например для распределения тяжелых вычислений на несколько кадров, для последовательного изменения чего-то и т.д.

Ну вот и все на сегодня :).

LLI.T.A.L.K.E.R. 18.12.2014 01:30

Ответ: Сопрограммы (Coroutine) в Unity
 
Напишу что у меня по поводу короутинов:
(коды только изучаю, особо не знаю как ошибки исправлять, просто стараюсь их обходить и не создавать)
просто пишу свою мыслю


void Start() {
StartCoroutine("AimCoroutine");
StartCoroutine(SpawnCoroutine());
}

IEnumerator AimCoroutine() {
while(true)
{
yield return new WaitForSeconds(0.1f);
//некий код
StopCoroutine("AimCoroutine"); //короутин сам себя прекратит, если нужно одноразовое использование
}
}

IEnumerator SpawnCoroutine() {
while(true)
{
yield return new WaitForSeconds(3.0f-speed);
//некий код
StopCoroutine(SpawnCoroutine); //либо StopCoroutine(SpawnCoroutine())
//тут уже не получится самопрекратить короутин (у меня Unity ругался поэтому я пишу названия в двойный кавычках)
}
}

если сможете, то можете поправить и указать причину ошибки StopCoroutine(SpawnCoroutine())

Mr_F_ 18.12.2014 01:36

Ответ: Сопрограммы (Coroutine) в Unity
 
Цитата:

while(true)
{
yield return new WaitForSeconds(0.1f);
//некий код
StopCoroutine("AimCoroutine"); //короутин сам себя прекратит, если нужно одноразовое использование
}
почему не сделать просто break?

LLI.T.A.L.K.E.R. 18.12.2014 01:46

Ответ: Сопрограммы (Coroutine) в Unity
 
всех нюансов ещё не знаю :ok:

теперь нашёл
Возможные прерывания выполнения кода:

но в коде не видел...

и уточню - почему не сделать просто: yield break;
а то уже хотел: break; писать =)

pax 18.12.2014 07:26

Ответ: Сопрограммы (Coroutine) в Unity
 
Тут вот третий пост показывает, как остановить корутину, которая принимает IEnumerator:
http://forum.unity3d.com/threads/how...erator.249732/

Цитата:

PHP код:

// variable that holds reference to coroutine (IEnumerator)
public IEnumerator routine;
 
// get IEnumerator from Coroutine and start
routine DoWork();
StartCoroutineroutine );
 
// stop Coroutine
StopCoroutineroutine ); 


т.е. надо сохранить в публичной переменной ссылку на созданный первым запуском IEnumerator. Единственное я вижу в этом проблему: корутина сработает при запуске два раза, надо в начало ставить yield return null; наверное...


Часовой пояс GMT +4, время: 21:46.

vBulletin® Version 3.6.5.
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
Перевод: zCarot