欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java OCR tesseract 圖像智能文字字符識(shí)別技術(shù)實(shí)例代碼

 更新時(shí)間:2017年06月20日 14:43:00   作者:鴻洋_  
這篇文章主要介紹了Java OCR tesseract 圖像智能文字字符識(shí)別技術(shù)實(shí)例代碼,非常具有實(shí)用價(jià)值,需要的朋友可以參考下

接著上一篇OCR所說(shuō)的,上一篇給大家介紹了tesseract 在命令行的簡(jiǎn)單用法,當(dāng)然了要繼承到我們的程序中,還是需要代碼實(shí)現(xiàn)的,下面給大家分享下Java實(shí)現(xiàn)的例子。

拿代碼掃描上面的圖片,然后輸出結(jié)果。主要思想就是利用Java調(diào)用系統(tǒng)任務(wù)。

下面是核心代碼:

package com.zhy.test; 
 
import java.io.BufferedReader; 
 
import java.io.File; 
import java.io.FileInputStream; 
import java.io.InputStreamReader; 
import java.util.ArrayList; 
import java.util.List; 
 
import org.jdesktop.swingx.util.OS; 
 
public class OCRHelper 
{ 
 private final String LANG_OPTION = "-l"; 
 private final String EOL = System.getProperty("line.separator"); 
 /** 
  * 文件位置我防止在,項(xiàng)目同一路徑 
  */ 
 private String tessPath = new File("tesseract").getAbsolutePath(); 
 
 /** 
  * @param imageFile 
  *   傳入的圖像文件 
  * @param imageFormat 
  *   傳入的圖像格式 
  * @return 識(shí)別后的字符串 
  */ 
 public String recognizeText(File imageFile) throws Exception 
 { 
  /** 
   * 設(shè)置輸出文件的保存的文件目錄 
   */ 
  File outputFile = new File(imageFile.getParentFile(), "output"); 
 
  StringBuffer strB = new StringBuffer(); 
  List<String> cmd = new ArrayList<String>(); 
  if (OS.isWindowsXP()) 
  { 
   cmd.add(tessPath + "\\tesseract"); 
  } else if (OS.isLinux()) 
  { 
   cmd.add("tesseract"); 
  } else 
  { 
   cmd.add(tessPath + "\\tesseract"); 
  } 
  cmd.add(""); 
  cmd.add(outputFile.getName()); 
  cmd.add(LANG_OPTION); 
//  cmd.add("chi_sim"); 
  cmd.add("eng"); 
 
  ProcessBuilder pb = new ProcessBuilder(); 
  /** 
   *Sets this process builder's working directory. 
   */ 
  pb.directory(imageFile.getParentFile()); 
  cmd.set(1, imageFile.getName()); 
  pb.command(cmd); 
  pb.redirectErrorStream(true); 
  Process process = pb.start(); 
  // tesseract.exe 1.jpg 1 -l chi_sim 
  // Runtime.getRuntime().exec("tesseract.exe 1.jpg 1 -l chi_sim"); 
  /** 
   * the exit value of the process. By convention, 0 indicates normal 
   * termination. 
   */ 
//  System.out.println(cmd.toString()); 
  int w = process.waitFor(); 
  if (w == 0)// 0代表正常退出 
  { 
   BufferedReader in = new BufferedReader(new InputStreamReader( 
     new FileInputStream(outputFile.getAbsolutePath() + ".txt"), 
     "UTF-8")); 
   String str; 
 
   while ((str = in.readLine()) != null) 
   { 
    strB.append(str).append(EOL); 
   } 
   in.close(); 
  } else 
  { 
   String msg; 
   switch (w) 
   { 
   case 1: 
    msg = "Errors accessing files. There may be spaces in your image's filename."; 
    break; 
   case 29: 
    msg = "Cannot recognize the image or its selected region."; 
    break; 
   case 31: 
    msg = "Unsupported image format."; 
    break; 
   default: 
    msg = "Errors occurred."; 
   } 
   throw new RuntimeException(msg); 
  } 
  new File(outputFile.getAbsolutePath() + ".txt").delete(); 
  return strB.toString().replaceAll("\\s*", ""); 
 } 
} 

代碼很簡(jiǎn)單,中間那部分ProcessBuilder其實(shí)就類似Runtime.getRuntime().exec("tesseract.exe 1.jpg 1 -l chi_sim"),大家不習(xí)慣的可以使用Runtime。

測(cè)試代碼:

package com.zhy.test; 
 
import java.io.File; 
 
public class Test 
{ 
 public static void main(String[] args) 
 { 
  try 
  { 
    
   File testDataDir = new File("testdata"); 
   System.out.println(testDataDir.listFiles().length); 
   int i = 0 ; 
   for(File file :testDataDir.listFiles()) 
   { 
    i++ ; 
    String recognizeText = new OCRHelper().recognizeText(file); 
    System.out.print(recognizeText+"\t"); 
 
    if( i % 5 == 0 ) 
    { 
     System.out.println(); 
    } 
   } 
    
  } catch (Exception e) 
  { 
   e.printStackTrace(); 
  } 
 
 } 
} 

輸出結(jié)果:

對(duì)比第一張圖片,是不是很完美~哈哈 ,當(dāng)然了如果你只需要實(shí)現(xiàn)驗(yàn)證碼的讀寫,那么上面就足夠了。下面繼續(xù)普及圖像處理的知識(shí)。

當(dāng)然了,有時(shí)候圖片被扭曲或者模糊的很厲害,很不容易識(shí)別,所以下面我給大家介紹一個(gè)去噪的輔助類,絕對(duì)碉堡了,先看下效果圖。

 

來(lái)張?zhí)貙懀?/p>

一個(gè)類,不依賴任何jar,把圖像中的干擾線消滅了,是不是很給力,然后再拿這樣的圖片去識(shí)別,會(huì)不會(huì)效果更好呢,嘿嘿,大家自己實(shí)驗(yàn)~

代碼:

package com.zhy.test; 
 
import java.awt.Color; 
import java.awt.image.BufferedImage; 
import java.io.File; 
import java.io.IOException; 
 
import javax.imageio.ImageIO; 
 
public class ClearImageHelper 
{ 
 
 public static void main(String[] args) throws IOException 
 { 
 
   
  File testDataDir = new File("testdata"); 
  final String destDir = testDataDir.getAbsolutePath()+"/tmp"; 
  for (File file : testDataDir.listFiles()) 
  { 
   cleanImage(file, destDir); 
  } 
 
 } 
 
 /** 
  * 
  * @param sfile 
  *   需要去噪的圖像 
  * @param destDir 
  *   去噪后的圖像保存地址 
  * @throws IOException 
  */ 
 public static void cleanImage(File sfile, String destDir) 
   throws IOException 
 { 
  File destF = new File(destDir); 
  if (!destF.exists()) 
  { 
   destF.mkdirs(); 
  } 
 
  BufferedImage bufferedImage = ImageIO.read(sfile); 
  int h = bufferedImage.getHeight(); 
  int w = bufferedImage.getWidth(); 
 
  // 灰度化 
  int[][] gray = new int[w][h]; 
  for (int x = 0; x < w; x++) 
  { 
   for (int y = 0; y < h; y++) 
   { 
    int argb = bufferedImage.getRGB(x, y); 
    // 圖像加亮(調(diào)整亮度識(shí)別率非常高) 
    int r = (int) (((argb >> 16) & 0xFF) * 1.1 + 30); 
    int g = (int) (((argb >> 8) & 0xFF) * 1.1 + 30); 
    int b = (int) (((argb >> 0) & 0xFF) * 1.1 + 30); 
    if (r >= 255) 
    { 
     r = 255; 
    } 
    if (g >= 255) 
    { 
     g = 255; 
    } 
    if (b >= 255) 
    { 
     b = 255; 
    } 
    gray[x][y] = (int) Math 
      .pow((Math.pow(r, 2.2) * 0.2973 + Math.pow(g, 2.2) 
        * 0.6274 + Math.pow(b, 2.2) * 0.0753), 1 / 2.2); 
   } 
  } 
 
  // 二值化 
  int threshold = ostu(gray, w, h); 
  BufferedImage binaryBufferedImage = new BufferedImage(w, h, 
    BufferedImage.TYPE_BYTE_BINARY); 
  for (int x = 0; x < w; x++) 
  { 
   for (int y = 0; y < h; y++) 
   { 
    if (gray[x][y] > threshold) 
    { 
     gray[x][y] |= 0x00FFFF; 
    } else 
    { 
     gray[x][y] &= 0xFF0000; 
    } 
    binaryBufferedImage.setRGB(x, y, gray[x][y]); 
   } 
  } 
 
  // 矩陣打印 
  for (int y = 0; y < h; y++) 
  { 
   for (int x = 0; x < w; x++) 
   { 
    if (isBlack(binaryBufferedImage.getRGB(x, y))) 
    { 
     System.out.print("*"); 
    } else 
    { 
     System.out.print(" "); 
    } 
   } 
   System.out.println(); 
  } 
 
  ImageIO.write(binaryBufferedImage, "jpg", new File(destDir, sfile 
    .getName())); 
 } 
 
 public static boolean isBlack(int colorInt) 
 { 
  Color color = new Color(colorInt); 
  if (color.getRed() + color.getGreen() + color.getBlue() <= 300) 
  { 
   return true; 
  } 
  return false; 
 } 
 
 public static boolean isWhite(int colorInt) 
 { 
  Color color = new Color(colorInt); 
  if (color.getRed() + color.getGreen() + color.getBlue() > 300) 
  { 
   return true; 
  } 
  return false; 
 } 
 
 public static int isBlackOrWhite(int colorInt) 
 { 
  if (getColorBright(colorInt) < 30 || getColorBright(colorInt) > 730) 
  { 
   return 1; 
  } 
  return 0; 
 } 
 
 public static int getColorBright(int colorInt) 
 { 
  Color color = new Color(colorInt); 
  return color.getRed() + color.getGreen() + color.getBlue(); 
 } 
 
 public static int ostu(int[][] gray, int w, int h) 
 { 
  int[] histData = new int[w * h]; 
  // Calculate histogram 
  for (int x = 0; x < w; x++) 
  { 
   for (int y = 0; y < h; y++) 
   { 
    int red = 0xFF & gray[x][y]; 
    histData[red]++; 
   } 
  } 
 
  // Total number of pixels 
  int total = w * h; 
 
  float sum = 0; 
  for (int t = 0; t < 256; t++) 
   sum += t * histData[t]; 
 
  float sumB = 0; 
  int wB = 0; 
  int wF = 0; 
 
  float varMax = 0; 
  int threshold = 0; 
 
  for (int t = 0; t < 256; t++) 
  { 
   wB += histData[t]; // Weight Background 
   if (wB == 0) 
    continue; 
 
   wF = total - wB; // Weight Foreground 
   if (wF == 0) 
    break; 
 
   sumB += (float) (t * histData[t]); 
 
   float mB = sumB / wB; // Mean Background 
   float mF = (sum - sumB) / wF; // Mean Foreground 
 
   // Calculate Between Class Variance 
   float varBetween = (float) wB * (float) wF * (mB - mF) * (mB - mF); 
 
   // Check if new maximum found 
   if (varBetween > varMax) 
   { 
    varMax = varBetween; 
    threshold = t; 
   } 
  } 
 
  return threshold; 
 } 
} 

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。 

相關(guān)文章

  • 新手Hadoop安裝 環(huán)境搭建

    新手Hadoop安裝 環(huán)境搭建

    這篇文章主要介紹了Hadoop的安裝與環(huán)境搭建教程圖解,本文圖文并茂給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下,希望能給您帶來(lái)幫助
    2021-06-06
  • SpringBoot @Value注解支持配置自動(dòng)刷新能力擴(kuò)展方式

    SpringBoot @Value注解支持配置自動(dòng)刷新能力擴(kuò)展方式

    本文介紹了如何通過(guò)自定義注解和BeanPostProcessor實(shí)現(xiàn)SpringBoot中@Value注解的配置自動(dòng)刷新能力,主要步驟包括:定義一個(gè)支持動(dòng)態(tài)刷新的注解,實(shí)現(xiàn)配置的動(dòng)態(tài)變更,以及通過(guò)BeanPostProcessor掃描并刷新使用@Value注解的變量
    2024-12-12
  • struts2實(shí)現(xiàn)多文件上傳的示例代碼

    struts2實(shí)現(xiàn)多文件上傳的示例代碼

    本篇文章主要介紹了struts2實(shí)現(xiàn)多文件上傳的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-03-03
  • javaBean的基礎(chǔ)知識(shí)及常見(jiàn)亂碼解決方法

    javaBean的基礎(chǔ)知識(shí)及常見(jiàn)亂碼解決方法

    這篇文章主要介紹了javaBean的基礎(chǔ)知識(shí)及常見(jiàn)亂碼解決方法的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • 關(guān)于springcloud集成nacos遇到的問(wèn)題

    關(guān)于springcloud集成nacos遇到的問(wèn)題

    這篇文章主要介紹了關(guān)于springcloud集成nacos遇到的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • Spark學(xué)習(xí)筆記之Spark SQL的具體使用

    Spark學(xué)習(xí)筆記之Spark SQL的具體使用

    這篇文章主要介紹了Spark學(xué)習(xí)筆記之Spark SQL的具體使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-06-06
  • MybatisPlus處理大表查詢的實(shí)現(xiàn)步驟

    MybatisPlus處理大表查詢的實(shí)現(xiàn)步驟

    在實(shí)際工作中當(dāng)指定查詢數(shù)據(jù)過(guò)大時(shí),我們一般使用分頁(yè)查詢的方式一頁(yè)一頁(yè)的將數(shù)據(jù)放到內(nèi)存處理,本文主要介紹了MybatisPlus處理大表查詢的實(shí)現(xiàn)步驟,感興趣的可以了解一下
    2024-08-08
  • Java中Quartz高可用定時(shí)任務(wù)快速入門

    Java中Quartz高可用定時(shí)任務(wù)快速入門

    如果你想做定時(shí)任務(wù),有高可用方面的需求,或者僅僅想入門快,上手簡(jiǎn)單,那么選用它準(zhǔn)沒(méi)錯(cuò),感興趣的小伙伴們可以參考一下
    2022-04-04
  • springboot從application.properties中注入list,?map方式

    springboot從application.properties中注入list,?map方式

    這篇文章主要介紹了springboot從application.properties中注入list,map方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • MyBatis攔截器如何自動(dòng)設(shè)置創(chuàng)建時(shí)間和修改時(shí)間

    MyBatis攔截器如何自動(dòng)設(shè)置創(chuàng)建時(shí)間和修改時(shí)間

    文章介紹了如何通過(guò)實(shí)現(xiàn)MyBatis的Interceptor接口,在實(shí)體類中自動(dòng)設(shè)置創(chuàng)建時(shí)間和修改時(shí)間,從而提高開(kāi)發(fā)效率
    2025-02-02

最新評(píng)論