ROS : Map - 利用 opencv 删除无意义的未知区域

本文档主要是使用 opencv 对 ROS 地图数据进行操作的一次尝试。

ROS : Map - 利用 opencv 删除无意义的未知区域

整体思路

这里的思路很简单,利用 opencv 找到地图的外接矩形 ,取出该外接矩形的地图数据并发布出来。

代码实现 Python

需要注意的是 ROS 中 Map 是从左下角开始保存的,即左下角的索引为 0 ,而 opencv 图片左上角的索引为 0 。

#!/usr/bin/env python
# -*- coding: utf-8 -*-

"""
Created on 20-05-13

Updated on 20-05-13

@author: eln

@requirements: Ubuntu 16.04.6 LTS, Python 2.7.12 (default, Nov 12 2018, 14:36:49), ROS Version: kinetic 1.12.13

@decription: 删除无意义的未知区域
"""
import numpy as np
import copy
import cv2
import rospy
from nav_msgs.msg import OccupancyGrid


class Reducing(object):
    def __init__(self):
        rospy.init_node('map_reducing', anonymous=False)

        self.map_adj_pub = rospy.Publisher('/map_reducing', OccupancyGrid, queue_size=10, latch=True)
        rospy.Subscriber("/map", OccupancyGrid, self.map_callback, queue_size=10)

    def map_callback(self, msg):
        map_adj = OccupancyGrid()
        map_adj.header.frame_id = '/map'
        map_adj.header.stamp = rospy.Time.now()
        map_adj.info = msg.info
        original = msg.data
        print(len(original), map_adj.info.width * map_adj.info.height, map_adj.info.width, map_adj.info.height)

        img_tmp = self.map2grayscale(original)
        img_gray = np.array(img_tmp).reshape(map_adj.info.height, map_adj.info.width)
        img_gray = img_gray.astype(np.uint8)
        print(type(img_gray), img_gray.shape, img_gray.dtype)
        ret, thresh = cv2.threshold(img_gray, 180, 255, cv2.THRESH_BINARY)
        binary, contours, hierarchy = cv2.findContours(thresh, cv2.RETR_TREE, cv2.CHAIN_APPROX_NONE)
        cnt = contours[0]
        x, y, w, h = cv2.boundingRect(cnt)
        print(x, y, w, h)

        h_min = y
        h_max = y + h
        w_min = x
        w_max = x + w
        print(h_min, h_max, w_min, w_max)

        arr = np.array(original).reshape(map_adj.info.height, map_adj.info.width)
        print(type(arr), arr.shape, arr.dtype)
        arr_a = arr[h_min:h_max]
        new = arr_a[:, w_min:w_max]
        new = new.flatten()

        origin_index = h_min * map_adj.info.width + w_min
        origin_point = point_of_index(map_adj.info, origin_index)

        map_adj.info.width = w_max - w_min
        map_adj.info.height = h_max - h_min
        map_adj.info.origin.position.x = origin_point[0]
        map_adj.info.origin.position.y = origin_point[1]
        map_adj.data = new
        print(len(map_adj.data), map_adj.info.width * map_adj.info.height, map_adj.info.width, map_adj.info.height)

        self.map_adj_pub.publish(map_adj)
        print("========= end =========")

    @staticmethod
    def map2grayscale(data):
        data_new = copy.deepcopy(list(data))
        for idx, value in enumerate(data):
            if value == -1:
                data_new[idx] = 0
            else:
                data_new[idx] = 255
        return data_new


def point_of_index(map_info, i):
    y = map_info.origin.position.y + (i / map_info.width) * map_info.resolution
    x = map_info.origin.position.x + (i - (i / map_info.width) * map_info.width) * map_info.resolution
    return [x, y]


if __name__ == '__main__':
    try:
        Reducing()
        rospy.spin()
    except rospy.ROSInterruptException:
        rospy.loginfo('Reducing /map redundant data terminated.')

仿真效果

修改后的地图:
修改后的地图
原始地图:
原始地图

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值