集装箱OCR:EAST和tesseract

1. 问题的分析

在解决问题前我们一般都会先看看获取的图像是什么样的,这样可以大概的评估出来我们应该使用什么样的算法。下图是监控摄像头拍摄到的画面,我们需要完成的就是把及上面的那些字母和数字识别出来。


7.PNG

从上图中看到,这显然是属于自然场景下的OCR问题,首先要做的就是文本的检测,文本检测目前效果不错的有EAST,CTPN等等。考虑到EAST有现成的模型和开源代码可以使用,首先就使用EAST来测试看看文本检测的效果

2. EAST文本检测

使用的模型来源于Adrian小哥的这篇文章。
因为加载模型的时候我们需要使用到OpenCV的dnn模块,所以我们需要使用OpenCV3.4.2或更高版本, 这里我使用的环境是:

opencv-python==3.4.5.20
numpy==1.15.0
imutils==0.5.2
pytesseract==0.2.6
Pillow==5.1.0

上面的库中imutils是为了方便的得到文本区域的bounding box的,Pillow是一个图像处理的库,这里我们用它来获取图像矩形的数据。pytesseract则是tesseract的接口,用来调用tesseract进行文本内容识别的。
环境完成好之后我们首先来实现文本的检测。

import numpy as np
import cv2
import time
from imutils.object_detection import non_max_suppression
import pytesseract
from PIL import Image

WIDTH = 640
HEIGHT = 800

net_file = "frozen_east_text_detection.pb"
min_confidence = 0.5

image_path = "3.png"
image = cv2.imread(image_path)
orig = image.copy()
(h, w) = image.shape[:2]

# 设置图像的宽和高
(newW, newH) = (WIDTH, HEIGHT)
rW = w / float(newW)
rH = h / float(newH)

# 将图像放缩为指定的大小
image = cv2.resize(image, (newW, newH))
(h, w) = image.shape[:2]

layers = ["feature_fusion/Conv_7/Sigmoid",
          "feature_fusion/concat_3"]

print("[INFO] 加载检测模型")
net = cv2.dnn.readNet(net_file)

blob = cv2.dnn.blobFromImage(image, 1.0, (w, h),
                             (123.68, 116.78, 103.94), swapRB=True, crop=False)

start = time.time()
net.setInput(blob)
(scores, geometry) = net.forward(layers)
end = time.time()

print("[INFO] 检测使用 {:.6f} 秒".format(end - start))

# 从scores中获取行和列
(numrows, numcols) = scores.shape[2:4]
rects = []  
confidences = []

for y in range(0, numrows):
    scoresData = scores[0, 0, y]
    xData0 = geometry[0, 0, y]
    xData1 = geometry[0, 1, y]
    xData2 = geometry[0, 2, y]
    xData3 = geometry[0, 3, y]
    anglesData = geometry[0, 4, y]

    for x in range(0, numcols):
        # 忽略置信度小于指定的概率
        if scoresData[x] < min_confidence:
            continue

        # 计算偏移因子,因为我们得到的特征图将比输入图像小4倍
        (offsetX, offsetY) = (x * 4.0, y * 4.0)

        # 提取旋转角度进行预测,然后计算sin和cosine
        angle = anglesData[x]
        cos = np.cos(angle)
        sin = np.sin(angle)

        # 使用几何体体积导出边界框的宽度和高度
        h = xData0[x] + xData2[x]
        w = xData1[x] + xData3[x]

        # 计算文本预测边界框的开始和结束(x,y)坐标
        endX = int(offsetX + (cos * xData1[x]) + (sin * xData2[x]))
        endY = int(offsetY - (sin * xData1[x]) + (cos * xData2[x]))
        startX = int(endX - w)
        startY = int(endY - h)

        # 将边界框坐标和概率分数添加到各自的列表中
        rects.append((startX, startY, endX, endY))
        confidences.append(scoresData[x])

# 非极大值抑制在弱边界框和重叠边界框上的应用
boxes = non_max_suppression(np.array(rects), probs=confidences)

for (startx, starty, endx, endy) in boxes:
    startx = int(startx * rW)
    starty = int(starty * rH)
    endx = int(endx * rW)
    endy = int(endy * rH)

    roi = orig[starty:endy, startx:endx]
    cv2.rectangle(orig, (startx, starty), (endx, endy), (0, 255, 0), 1)

cv2.imshow("text Detection", orig)
cv2.waitKey(0)

运行上述的代码就可以得到文本检测后的结果了。


文本检测结果

从上面的检测结果可以看到bounding box可以将我们想要的信息获取出来了,那么接下来的一步就是识别这些bounding box中的文本内容了。tesseract是OCR识别中一个现成的方案,所以就试试tesseract。

3. tesseract识别

有了上面的基础,使用tesseract识别就变得非常的简单,首先我们要确保我们的环境中已经安装了tesseract,而且可以在我们的cmd line中调用(即:环境变量中已经配置了它)。
识别的步骤就是讲上述的bounding box获取出来,依次的调用识别接口就可以完成了。


识别结果

识别结果中看到,中间有相当一些字符识别的是不正确的。这个是因为tesseract本来的识别精度就不是特别的高,针对不同场景下的文本识别,如果需要使用tesseract的时候,一般都是需要自己重新训练的。在工程上,一般很少见到直接使用tesseract的,只是在验证阶段初步看看效果的时候才会使用tesseract。

完整代码:

import numpy as np
import cv2
import time
from imutils.object_detection import non_max_suppression
import pytesseract
from PIL import Image

WIDTH = 640
HEIGHT = 800

net_file = "frozen_east_text_detection.pb"
min_confidence = 0.5

image_path = "3.png"
image = cv2.imread(image_path)
orig = image.copy()
(h, w) = image.shape[:2]

# 设置图像的宽和高
(newW, newH) = (WIDTH, HEIGHT)
rW = w / float(newW)
rH = h / float(newH)

# 将图像放缩为指定的大小
image = cv2.resize(image, (newW, newH))
(h, w) = image.shape[:2]

layers = ["feature_fusion/Conv_7/Sigmoid",
          "feature_fusion/concat_3"]

print("[INFO] 加载检测模型")
net = cv2.dnn.readNet(net_file)

blob = cv2.dnn.blobFromImage(image, 1.0, (w, h),
                             (123.68, 116.78, 103.94), swapRB=True, crop=False)

start = time.time()
net.setInput(blob)
(scores, geometry) = net.forward(layers)
end = time.time()

print("[INFO] 检测使用 {:.6f} 秒".format(end - start))

# 从scores中获取行和列
(numrows, numcols) = scores.shape[2:4]
rects = []  
confidences = []

for y in range(0, numrows):
    scoresData = scores[0, 0, y]
    xData0 = geometry[0, 0, y]
    xData1 = geometry[0, 1, y]
    xData2 = geometry[0, 2, y]
    xData3 = geometry[0, 3, y]
    anglesData = geometry[0, 4, y]

    for x in range(0, numcols):
        # 忽略置信度小于指定的概率
        if scoresData[x] < min_confidence:
            continue

        # 计算偏移因子,因为我们得到的特征图将比输入图像小4倍
        (offsetX, offsetY) = (x * 4.0, y * 4.0)

        # 提取旋转角度进行预测,然后计算sin和cosine
        angle = anglesData[x]
        cos = np.cos(angle)
        sin = np.sin(angle)

        # 使用几何体体积导出边界框的宽度和高度
        h = xData0[x] + xData2[x]
        w = xData1[x] + xData3[x]

        # 计算文本预测边界框的开始和结束(x,y)坐标
        endX = int(offsetX + (cos * xData1[x]) + (sin * xData2[x]))
        endY = int(offsetY - (sin * xData1[x]) + (cos * xData2[x]))
        startX = int(endX - w)
        startY = int(endY - h)

        # 将边界框坐标和概率分数添加到各自的列表中
        rects.append((startX, startY, endX, endY))
        confidences.append(scoresData[x])

# 非极大值抑制在弱边界框和重叠边界框上的应用
boxes = non_max_suppression(np.array(rects), probs=confidences)

for (startx, starty, endx, endy) in boxes:
    startx = int(startx * rW)
    starty = int(starty * rH)
    endx = int(endx * rW)
    endy = int(endy * rH)

    roi = orig[starty:endy, startx:endx]
    cv2.rectangle(orig, (startx, starty), (endx, endy), (0, 255, 0), 1)
    roi = cv2.cvtColor(roi, cv2.COLOR_BGR2GRAY)
    val, roi = cv2.threshold(roi, 0, 255, cv2.THRESH_BINARY + cv2.THRESH_OTSU)
    roi = Image.fromarray(roi)
    text = pytesseract.image_to_string(roi)
    cv2.putText(orig, text, (startx, starty), cv2.FONT_HERSHEY_COMPLEX, 0.8, (0, 0, 255), 1)

cv2.imshow("text Detection", orig)
cv2.waitKey(0)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,681评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,710评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,623评论 0 334
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,202评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,232评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,368评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,795评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,461评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,647评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,476评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,525评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,226评论 3 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,785评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,857评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,090评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,647评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,215评论 2 341

推荐阅读更多精彩内容