Pada kesempatan kali ini saya akan membagikan sedikit pengalaman saya dalam javaFX kepada pembaca semua yang mungkin bisa mengambil manfaat. Sekaligus mempopulerkan blog saya ini.
Yang saya akan ungkapkan kali ini adalah algoritma barnes-hut yang merupakan algoritma yang biasa digunakan dalam software-software simulasi fisika (semisal molekular dinamik dan semisalnya) [1].
Kali ini saya belum membahas tentang bagaimana algoritma tersebut digunakan untuk menghitung interaksi dalam simulasi. Akan tetapi yang saya lakukan adalah memberi gambaran dasar bagaimana algoritma tersebut bekerja. Tepatnya saya akan melakukan penggambaran animasi barnes-hut tree yang merupakan elemen inti dari algoritma ini.
Mengapa algoritma ini digunakan, adalah karena efisiensinya yang lebih bagus ketimbang skema brute-force yang biasa kita gunakan. Di mana kalo pake brute-force itu kompleksitasnya sampe O(N^2) maka dalam barnes-hut ini, kompleksitasnya adalah O(N log N).
Untuk source-nya dapat anda download di link berikut:
https://docs.google.com/file/d/0B1irLqfPwjq0YXA0emt2WWMtNkU/edit?usp=sharing
Referensi:
1. http://arborjs.org/docs/barnes-hut
Sesungguhnya shalat itu mencegah dari (perbuatan-perbuatan) keji dan mungkar
Q.S. Al-'Ankabut Ayat 45
Tuesday, September 3, 2013
Friday, August 23, 2013
Cara sortir di Java
Tanpa perlu basa-basi saya akan memberikan sedikit tutorial yakni bagaimana cara menyortir list di java. Seperti kita ketahui list itu sendiri merupakan suatu kumpulan objek. Untuk keperluan sortir sendiri, java sudah menyediakan fasilitas built-in ke kita. Yakni dengan menggunakan utilitas pada java Collections. Berdasarkan informasi yang saya peroleh, algoritma yang digunakan dalam java sorting ini adalah merge sort [1]. Tanpa perlu berbasa basi saudara tinggal menyalin contoh berikut dan langsung di running di komputer anda.
Jika sudah maka hasil yang akan ditampilkan adalah sebagai berikut:
1. http://www.coderanch.com/t/520171/java/java/Collections-sort-merge-sort-insteadof
Jika sudah maka hasil yang akan ditampilkan adalah sebagai berikut:
1. http://www.coderanch.com/t/520171/java/java/Collections-sort-merge-sort-insteadof
Tuesday, August 20, 2013
Animasi Bola Biliard dengan JavaFX
Kali ini saya akan memberikan sebuah tutorial mengenai bagaimana membuat animasi bola billiard dengan javaFX yang preview-nya dapat dilihat pada gambar berikut:
Adapun source codenya adalah sebagai berikut:
package fjr.collision; import java.util.ArrayList; import java.util.Random; import javafx.animation.KeyFrame; import javafx.animation.Timeline; import javafx.application.Application; import javafx.event.ActionEvent; import javafx.event.EventHandler; import javafx.scene.Group; import javafx.scene.Scene; import javafx.scene.control.Button; import javafx.scene.control.ButtonBuilder; import javafx.scene.shape.Circle; import javafx.scene.shape.CircleBuilder; import javafx.stage.Stage; import javafx.util.Duration; public class BilliardBallDetection extends Application { ArrayListlistParticle; ArrayList listCircle; double xwidth = 400; double ywidth = 400; double timeStep = 2.5; double numberParticle = 9; Random rand = new Random(); double dmin = 30.0; double xmax = xwidth; double xmin = 0.0; double ymax = ywidth; double ymin = 0.0; double size = 0.0; double dmax = 0.0; double vmax = 3.0; Timeline animation; Button buttonPause, buttonPlay; @Override public void start(Stage primarySrage) throws Exception { // TODO Auto-generated method stub generateParticle(); removeOverlap(listParticle); generateCircle(); Group root = new Group(); for(int i =0; i< listCircle.size(); i++){ Circle c = listCircle.get(i); root.getChildren().add(c); } animation = new Timeline(); animation.setCycleCount(Timeline.INDEFINITE); animation.setAutoReverse(false); KeyFrame kf = new KeyFrame(Duration.millis(20), new EventHandler (){ @Override public void handle(ActionEvent event) { // TODO Auto-generated method stub checkBoundary(); calculateCollision(); moveParticle(); redrawCircle(); } }); animation.getKeyFrames().add(kf); root.getChildren().addAll(buttonPlay = ButtonBuilder.create() .text("PLAY").onAction(new EventHandler () { @Override public void handle(ActionEvent arg0) { // TODO Auto-generated method stub animation.play(); } }).translateX(10).translateY(10) .build(), buttonPause = ButtonBuilder.create() .text("PAUSE").translateX(80).translateY(10) .onAction(new EventHandler () { @Override public void handle(ActionEvent arg0) { // TODO Auto-generated method stub animation.pause(); } }) .build() ); primarySrage.setScene(new Scene(root, xwidth, ywidth)); primarySrage.show(); } private void redrawCircle(){ for(int i=0; i< listCircle.size(); i++){ Circle c = listCircle.get(i); Particle p = listParticle.get(i); c.setCenterX(p.x); c.setCenterY(p.y); } } private void generateCircle(){ listCircle = new ArrayList<>(); for(int i=0; i< listParticle.size(); i++){ Particle p = listParticle.get(i); Circle c = CircleBuilder.create() .centerX(p.x).centerY(p.y).radius(dmin) .build(); listCircle.add(c); } } private void generateParticle() { listParticle = new ArrayList<>(); for (int i = 0; i < numberParticle; i++) { Particle particle = new Particle(rand.nextDouble() * xwidth, rand.nextDouble() * ywidth, rand.nextDouble() * vmax, rand.nextDouble() * vmax); listParticle.add(particle); } } private void calculateCollision(){ for(int i=0; i< listParticle.size(); i++){ Particle p1 = listParticle.get(i); for(int j= i+1; j< listParticle.size(); j++ ){ Particle p2 = listParticle.get(j); chekCollision(p1, p2); } } } public void chekCollision(Particle p1, Particle p2) { if (isOverlap(p1, p2)) { double deltaX = p1.x - p2.x; double deltaY = p1.y - p2.y; double distance = Math.sqrt(deltaX * deltaX + deltaY * deltaY); deltaX = deltaX/ distance; deltaY = deltaY/distance; double aci = p1.vx * deltaX + p1.vy * deltaY; double bci = p2.vx * deltaX + p2.vy * deltaY; double acf = bci; double bcf = aci; p1.vx = p1.vx + (acf - aci) * deltaX; p1.vy = p1.vy + (acf - aci) * deltaY; p2.vx = p2.vx + (bcf - bci) * deltaX; p2.vy = p2.vy + (bcf - bci) * deltaY; } } private void moveParticle() { for (int i = 0; i < listParticle.size(); i++) { Particle p = listParticle.get(i); p.x += p.vx * timeStep; p.y += p.vy * timeStep; } } public void removeOverlap(ArrayList list) { for (int i = 0; i < list.size(); i++) { Particle p1 = list.get(i); for (int j = list.size() - 1; j > i; j--) { Particle p2 = list.get(j); if(isOverlap(p1, p2)) list.remove(j); } } } public void checkBoundary(){ for(int i =0; i< listParticle.size() ;i++){ Particle p = listParticle.get(i); if(p.x > xmax - dmin) p.vx = -Math.abs(p.vx) ; if(p.x < xmin + dmin) p.vx = Math.abs(p.vx); if(p.y > ymax - dmin) p.vy = - Math.abs(p.vy); if(p.y < ymin + dmin) p.vy = Math.abs(p.vy); } } private boolean isOverlap(Particle p1, Particle p2){ double deltax = p1.x - p2.x; double deltay = p1.y - p2.y; if(deltax * deltax + deltay * deltay < 4 * dmin * dmin) return true; return false; } public static void main(String[] args) { launch(args); } private class Particle { double x = 0.0; double y = 0.0; double vx = 0.0; double vy = 0.0; public Particle(double x, double y , double vx , double vy ) { this.x = x; this.y = y; this.vx = vx ; this.vy = vy; } } }
Monday, August 19, 2013
Deteksi tumbukan dengan javaFX
Pada tutorial ini saya akan memberikan bagaimana cara mendeteksti tumbukan dengan javaFX yang previewnya dapat dilihat pada gambar berikut:
Adapaun source codenya adalah:
package fjr.collision; import java.util.Random; import javafx.application.Application; import javafx.collections.ObservableList; import javafx.event.ActionEvent; import javafx.event.EventHandler; import javafx.scene.Group; import javafx.scene.Node; import javafx.scene.Scene; import javafx.scene.control.Button; import javafx.scene.control.ButtonBuilder; import javafx.scene.shape.Circle; import javafx.scene.shape.CircleBuilder; import javafx.stage.Stage; public class TestCollision extends Application { double r = 14.0; Random rand = new Random(); double xwidth = 400; double ywidth = 400; int numberParticle = 100; // uji tumbukan public boolean testCol(Circle p1, Circle p2) { double dx = p1.getCenterX() - p2.getCenterX(); double dy = p1.getCenterY() - p2.getCenterY(); if (dx * dx + dy * dy < 4 * r * r) { return true; } return false; } public void removeOverlap(ObservableListlist) { for (int i = 0 ; i< list.size() ; i++ ) { if (list.get(i) instanceof Circle) { Circle p1 = (Circle) list.get(i); for (int j = list.size()- 1 ; j > i ; j-- ) { if (list.get(j) instanceof Circle) { Circle p2 = (Circle) list.get(j); if (testCol(p1, p2)) { list.remove(j); } } } } } } public static void main(String[] args) { launch(args); } Button button, button1; boolean state = true; @Override public void start(Stage primaryStage) throws Exception { final Group root = new Group(); primaryStage.setScene(new Scene(root, xwidth, ywidth)); for (int i = 0; i < numberParticle; i++) { root.getChildren().add( CircleBuilder.create().centerX(rand.nextDouble() * xwidth) .centerY(rand.nextDouble() * ywidth).radius(r) .build()); } button = ButtonBuilder.create().text("ORGANIZE").translateX(10) .translateY(10).build(); button1 = ButtonBuilder.create().text("RESET").translateX(100) .translateY(10).build(); root.getChildren().addAll(button, button1); button.setOnAction(new EventHandler () { @Override public void handle(ActionEvent arg0) { if (state) { removeOverlap(root.getChildren()); } state = false; } }); button1.setOnAction(new EventHandler () { @Override public void handle(ActionEvent arg0) { for (int i = root.getChildren().size() - 1; i >= 0; i--) { root.getChildren().remove(i); } for (int i = 0; i < numberParticle; i++) { root.getChildren().add( CircleBuilder.create() .centerX(rand.nextDouble() * xwidth) .centerY(rand.nextDouble() * ywidth) .radius(r).build()); } root.getChildren().addAll(button, button1); state = true; } }); primaryStage.show(); } }
Sunday, August 18, 2013
Export gambar ke kode Latex
Seperti kita ketahui bahwa menggambar di Latex itu biasanya menggunakan beberapa library, di mana dua di antaranya yang paling populer adalah tikZ dan pstrick. Dalam menggunakan kedua librari ini otomatis kita harus menuliskan kode-kodenya. Masalahnya adalah kode-kode ini tidak mudah dihapal, sehingga sangat menyulitkan bagi kita yang tidak punya banyak waktu untuk keperluan ini.
Oleh karena itu beberapa developer sudah menyediakan fasilitas untuk mengeksport gambar yang dibuat tool-tool gui yang ada ke kode Latex. Beberapa software yang punya kapasitas ini antara lain, adalah:
1. Dia http://dia-installer.de/ : mempunyai banyak fasilitas dan built-in library untuk menggambar diagram atau sirkuit elektronik. Dan hasilnya bisa dieksport ke kode Latex:
2. Inkscape, http://inkscape.org/ :
Akan tetapi untuk inkscape ini, yang versi 0.48 hanya tersedia untuk pstrick. Masalahnya adalah pstrick sendiri kurang portable jika dibandingkan dengan tikZ.
3. MatplotLib (http://matplotlib.org/users/whats_new.html#pgf-tikz-backend). Yang terakhir ini merupakan library dari python untuk keperluan plot.
Wednesday, July 17, 2013
Jawaban soal nomor dua pelatihan MATLAB
Bagi teman-teman sekalian yang mungkin belum mengetahui bagaimana jawaban soal nomor dua, berikut saya berikan listingnya. Silakan dicoba sendiri di rumah. Semoga bermanfaat
% jawaban nomor dua function plot_bolak_balik clc; f = figure('menubar', 'none'); x = linspace(45 ,405,100); y1 = cosd(x); y2 = sind(x); plot(x, y1, 'b.- ', x, y2, 'g.-'); % untuk menjaga sumbunya konstan hold on ; i = 1; state = true; while ishandle(f), if state p = plot(x(i), y1(i), 'r.' , 'Markersize' ,30); if(i == numel(x) ), state = false; i = i - 1; end i = i+ 1; pause(0.02); if ishandle(p),delete(p); end else p = plot(x(i), y2(i), 'r.' , 'Markersize' ,30); if(i == 1 ), state = true ; i = i+ 1; end i = i- 1; pause(0.02); if ishandle(p),delete(p); end end endAdapun hasil cuplikan animasinya adalah sebagai berikut
Hubungan ukuran array dengan performa indeks di MATLAB
Posting kali ini masih berhubungan dengan posting saya sebelumnya. Cuma pada bagian ini saya akan memperlihatkan bagaimana hubungan antara ukuran array dengan lama eksekusi dengan metode indeks. Ternyata hasil yang diperoleh meunjukkan bahwa dengan menggunakan looping di MATLAB, waktu eksekusi yang diberikan bertambah secara signifikan dan cenderung eksponensial. Sementara dengan bantuan indeks, waktu eksekusi datar-datar saja. Adapun codenya bisa dilihat pada listing berikut
function indeks1 clear all ; clc; step = 100:50:700; cara1 = zeros(length(step),1); cara2 = zeros(length(step),1); for ii=1:numel(step) % first way, iterate trough looping dim = step(ii); Y = zeros(dim,dim) ; tic ; for i=1:length(Y), for j=1:length(Y(1,:)), if mod(i+j, 2) == 0 Y(i,j) = 2; else Y(i,j) = 1; end end end cara1(ii) = toc; % second way, using MATLAB index X = zeros(dim,dim) ; tic; [a,b] = ind2sub(size(X), 1:numel(X)) ; X((mod(a+b, 2) ==0)) = 2; X((mod(a+b, 2) > 0)) = 1; cara2(ii) = toc; end plot(step, cara1, 'r.-', step, cara2, 'g.-' ) disp('Is the first get the same output with the second?') indeks = find(Y(:,:) ~= X(:,:)); xlabel('ukuran array'); ylabel('lama eksekusi dalam detik'); if(numel(indeks) == 0), disp('oh... yes.. '); endDi mana saya sudah melakukan perubahan time step dalam rentang yang berbeda-beda. Pada hasilnya juga diperlihatkan bahwa penggunaan indeks ini sangat terasa ketika ukuran array semakin besar.
Anda bisa mengetes sendiri untuk selang waktu yang lain. Tentunya MATLAB membatasi kita dalam mengalokasikan array, mengingat kapasitas memori yang terbatas. Untuk mengetahui kapasitas memori di komputer anda, matlab menyediakan perintah
memory
yang bisa digunakan untuk memperkirakan kapasitas array yang bisa anda alokasikan. Performa indeks di MATLAB bagian 2
Pada kesempatan kali ini sekali lagi saya menekankan pentingnya menghindari looping dalam pemrograman MATLAB. Kebetulan ini merupakan soal yang saya berikan dalam pelatihan MATLAB kemarin kepada teman-teman. Saya g tau apa ini bisa dikerjakan atau tidak, yang jelas saya kasikan aja jawabannya, sekalian berbagi pengetahuan. Semoga bisa diambil manfaatnya.
Soalnya adalah diberikan sebuah matriks yang ukurannya 1000 x 1000. Kenapa 1000 x 1000, karena biar kita terbiasa dengan perhitungan sesungguhnya di lapangan. Pada matriks ini, sederhana saja yang dilakukan yakni, jika penjumlahan baris dan kolom genap, isi matriks tersebut dengan angka dua, sementara jika ganjil, isi dengan 1. Adapaun hasil codingnya adalah sebagai berikut.
Jadi terlihat jelas bahwa performa indeks 10 kali lipat lebih cepat ketimbang pake looping.
Soalnya adalah diberikan sebuah matriks yang ukurannya 1000 x 1000. Kenapa 1000 x 1000, karena biar kita terbiasa dengan perhitungan sesungguhnya di lapangan. Pada matriks ini, sederhana saja yang dilakukan yakni, jika penjumlahan baris dan kolom genap, isi matriks tersebut dengan angka dua, sementara jika ganjil, isi dengan 1. Adapaun hasil codingnya adalah sebagai berikut.
Jadi terlihat jelas bahwa performa indeks 10 kali lipat lebih cepat ketimbang pake looping.
Sunday, July 14, 2013
Rut-etra-izer versi JavaFX
Pada tutorial kali ini saya akan mendemonstrasikan salah satu kegunaan dari JavaFX yakni dalam melakukan translate terhadap salah satu aplikasi berbasis javascript yakni Rut-etra-izer kedalam bahasa java. Kebetulan saya agak penasaran dengan kecanggihan aplikasi ini. Saya baru tahu soalnya dan saya harap pembaca bisa mengambil manfaat dari tutorial ini. Adapun hasil mentahnya dapat dilihat pada gambar berikut:
Maksud saya adalah saya pengen berbagi tutorial aja. Semoga bermanfaat!
package fjr.main.rutte.traizer; import javafx.application.Application; import javafx.scene.Group; import javafx.scene.Scene; import javafx.scene.canvas.Canvas; import javafx.scene.canvas.GraphicsContext; import javafx.scene.image.Image; import javafx.scene.image.ImageView; import javafx.scene.image.PixelReader; import javafx.scene.layout.StackPane; import javafx.scene.paint.Color; import javafx.scene.shape.LineTo; import javafx.scene.shape.MoveTo; import javafx.scene.shape.Path; import javafx.scene.shape.PathBuilder; import javafx.scene.shape.QuadCurve; import javafx.scene.transform.Affine; import javafx.scene.transform.Rotate; import javafx.stage.Stage; public class Rutte extends Application { static String main = Rutte.class.getResource("mami.jpg").toExternalForm(); int step = 5; int stepY = 50; double depth = 56; double width = 0 ; double height = 0; @Override public void start(Stage primaryStage) throws Exception { Group root = getEsembel(); primaryStage.setScene(new Scene(root)); primaryStage.setTitle("TUT WURI HANDAYANI"); primaryStage.show(); } public Group getEsembel() { Group root = new Group(); Image image = new Image(main); width = image.getWidth(); height = image.getHeight(); PixelReader pixel = image.getPixelReader(); double originX = 0; double destinyX = 0; double originY = 100; double destinyY = 0.0; Canvas canvas = new Canvas(width+50,height+50); GraphicsContext gc = canvas.getGraphicsContext2D(); double shifty =0; for(int y = 0 ; y < height; y+= step){ for(int x = 0; x < width-step ;x+= step){ Color color = pixel.getColor(x, y); Color color1 = pixel.getColor(x+step, y); double brightness = color.getBrightness(); double brightness1 = color1.getBrightness(); originX = x ; destinyX = x+step; originY = - brightness*depth+brightness/2 + shifty; destinyY = - brightness1*depth+brightness1/2+ shifty; gc.setStroke(color); this.rotateX(gc, -30); gc.strokeLine(originX,originY, destinyX, destinyY ); originX = destinyX ; originY = destinyY; } shifty+= 5; } root.setTranslateX(20); root.setTranslateY(20); root.getChildren().add(canvas); return root; } /* * I'm not found the way to rotate GraphicContext with simple way like other node */ static void rotateX(GraphicsContext gc , double angle){ Affine affine = new Affine(); double angleRadi = Math.toRadians(angle); double sinus = Math.sin(angleRadi); double cosinus = Math.cos(angleRadi); affine.setMxx(1); affine.setMxy(0); affine.setMxz(0); affine.setTx(0); affine.setMyx(0); affine.setMyy(cosinus); affine.setMyz(-sinus); affine.setTy(0); affine.setMzx(0); affine.setMzy(sinus); affine.setMzz(cosinus); affine.setTz(0); gc.setTransform(affine); } static void rotateY(GraphicsContext gc , double angle){ Affine affine = new Affine(); double angleRadi = Math.toRadians(angle); double sinus = Math.sin(angleRadi); double cosinus = Math.cos(angleRadi); affine.setMxx(cosinus); affine.setMxy(0); affine.setMxz(sinus); affine.setTx(0); affine.setMyx(0); affine.setMyy(1); affine.setMyz(0); affine.setTy(0); affine.setMzx(-sinus); affine.setMzy(0); affine.setMzz(-cosinus); affine.setTz(0); gc.setTransform(affine); } static void rotateZ(GraphicsContext gc , double angle){ Affine affine = new Affine(); double angleRadi = Math.toRadians(angle); double sinus = Math.sin(angleRadi); double cosinus = Math.cos(angleRadi); affine.setMxx(cosinus); affine.setMxy(-sinus); affine.setMxz(0); affine.setTx(0); affine.setMyx(sinus); affine.setMyy(cosinus); affine.setMyz(0); affine.setTy(0); affine.setMzx(0); affine.setMzy(0); affine.setMzz(1); affine.setTz(0); gc.setTransform(affine); } public Group getLine(double y) { Group root = new Group(); return root; } public static void main(String[] args) { launch(args); } }Sebenarnya masih ada beberapa bagian yang ingin saya sempurnakan, misalnya respon terhadap mouse, drag n drop terhadap foto serta menyimpan hasil ke dalam gambar JPEG. Tapi biar g terlalu basi, mendingan dipublish aja versi BETA-nya. Itupun kalo layak disebut BETA.
Maksud saya adalah saya pengen berbagi tutorial aja. Semoga bermanfaat!
Tuesday, July 9, 2013
Performa indeks dalam mengganti logika if di MATLAB
Posting ini berkaitan dengan posting saya sebelumnya yakni tentang vektorisasi molekular dinamik di MATLAB:
yang sampai saat ini masih menyisakan pertanyaan besar di kepala saya. Antara lain, mengapa vektorisasi justru membutuhkan waktu lebih lama, serta yang kedua, entah kenapa simulasinya g sesuai dengan yang diharapkan: terjadi kolaps setelah waktu tertentu dan perpindahan posisi yang cukup lambat ketimbang companion-nya yang dibuat di java [1]. Dalam menyelidiki masalah ini saya sekali lagi pengen menunjukin kenapa vektorisasi itu penting di MATLAB dan sebisa mungkin kita harus menghindari loop dalam menulis program di MATLAB.
Oleh karena itu saudara bisa menyimak listing berikut
Terlihat jelas bahwa penggunaan indeks di MATLAB bisa digunakan untuk mengganti logika if pada looping. Lantas bagaimana dengan performanya. Pada listing berikut ini telah saya ukur bagaimana performa dari indeks sendiri.
Yang mana menunjukkan bahwa dengan menggunakan indeks kita bisa mengurangi waktu eksekusi program hingga sampai setengahnya. Hal ini jelas disebabkan oleh perintah indeks sendiri ditangani secara 'native' oleh MATLAB interpreter.
Fakta kedua yang bisa digaris bawahi adalah ternyata dua fungsi dalam satu file pada MATLAB alokasi eksekusinya ditempatkan pada core processor yang berbeda seperti sudah saya komentari pada listing di atas.
Oleh karena itu saudara bisa menyimak listing berikut
Terlihat jelas bahwa penggunaan indeks di MATLAB bisa digunakan untuk mengganti logika if pada looping. Lantas bagaimana dengan performanya. Pada listing berikut ini telah saya ukur bagaimana performa dari indeks sendiri.
Yang mana menunjukkan bahwa dengan menggunakan indeks kita bisa mengurangi waktu eksekusi program hingga sampai setengahnya. Hal ini jelas disebabkan oleh perintah indeks sendiri ditangani secara 'native' oleh MATLAB interpreter.
Fakta kedua yang bisa digaris bawahi adalah ternyata dua fungsi dalam satu file pada MATLAB alokasi eksekusinya ditempatkan pada core processor yang berbeda seperti sudah saya komentari pada listing di atas.
1. Ternyata perintah
axis manual
di MATLAB hanya berlaku ketika ada perintah hold on
.
Subscribe to:
Posts (Atom)