I am participating in the nuggets Community game creative submission Contest. For details, please see: Game Creative Submission Contest.

One, the introduction

Before we start, let’s first understand what the Flocking algorithm is.

Flocking algorithm is generally known as the swarm algorithm in China. It is formed by many discrete animals, but the group as a whole is flowing, which is the comprehensive result of individual behavior.

Typical natural phenomena include: bees, birds, fish, herds, etc. The phenomenon of animals gathering (including human beings) can help organisms better avoid natural enemies, migration, and access to food……

Here are some pictures:

The beast migration

Wolves hunt

Bees make honey

Geese fly south

The fish moved

Second, the development of

In July 1987, Craig Reynolds took the lead in proposing the classic Flocking model, which requires group behavior to meet three rules:

  • Convergence: the gradual joining of independent individuals into a group
  • Speed matching: Keep the individual and the group on the same course and do not disengage
  • Separation: To avoid collisions between individuals in a group

Three, the fish

Below, a simple fish swarm simulation is mainly realized in Unity3D, realizing the functional modules of fish generation, fish aggregation, speed matching, predation, separation and so on. Here is a brief introduction.

1. The components

To better manage the fish, we define a component in the script.

    [Header("Fish Setting")]// Control panel
    [Range (0.0 f, 5.0 f)]
    public float min;// Minimum speed
    [Range (0.0 f, 5.0 f)]
    public float max;// Minimum speed
    [Range (1.0 f, 10.0 f)]
    public float neighborDistance;// The aggregation distance
    [Range (0.0 f, 5.0 f)]
    public float RotationSpeed;/ / speed
Copy the code

2. Create fish

In the Create script, you define a range within which the fish in the array can be generated and moved.

    public GameObject prefab1;/ / 🐟 type 1
    public GameObject prefab2;/ / 🐟 type 2
    public int fishnum=50;// Initialize 🐟 quantity
    public GameObject[] fish;// Array storage
    public Vector3 swimlimt = new Vector3(5.5.5);10 * 10 * 10 / / boundary
Copy the code

To generate, we use random generation, and the range is still fixed inside the boundary range.

    public void Start()
    {
        fish = new GameObject[fishnum];
        for(int i=0; i<fishnum; i++) { Vector3 pos =this.transform.position + new Vector3(Random.Range(-swimlimt.x,swimlimt.x),
                                                                Random.Range(0, swimlimt.y),
                                                                Random.Range(-swimlimt.z, swimlimt.z));
            if(i%2= =0)
                fish[i] = (GameObject)Instantiate(prefab1,pos,Quaternion.identity);/ / clone 🐟
            else
                fish[i] = (GameObject)Instantiate(prefab2, pos, Quaternion.identity);
            fish[i].GetComponent<FlockSpeed>().sp = this;// Contact between two scripts}}Copy the code

3. Fish movement

For fish movement, add speed and direction to the FlockSpeed script.

    private void Update()
    {
        speed = Random.Range(sp.min, sp.max);// Speed range
        this.transform.Translate(0.0, speed * Time.deltaTime);// Start moving
    }
Copy the code

At this time, the fish will only move towards the Z-axis, and nothing else can be done. The following steps are the key to the application of Flocking algorithm.

Four, aggregation,

Aggregation is the key to the Flocking algorithm, which brings the fish together.

Remember that our component defined a variable called neighborDistance, which is the aggregate distance. If the distance between two fish is <=neighborDistance, it’s in the cluster, and we need to find a way to add that fish to the cluster.

So how do you get the fish to meet the whole and not break away? This is where you need the center of the shoal, or mean position.

For a shoal, each 🐟 has a specific location, and it is impossible for two 🐟 to coincide, so:

Average position = sum of fish positions/number of fish

The red star is the center of the calculation, 🐟neighborDistance at the bottom, but if the location is not corrected, it will run out of the cluster. You know vectors, for a three dimensional coordinate, the vector of the red arrow in the figure above is equal to the coordinate:

Business – desired.

So, this fish needs to pivot, and it also needs to be careful not to touch other fish when it turns, and I’ll leave that for separation.

void Community()
    {
        GameObject[] gos=sp.fish;
        float Distance;
        Vector3 center = Vector3.zero;
        Vector3 avoid = Vector3.zero;
        int size=0;
        float goSpeed=0.01 f;
        foreach(GameObject go in gos)
        {
            if(go! =this.gameObject)// Get rid of all the fish in the group
            {
                Distance = Vector3.Distance(this.transform.position, go.transform.position);// The distance between two adjacent fish
                if (Distance<=sp.neighborDistance)// Cluster rules are met
                {
                    size++;
                    center+= go.transform.position;

                    if (Distance < limt)// Avoid collisions
                    {
                        avoid += this.transform.position - go.transform.position; } FlockSpeed flockSpeed = go.GetComponent<FlockSpeed>(); goSpeed += flockSpeed.speed; }}}if(size>0)
        {
            center = center / size+(sp.goal- this.transform.position );// Add the target position, move towards the target
            goSpeed = goSpeed / size;// Average speed
            Vector3 direction = (center + avoid) - transform.position;
            if(direction! =Vector3.zero)// Need to turn
            {
                transform.rotation = Quaternion.Slerp(transform.rotation, Quaternion.LookRotation(direction), sp.RotationSpeed * Time.deltaTime);//from to smooth Quaternion.LookRotation watch rotation}}}Copy the code

Five, speed matching

As shown in the figure above, each 🐟 has a direction of travel. We want to make sure that the speed of all the fish eventually matches the speed of the virtual pilot, so what is the speed of the virtual pilot for the whole cluster?

Virtual pilot's speed = total speed and/fish count

As shown in the figure above, the current heading and average heading are not oriented in the right direction, so the direction should be adjusted.

After adjustment, the two directions remain consistent.

Finally, each fish in the cluster has to achieve this effect.

Speed matching is the goSpeed variable, and the code part is also in the Community() function above.

Six, predation

To realize fish predation, first define a target, which has a 10% probability of moving, and the fish will follow the target.

private void Update()
    {
        //10% chance to change targets
        if(Random.Range(0.100) <10)
            goal = this.transform.position + new Vector3(Random.Range(-swimlimt.x, swimlimt.x),
                                                         Random.Range(0, swimlimt.y),
                                                         Random.Range(-swimlimt.z, swimlimt.z));
    }
Copy the code

In order to see the movement of the fish more clearly, I turned off the random movement and manually changed the position of the target object. The shoal of fish can be seen following the movement of the target.

Seven, separation

1. Avoid 🐟

To avoid collisions between fish within a shoal, specify a separation principle.

As shown in the image above, the fish would have converged toward the center, but they would have collided with other fish, so they would have to turn.

The dodge variable between 🐟 and 🐟 is avoid, and the code part is in the Community() function above.

2. Avoid borders

At the beginning, we defined a boundary for 10∗10∗1010*10*1010∗10∗10 remember?

To prevent the fish from swimming beyond the boundary, we define a turn so that when the fish is about to go beyond the boundary, it turns around and moves toward the center.

        Bounds b = new Bounds(sp.transform.position,sp.swimlimt);// The bounding box
        Vector3 direction = Vector3.zero;
        RaycastHit hit = new RaycastHit();
        if(! b.Contains(transform.position))// About to go out of bounds
        {
            turning = true;// Start to turn
            direction = sp.transform.position - transform.position;// Turn direction to center position
        }  
        else
            turning = false;
Copy the code

3. Avoid obstacles

In the process of moving, fish can meet obstacles and predators, at this time, they need to make fast obstacle avoidance function.

Physics knowledge in junior high school, Angle of incidence = Angle of reflection, we need the fish brain above the launch of a long ray, when encountering obstacles.

The original direction is equal to the reflected light.

Judgment to

        if(Physics.Raycast(this.transform.position,transform.forward*5.out hit))// Hit an obstacle
        {
            //Debug.DrawRay(this.transform.position,this.transform.forward*5,Color.red); The red ray
            direction = Vector3.Reflect(this.transform.forward, hit.normal);// Reflection Angle = incident Angle turn
            turning = true;
        }
Copy the code

Turning to

        if(turning)
        {
            this.transform.rotation = Quaternion.Slerp(transform.rotation, 
            Quaternion.LookRotation(direction), sp.RotationSpeed * Time.deltaTime);  
        }
        else
        {  
            Community();// Aggregate function
        }
Copy the code

8. Effect display

In the GIF above, I purposely placed the fish target inside the pillar to see if the fish will hit the pillar directly.

Fish can normally see objects within 12 meters, but I purposely enlarged the fish’s field of vision by five times. In fact, when the red line emitted from the fish’s head hits this big pillar on the bottom of the sea, it turns.

But since the target didn’t move, the fish had to rotate around the pole (team hunt) until I moved the target out of the pole and the fish started to move away.

Nine,

Flocking algorithm is the cooperative behavior of group animals and achieves the overall goal through the interaction between individuals. This paper is only a simple 🐟 cluster simulation, only an algorithm branch of pattern recognition domain data clustering, steering Angle, field of vision, leader has not been implemented.

In short, thank you very much for patiently reading this long article ( ̄)  ̄).

【 References 】

  • Flocking Algorithm in Unity

  • Flocking clustering algorithm with virtual leadership

  • A Flocking algorithm based on hybrid agent system

  • Simulation of group behavior: Flocking algorithm

  • Flocks, Herds, and Schools: A Distributed Behavioral Model

  • AI for Game Developers by Glenn Seemann, David M Bourg