Algoritma GSA (Gravitational Search Algorithm)

Algoritma GSA (Gravitational Search Algorithm) adalah salah satu algoritma optimasi yang dapat digunakan untuk pengambilan keputusan. Contoh yang dibahas kali ini adalah mengenai pencarian posisi dengan pengembalian nilai fungsi maksimal.
Algoritma ini terinspirasi dari sebuah teori hukum gravitasi yaitu teori Newton. Inti dari teori tersebut adalah “Setiap partikel yang ada di dunia akan saling menarik satu sama lain dengan kekuatan yang berbanding lurus dengan massa partikel dan berbanding terbalik dengan jarak antar partikel tersebut”. Sistem tarik menarik tersebut yang akan digunakan dalam melakukan pemecahan permasalahan optimasi pada kasus ini.



Diasumsikan ada sebaran titik 2 dimensi antara -2 sampai dengan 2
Fungsi yang diketahui adalah fungsi Himmelblau, dengan rumus f(x, y) = (x^2+y-11)^2 + (x+y^2-7)^2
Tentukan posisi dimana fungsi tersebut mengembalikan nilai maksimal



Fungsi Himmelblau adalah salah satu fungsi yang dapat digunakan untuk mengoptimasi suatu permasalahan. Fungsi ini memiliki sebuah nilai maksimum pada x = -0.270845, and y = -0.923039 dengan nilai fungsi sebesar f(x,y) = 181.617, dengan asumsi bahwa rentang minimal dan maksimal dari sebaran titik adalah -2 sampai dengan 2

Grafik fungsi Himmelblau yang normal, atau untuk sebaran titik tak terbatas adalah sebagai berikut.
Grafik Himmelblau

Sedangkan Grafik fungsi Himmelblau untuk sebaran titik dengan rentang minimal -2 dan maksimal 2 adalah sebagai berikut.
Grafik Himmelblau -2sd2
Dapat dilihat bahwa pada gambar tersebut, didapatkan area dengan titik tertinggi (berwarna merah) berada pada area x = -0, and y = -1, dimana titik tersebut mengembalikan nilai fungsi tertinggi. Oleh sebab itu digunakan algoritma ini untuk mencari titik di area berwarna merah tersebut.



Sebelum masuk kedalam langkah-langkah pembahasan algoritma, ada beberapa konstanta atau parameter yang harus diketahui, yaitu:
* Tentukan dimensi permasalahan dalam sebuah solusi
Diasumsikan dalam kasus ini, dimensi bernilai 2 karena ada 2 dimensi yang akan dicari solusinya yaitu x dan y

Const dimensi As Integer = 2

* Tentukan posisi minimal dan maksimal dari fungsi yang akan dihitung
Jika tidak ada batasan posisi, tentu saja posisi yang mendekati tak terhingga akan terpilih karena akan mengembalikan nilai fungsi yang sangat besar
Diasumsikan dalam kasus ini, posisi minimal adalah -2, dan posisi maksimal adalah +2

Const minPosisi As Double = -2
Const maksPosisi As Double = +2

* Tentukan jumlah iterasi yang digunakan dalam perhitungan
Diasumsikan dalam kasus ini, jumlah iterasi adalah 500 kali

Const jumlahIterasi As Integer = 500

* Tentukan jumlah individu yang digunakan
Diasumsikan dalam kasus ini, jumlah individu adalah 15 individu

Const jumlahIndividu As Integer = 15

* Tentukan persentase individu elit yang akan digunakan dalam proses perhitungan percepatan
Batas nilai yang digunakan adalah 1 sampai dengan jumlah individu
Diasumsikan dalam kasus ini, persentase individu elit adalah 2

Const persentaseIndividuElit As Integer = 2


Langkah-langkah penggunaan algoritma ini adalah

* Lakukan proses pencarian posisi terbaik
Penjelasan lebih detail tentang fungsi ini dapat dilihat pada penjelasan skrip dibawah ini (poin 1 – 2)

Dim posisiTerbaik() As Double = GSA(dimensi, minPosisi, maksPosisi, jumlahIterasi, jumlahIndividu, _
									persentaseIndividuElit)

Memasuki perhitungan pada fungsi GSA

* Inisialisasi individu yang digunakan sebanyak parameter jumlah individu

1. Lakukan perulangan sebanyak jumlah individu (poin 1a – 1b)

For i As Integer = 0 To jumlahIndividu - 1
. . .

1a. Beri posisi acak pada masing-masing individu sebanyak jumlah dimensi
Kemudian hitung nilai fungsi pada posisi tersebut
Penjelasan lebih detail tentang fungsi ini dapat dilihat pada penjelasan skrip dibawah ini

For k As Integer = 0 To dimensi - 1
	daftarIndividu(i).posisi(k) = (maksPosisi - minPosisi) * rnd.NextDouble() + minPosisi
Next k
daftarIndividu(i).nilaiFungsi = hitungNilaiFungsi(daftarIndividu(i).posisi)

* Gunakan fungsi ini untuk menghitung nilai fungsi yang diinginkan
Fungsi yang diketahui adalah fungsi Himmelblau, dengan rumus f(x, y) = (x^2+y-11)^2 + (x+y^2-7)^2

Public Function HitungNilaiFungsi(x1 As Double, y As Double) As Double
	Dim hasil As Double = Math.Pow(x1 * x1 + y - 11, 2) + Math.Pow(x1 + y * y - 7, 2)
	Return hasil
End Function

1b. Ambil posisi individu terbaik sebagai posisi terbaik sementara

If daftarIndividu(i).nilaiFungsi > nilaiFungsiTerbaik Then
	Array.Copy(daftarIndividu(i).posisi, PosisiTerbaik, dimensi)
	nilaiFungsiTerbaik = daftarIndividu(i).nilaiFungsi
End If

* Lakukan proses pencarian posisi terbaik (poin 2)

2. Lakukan proses perhitungan sebanyak jumlah iterasi (poin 2a – 2d)

Dim iterasi As Integer = 0
Do While iterasi < jumlahIterasi
	iterasi += 1
	. . .

2a. Lakukan perhitungan untuk mencari konstanta gravitasi yang digunakan
Rumus yang digunakan bisa diganti sesuai kebutuhan, dengan catatan bahwa
Semakin lama perulangan yang dilakukan, maka nilai konstanta gravitasi akan semakin turun mendekati 0

Const alpha As Double = 20
Const g0 As Double = 1
Dim G As Double = g0 * Math.Exp(-alpha * iterasi / jumlahIterasi)

* Memasuki proses perhitungan massa dari masing-masing individu (poin 2b)

2b1. Tentukan nilai fungsi terendah dan tertinggi dari daftar individu

Dim nilaiFungsiTertinggi As Double = Double.MinValue
Dim nilaiFungsiTerendah As Double = Double.MaxValue
For i As Integer = 0 To jumlahIndividu - 1
	If daftarIndividu(i).nilaiFungsi > nilaiFungsiTertinggi Then
		nilaiFungsiTertinggi = daftarIndividu(i).nilaiFungsi
	End If

	If daftarIndividu(i).nilaiFungsi < nilaiFungsiTerendah Then
		nilaiFungsiTerendah = daftarIndividu(i).nilaiFungsi
	End If
Next

2b2. Hitung nilai massa dengan rumus:
m = (fit-min)/(maks-min)
Kemudian lakukan normalisasi dengan cara membagi massa dengan semua total massa yang ada

Dim totalMassa As Double = 0
For i As Integer = 0 To jumlahIndividu - 1
	daftarIndividu(i).massa = (daftarIndividu(i).nilaiFungsi - nilaiFungsiTerendah) / (nilaiFungsiTertinggi - nilaiFungsiTerendah)
	totalMassa += daftarIndividu(i).massa
Next

For i As Integer = 0 To jumlahIndividu - 1
	daftarIndividu(i).massa /= totalMassa
Next

* Memasuki proses perhitungan percepatan di dalam ruang gravitasi (poin 2c)

2c1. Lakukan pengurutan pada daftar individu ini

Dim daftarIndividuTerurut() As Individu = daftarIndividu.Clone
Array.Sort(daftarIndividuTerurut)

2c2. Tentukan jumlah individu terpilih yang digunakan dalam daftar individu terurut
Nantinya masing-masing individu akan mengalami pergerakan menuju individu-individu terbaik
Semakin lama perulangan, maka jumlah individu terpilih akan semakin sedikit untuk mengurangi pergerakan dari masing-masing individu

Dim jumlahIndividuElit As Integer = persentaseIndividuElit + (1 - iterasi / jumlahIterasi) * (100 - persentaseIndividuElit)
jumlahIndividuElit = Math.Round(jumlahIndividu * jumlahIndividuElit / 100)

2c3. Lakukan perulangan pada masing-masing individu, dan bandingkan dengan individu terbaik sebanyak jumlah individu terpilih (poin 2c3a - 2c3b)

For i As Integer = 0 To jumlahIndividu - 1
	For idxTerurut As Integer = 0 To jumlahIndividuElit - 1
	. . .

2c3a. Hitung jarak Euclidean dari individu dan individu terpilih
Penjelasan tentang fungsi ini dapat dilihat pada penjelasan skrip dibawah ini

Dim jarakEuclidean As Double = hitungJarakEuclidean(daftarIndividu(i).posisi, daftarIndividuTerurut(idxTerurut).posisi, daftarIndividu(i).posisi.Length)

* Gunakan fungsi ini untuk menghitung jarak Euclidean antara 2 data

Private Function hitungJarakEuclidean(dataPertama As Double(), dataKedua As Double(), jumlahData As Integer) As Double
	If dataPertama.Length <> dataKedua.Length Then
		Throw New Exception("pada fungsi hitungJarakEuclidean, ukuran matriks data pertama tidak sama dengan ukuran matriks data kedua")
	End If

	Dim jumlah As Double = 0.0
	For i As Integer = 0 To jumlahData - 1
		Dim delta As Double = (dataPertama(i) - dataKedua(i)) * (dataPertama(i) - dataKedua(i))
		jumlah += delta
	Next

	If jumlah > 0 Then
		Return Math.Sqrt(jumlah)
	Else
		Return 0
	End If
End Function

2c3b. Jika jarak Euclidean lebih dari 0,
maka hitung total jarak Euclidean dari masing-masing dimensi dengan rumus:
E = E + nilai acak * m(idxTerurut) * (X(idxTerurut) - X(i)) / (jarak + eps)

If jarakEuclidean > 0 Then
	For j As Integer = 0 To dimensi - 1
		totalEuclidean(i)(j) += rnd.NextDouble * (daftarIndividuTerurut(idxTerurut).massa) * (daftarIndividuTerurut(idxTerurut).posisi(j) - daftarIndividu(i).posisi(j)) / (jarakEuclidean + 0.000001)
	Next
End If

2c4. Hitung percepatan dengan cara perkalian total jarak Euclidean dengan konstanta gravitasi

Dim percepatan(jumlahIndividu - 1)() As Double
For i As Integer = 0 To jumlahIndividu - 1
	percepatan(i) = New Double(dimensi - 1) {}

	For j As Integer = 0 To dimensi - 1
		percepatan(i)(j) = totalEuclidean(i)(j) * G
	Next
Next

2d. Lakukan perhitungan pada masing-masing dimensi dalam individu (poin 2d1 - 2d4)

For i As Integer = 0 To jumlahIndividu - 1
	For j As Integer = 0 To dimensi - 1
	. . .

2d1. Hitung kecepatan pergerakan individu dengan rumus:
v baru = v lama * nilai acak + percepatan
Kemudian lakukan perpindahan posisi yang baru dengan cara menjumlahkan posisi lama dengan kecepatan pergerakan

kecepatan(i)(j) = kecepatan(i)(j) * rnd.NextDouble + percepatan(i)(j)
daftarIndividu(i).posisi(j) += kecepatan(i)(j)

2d2. Jika posisi individu tersebut ternyata diluar batas posisi yang diperbolehkan,
maka kembalikan nilainya agar masuk dalam batas tersebut

If daftarIndividu(i).posisi(j) < minPosisi Then
	daftarIndividu(i).posisi(j) = minPosisi
ElseIf daftarIndividu(i).posisi(j) > maksPosisi Then
	daftarIndividu(i).posisi(j) = maksPosisi
End If

2d3. Hitung nilai fungsi pada individu tersebut

daftarIndividu(i).nilaiFungsi = hitungNilaiFungsi(daftarIndividu(i).posisi)

2d4. Jika nilai fungsi terbaik pada individu tersebut ternyata lebih baik dari nilai fungsi secara umum,
maka ambil posisi terbaik individu tersebut sebagai posisi terbaik

If daftarIndividu(i).nilaiFungsi > nilaiFungsiTerbaik Then
	Array.Copy(daftarIndividu(i).posisi, PosisiTerbaik, dimensi)
	nilaiFungsiTerbaik = daftarIndividu(i).nilaiFungsi

	Console.Write("Iterasi " & iterasi.ToString.PadRight(4) & ", ")
	Console.Write("individu " & i.ToString.PadRight(2) & ", ")
	Console.Write("Posisi sementara: [")
	For j As Integer = 0 To PosisiTerbaik.Length - 1
		Console.Write(PosisiTerbaik(j).ToString("F4").PadLeft(7) & " ")
	Next j
	Console.Write("], ")
	Console.WriteLine("nilai fungsi = " & nilaiFungsiTerbaik.ToString("F6"))
End If

* Agar dapat menjalankan skrip diatas, maka diperlukan sebuah Class Universe untuk menampung semua data posisi dan nilai fungsinya. Deklarasi Class Universe adalah sebagai berikut:

Public Class Individu
    Implements IComparable(Of Individu)
    Implements ICloneable

    Public posisi() As Double
    Public nilaiFungsi As Double
    Public massa As Double

    Public Sub New(ByVal dimensi As Integer)
        Me.posisi = New Double(dimensi - 1) {}
        Me.nilaiFungsi = 0.0
        Me.massa = 0.0
    End Sub
	
	. . .
End Class


Hasil akhir adalah: (klik untuk perbesar gambar)

cmd107


Contoh modul / source code dalam bahasa VB (Visual Basic) dapat didownload disini:

[sdm_download id="2818" fancy="0"]



Jika membutuhkan jasa kami dalam pembuatan program, keterangan selanjutnya dapat dilihat di Fasilitas dan Harga
Jika ada yang kurang paham dengan langkah-langkah algoritma diatas, silahkan berikan komentar Anda.
Selamat mencoba.

Comments

Leave a Reply

Your email address will not be published. Required fields are marked *