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