Accueil > > > TRANSFORMÉE DE HOUGH
TRANSFORMÉE DE HOUGH
Information sur la source
Description
Ce petit code permet d'éffectuer la transformée de Hough sur une image. Il prend en entrée une image du type (png / jpeg ou gif) qui est séléctionnée au début du programme Cette image ne doit contenir que le contour des formes (cf "filetre de sobel" par exemple). Il crée deux images. 1 contenant tout les pixel luent, l'autres contenant les pixels luent + la droite dominante de l'image trouvée. Plusieurs améliorations peuvent etre apportée surtout dans la recherche du / des maximas. Il est possible également de l'étendre à plusieurs droites dans l'image ...
Source
- package houghTransform;
-
- import java.awt.Canvas;
- import java.awt.Color;
- import java.awt.Graphics;
- import java.awt.Image;
- import java.awt.image.BufferedImage;
- import java.io.FileNotFoundException;
- import java.io.FileOutputStream;
- import java.io.OutputStream;
-
- import javax.imageio.ImageIO;
- import javax.swing.JFileChooser;
-
- /**
- * @author Zolotaya
- */
- public class Hough {
-
- int[][] accu;
- static int maxRho;
-
- /** Constructeur */
- public Hough() {
- // TODO Auto-generated constructor stub
- JFileChooser choix = new JFileChooser();
- if (choix.showSaveDialog(null) == JFileChooser.APPROVE_OPTION) {
- String filename = choix.getSelectedFile().getAbsolutePath();
- try {
- BufferedImage bi = loadImage(filename);
- Applyhough(bi);
- } catch (Exception e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
-
- /** Fonction d'application de la transformée de Hough */
- public void Applyhough(BufferedImage bi){
- int compt = 0;
- int maxRho = (int)(Math.sqrt( (bi.getWidth()*bi.getWidth()) + (bi.getHeight()*bi.getHeight()) ) + 0.5);
- Hough.maxRho = maxRho;
- // création du "conteneur" de données
- accu = new int [360][2*maxRho];
- for (int i = 0; i < accu.length ; i++){
- for (int j = 0 ; j < accu[i].length ; j++){
- accu[i][j] = 0;
- }
- }
- // création d'une image avec en bleu les pixels luent, en rouge les droites trouvée (cf plus loin)
- BufferedImage dst = new BufferedImage(bi.getWidth(), bi.getHeight(), BufferedImage.TYPE_INT_ARGB);
- // On parcour toute l'image
- for (int x = 0 ; x < bi.getWidth() ; x++) {
- for (int y = 0 ; y < bi.getHeight() ; y++){
- // Si le pixel n'appartient pas au fond (n'est pas noir)
- int rgb = bi.getRGB(x, y);
- rgb = rgb & 0xFF;
- if (rgb > 0){
- for (int angle = 0 ; angle < 360 ; angle ++){
- double theta = Math.toRadians(angle);
- double rho = x*Math.cos(theta) + y*Math.sin(theta);
- int indexAngle = (int) (angle);
- int indexRho = (int)(rho+maxRho + 0.5);
- accu[indexAngle][indexRho]++;
- }
- dst.setRGB(x, y, rgb);
- compt++;
- }
- }
- }
-
- try {
- SaveImage(dst, new FileOutputStream("PixelRead.png"));
- } catch (FileNotFoundException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (Exception e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
-
- int[] droite = maxi();
-
- try {
- dst = tracerDroite(droite, dst);
- SaveImage(dst, new FileOutputStream("droite.png"));
- } catch (FileNotFoundException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (Exception e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
-
- /** Fonction de recherche des maximums */
- public int[] maxi(){
- int maxi = 0;
- int[] droite = new int[3];
- // On parcours tout l'accumulateur pour trouver le maximum
- for (int i = 0 ; i < accu.length ; i++){
- for (int j = 0 ; j < accu[i].length ; j++){
- if ( maxi < accu[i][j]){
- maxi = accu[i][j];
- droite[0] = i; // angle
- droite[1] = j; // distance
- droite[2] = maxi; // peut servir pour la recherche des maximas suivant
- }
- }
- }
- return droite;
- }
-
- /** Fonction de tracage des droites + calcul de leur équation */
- private static BufferedImage tracerDroite(int[] droite, BufferedImage src){
-
- int angle = droite[0];
- int rho = droite[1] - maxRho;
-
- // droite de la forme ax +by + c = 0 -->
- // si b != 0 y = (-ax + c)/b
- // sinon droite verticale
- double a = Math.cos(Math.toRadians(angle));
- double b = Math.sin(Math.toRadians(angle));
- double c = rho;
-
- // On fixe 2 nombre :
- int x0 = -src.getWidth();
- int x1 = src.getWidth();
-
- double y0 , y1;
- // On calcul les deux droites :
- // SI la droite n'est pas verticale
- if (b != 0){
- y0 = (-a*x0 + c)/b;
- y1 = (-a*x1 + c)/b;
- // On affiche l'équation de la droite
- System.out.println("droite : y = "+a+"*x + "+b );
- // On trace les droites trouvées
- Graphics g = src.getGraphics();
- g.setColor(Color.red);
- g.drawLine(x0, (int)(y0 + 0.5), x1, (int)(y1 + 0.5));
- } else {
- x0 = rho;
- y0 = 0;
- x1 = rho;
- y1 = src.getHeight();
- // On affiche l'équation de la droite
- System.out.println("droite : y = "+a+"*x + "+b );
- // On trace les droites trouvées
- Graphics g = src.getGraphics();
- g.setColor(Color.red);
- g.drawLine(x0, (int)(y0 + 0.5), x1, (int)(y1 + 0.5));
- }
- // On retourne l'image source
- return src;
- }
-
- /** Fonction de sauvegarde d'image */
- public static void SaveImage(BufferedImage imBW, OutputStream os) throws Exception {
- Graphics g = imBW.createGraphics();
- g.drawImage(imBW, 0, 0, null);
- g.dispose();
- ImageIO.write(imBW, "png", os);
- os.close();
- imBW = null;
- os = null;
- }
-
- // Fonction d'ouverture des images (.jpeg / .gif / .png)
- public BufferedImage loadImage(String f) throws Exception {
- Image im2 = null;
- java.awt.MediaTracker mt2 = null;
-
- java.io.FileInputStream in = null;
- byte[] b = null;
- int size = 0;
-
- in = new java.io.FileInputStream(f);
- if (in != null) {
- size = in.available();
- b = new byte[size];
- in.read(b);
- im2 = java.awt.Toolkit.getDefaultToolkit().createImage(b);
- in.close();
- }
-
- mt2 = new java.awt.MediaTracker(new Canvas());
- if (im2 != null) {
- if (mt2 != null) {
- mt2.addImage(im2, 0);
- mt2.waitForID(0);
- }
- }
- BufferedImage input = new BufferedImage(im2.getWidth(null), im2
- .getHeight(null), BufferedImage.TYPE_INT_ARGB);
- Graphics g = input.createGraphics();
- g.setColor(Color.white);
- g.fillRect(0, 0, im2.getWidth(null), im2.getHeight(null));
- g.drawImage(im2, 0, 0, null);
- g.dispose();
- g = null;
- return input;
- }
-
- /** Point d'entrée */
- public static void main(String[] args) {
- // TODO Auto-generated method stub
- new Hough();
- }
-
- }
package houghTransform;
import java.awt.Canvas;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.image.BufferedImage;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.OutputStream;
import javax.imageio.ImageIO;
import javax.swing.JFileChooser;
/**
* @author Zolotaya
*/
public class Hough {
int[][] accu;
static int maxRho;
/** Constructeur */
public Hough() {
// TODO Auto-generated constructor stub
JFileChooser choix = new JFileChooser();
if (choix.showSaveDialog(null) == JFileChooser.APPROVE_OPTION) {
String filename = choix.getSelectedFile().getAbsolutePath();
try {
BufferedImage bi = loadImage(filename);
Applyhough(bi);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
/** Fonction d'application de la transformée de Hough */
public void Applyhough(BufferedImage bi){
int compt = 0;
int maxRho = (int)(Math.sqrt( (bi.getWidth()*bi.getWidth()) + (bi.getHeight()*bi.getHeight()) ) + 0.5);
Hough.maxRho = maxRho;
// création du "conteneur" de données
accu = new int [360][2*maxRho];
for (int i = 0; i < accu.length ; i++){
for (int j = 0 ; j < accu[i].length ; j++){
accu[i][j] = 0;
}
}
// création d'une image avec en bleu les pixels luent, en rouge les droites trouvée (cf plus loin)
BufferedImage dst = new BufferedImage(bi.getWidth(), bi.getHeight(), BufferedImage.TYPE_INT_ARGB);
// On parcour toute l'image
for (int x = 0 ; x < bi.getWidth() ; x++) {
for (int y = 0 ; y < bi.getHeight() ; y++){
// Si le pixel n'appartient pas au fond (n'est pas noir)
int rgb = bi.getRGB(x, y);
rgb = rgb & 0xFF;
if (rgb > 0){
for (int angle = 0 ; angle < 360 ; angle ++){
double theta = Math.toRadians(angle);
double rho = x*Math.cos(theta) + y*Math.sin(theta);
int indexAngle = (int) (angle);
int indexRho = (int)(rho+maxRho + 0.5);
accu[indexAngle][indexRho]++;
}
dst.setRGB(x, y, rgb);
compt++;
}
}
}
try {
SaveImage(dst, new FileOutputStream("PixelRead.png"));
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
int[] droite = maxi();
try {
dst = tracerDroite(droite, dst);
SaveImage(dst, new FileOutputStream("droite.png"));
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
/** Fonction de recherche des maximums */
public int[] maxi(){
int maxi = 0;
int[] droite = new int[3];
// On parcours tout l'accumulateur pour trouver le maximum
for (int i = 0 ; i < accu.length ; i++){
for (int j = 0 ; j < accu[i].length ; j++){
if ( maxi < accu[i][j]){
maxi = accu[i][j];
droite[0] = i; // angle
droite[1] = j; // distance
droite[2] = maxi; // peut servir pour la recherche des maximas suivant
}
}
}
return droite;
}
/** Fonction de tracage des droites + calcul de leur équation */
private static BufferedImage tracerDroite(int[] droite, BufferedImage src){
int angle = droite[0];
int rho = droite[1] - maxRho;
// droite de la forme ax +by + c = 0 -->
// si b != 0 y = (-ax + c)/b
// sinon droite verticale
double a = Math.cos(Math.toRadians(angle));
double b = Math.sin(Math.toRadians(angle));
double c = rho;
// On fixe 2 nombre :
int x0 = -src.getWidth();
int x1 = src.getWidth();
double y0 , y1;
// On calcul les deux droites :
// SI la droite n'est pas verticale
if (b != 0){
y0 = (-a*x0 + c)/b;
y1 = (-a*x1 + c)/b;
// On affiche l'équation de la droite
System.out.println("droite : y = "+a+"*x + "+b );
// On trace les droites trouvées
Graphics g = src.getGraphics();
g.setColor(Color.red);
g.drawLine(x0, (int)(y0 + 0.5), x1, (int)(y1 + 0.5));
} else {
x0 = rho;
y0 = 0;
x1 = rho;
y1 = src.getHeight();
// On affiche l'équation de la droite
System.out.println("droite : y = "+a+"*x + "+b );
// On trace les droites trouvées
Graphics g = src.getGraphics();
g.setColor(Color.red);
g.drawLine(x0, (int)(y0 + 0.5), x1, (int)(y1 + 0.5));
}
// On retourne l'image source
return src;
}
/** Fonction de sauvegarde d'image */
public static void SaveImage(BufferedImage imBW, OutputStream os) throws Exception {
Graphics g = imBW.createGraphics();
g.drawImage(imBW, 0, 0, null);
g.dispose();
ImageIO.write(imBW, "png", os);
os.close();
imBW = null;
os = null;
}
// Fonction d'ouverture des images (.jpeg / .gif / .png)
public BufferedImage loadImage(String f) throws Exception {
Image im2 = null;
java.awt.MediaTracker mt2 = null;
java.io.FileInputStream in = null;
byte[] b = null;
int size = 0;
in = new java.io.FileInputStream(f);
if (in != null) {
size = in.available();
b = new byte[size];
in.read(b);
im2 = java.awt.Toolkit.getDefaultToolkit().createImage(b);
in.close();
}
mt2 = new java.awt.MediaTracker(new Canvas());
if (im2 != null) {
if (mt2 != null) {
mt2.addImage(im2, 0);
mt2.waitForID(0);
}
}
BufferedImage input = new BufferedImage(im2.getWidth(null), im2
.getHeight(null), BufferedImage.TYPE_INT_ARGB);
Graphics g = input.createGraphics();
g.setColor(Color.white);
g.fillRect(0, 0, im2.getWidth(null), im2.getHeight(null));
g.drawImage(im2, 0, 0, null);
g.dispose();
g = null;
return input;
}
/** Point d'entrée */
public static void main(String[] args) {
// TODO Auto-generated method stub
new Hough();
}
}
Conclusion
J'essairais de rejouter d'autres options (quand j'aurais le temps) sur la détéction des contours d'une image, améliorations de la séléctions des maximas, séléction du nombre de droites... En gros essayer de faire un petit soft sympa ;-).
Sources de la même categorie
Commentaires et avis
Discussions en rapport avec ce code source dans le forum
Image instable, clignotante... [ par mleloc ]
Bonjour, j'ai créé une applet qui transforme une image grace a l'ondelette de Haar puis affiche sa transformée.Malheureusement, lorsque
transformée de hough en langage C [ par janouta ]
Bonjour, je suis une étudiante en train de préparer mon projet de fin d'étude , je débute en traitement d'image et je dois réaliser un analyseur de co
Trouver un cercle dans une image !! [ par kenada56 ]
Slt !! Je cherche un moyen permettant de détecter un cercle dans une image. J'ai essayé avec le Transformé de Hough, mais je n'ai pa r&
Détection du mammelon à partir du contour du sein [ par algeroc87 ]
Bonjour, D'abord j'espère que vous allez bien toute l'équipe parceque ça fait un moment que j'ai pas trainé ici En faite je voulais juste avoir des
centre de contour d'image [ par takwatouma ]
Bonjourj'ai testé un code de contour actif.je veux detecter centre de gravité de ce contour comment je peux acceder au different point de contour.et q
La transformée de Hough [ par dh11 ]
Salut, Je suis une étudiante débutante en java et je suis en train de faire une application :détection des visages dans une image et je veux utilise
[clos] detection de contour d'image avec java [ par mmlm ]
bonjour Je veut obtenir le code source de détection de contour d'image a travers une boutton avec java . MERCI
detection de contour d'image avec java [ par mmlm ]
Je souhaite afficher le contour d'une images avec une application java. Sur le site je n'ai trouvé que des exemples avec c++ or je dois utiliser JAVA
Afficher une image aleatoire et deplacable avec la souris sur une Applet java [ par liondes ]
salut tout le monde SVP je suit débutant en java si vous pouvez m'aidez n’hésiter pas mon problème concerne le chargement d'une image [color=bl
cacher une image dans un fichier wav [ par hajerboug ]
Bonjour, Est il possible de cacher une image dans un fichier wav en utilisant la méthode du LSB. La représentation binaire d'un fichier wav me semble
|
Derniers Blogs
GESTION D'EXCEPTION AVEC LES TASKSGESTION D'EXCEPTION AVEC LES TASKS par richardc
Nous avons vu dans un précédent article comment utiliser Task pour effectuer des opérations dans un autre thread.
Malheureusement, comme tout le monde n'est pas parfait, il se peut que cette exécution se passe mal et qu'une exception se produise.
La...
Cliquez pour lire la suite de l'article par richardc DéMARRONS AVEC LES TASKSDéMARRONS AVEC LES TASKS par richardc
Que vous le vouliez ou non, le développement multi-tâche est maintenant une obligation pour toute nouvelle application. Il est donc vital d'en comprendre les mécanismes et de s'y mettre le plus tôt possible.
En attendant le .NET Framework 4.5 avec le...
Cliquez pour lire la suite de l'article par richardc SLIDE & DéMO TECHDAYS 2012 - FAST & FURIOUS XAML APPSSLIDE & DéMO TECHDAYS 2012 - FAST & FURIOUS XAML APPS par Vko
Retrouvez les slides et les démo de ma session Fast & Furious XAML Apps. A ceux qui se posent la question : "est-ce que le code de la DataGrid est disponible?", je vous répondrais "pas encore". Je vais mettre en place un projet codeplex pour part...
Cliquez pour lire la suite de l'article par Vko XNA IS DEAD!XNA IS DEAD! par richardc
Depuis la semaine dernière (et grâce aux TechDays 2012), je me penche activement sur la nouvelle version de Windows, aka Windows 8. Vous me direz, il était temps puisque la première preview date de Septembre dernier.
OK. Remarquez, on n'en est qu'aux...
Cliquez pour lire la suite de l'article par richardc TECHDAYS PARIS 2012 : WINDOWS SERVER "8" QUOI DE 9 !TECHDAYS PARIS 2012 : WINDOWS SERVER "8" QUOI DE 9 ! par ROMELARD Fabrice
Speakers: Fabrice Meillon et Stanislas Quastana Cette session est basée entièrement sur celle donnée lors de la BUILD cet hiver. Il n'y a pas d'ajout d'information en rapport avec cet évènement passé. Windows 8 Server sera intégralem...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice
Forum
RE : COURRE : COUR par barhoum1111
Cliquez pour lire la suite par barhoum1111 RE : COURRE : COUR par Julien39
Cliquez pour lire la suite par Julien39
Logiciels
DocTranslate (V3.1.0.0)DOCTRANSLATE (V3.1.0.0)DocTranslate est un traducteur de document Microsoft Word, PowerPoint et Excel. Il permet d'autom... Cliquez pour télécharger DocTranslate Tribler (2012)TRIBLER (2012)Tribler est un client pair à pair (P2P/Peer-to-Peer) open source avec la capacité de regarder des... Cliquez pour télécharger Tribler OneSwarm (2012)ONESWARM (2012)Le peer-to-peer qui protège votre vie privée, c'est OneSwarm.
Ce logiciel de peer-to-peer crypté... Cliquez pour télécharger OneSwarm PONAMEDIA PREMIUM - HELLLOOO FLASH DEMO (V8.4)PONAMEDIA PREMIUM - HELLLOOO FLASH DEMO (V8.4)PONAMEDIA TV DEVIENS HELLLOOO FLASH
LA TV SUR VOTRE ORDINATEUR.
Toute une plateforme Multi... Cliquez pour télécharger PONAMEDIA PREMIUM - HELLLOOO FLASH DEMO Academy System (17.2.1.0)ACADEMY SYSTEM (17.2.1.0)Logiciel de gestion des établissements.
- élèves/étudiants (inscription, dossier, absence...)
-... Cliquez pour télécharger Academy System
|