GeoDjango - 正方形,六边形切割地理信息数据

SQL使用PostgreSQL

使用场景为生成马赛克风格的图片,即把某个区域分割成若干个小的区域,计算每个小的区域中包含的某些体量,用不同颜色显示。

1. 矩形(rectangle)

原理:每一行/列小矩形的中心点连线都是平行的,根据横纵切割的数量可以算出所有的中心点,再根据每个中心点计算出四个顶点的坐标。(如果想展现正方形可以1.手动控制横纵切割的比;2.简单修改代码逻辑,只传入横向切割数量,用边长计算纵向数量。)

def cal_rasterized_points_from_boundary_v2(obj, boundary_li, lng_count, lat_count):
    """
    计算边界内栅格化后的中心点,返回若干个矩形的坐标数组 -- 马赛克凹凸有致版
    :param obj: 多边形的object
    :param boundary_li: multipolygon.boundary.coords转化后的list
    :param lng_count: 横向切割数量
    :param lat_count: 纵向切割数量
    :return: 
    """
    # 目前只考虑MultiPolygon包含一个多边形的情况。
    boundary_list = boundary_li[0]
    lngs = []
    lats = []
    for bl in boundary_list:
        lngs.append(bl[0])
        lats.append(bl[1])
    max_lng = float(max(lngs))
    min_lng = float(min(lngs))
    max_lat = float(max(lats))
    min_lat = float(min(lats))
    max_lng_amap, max_lat_amap = wgs84togcj02(max_lng, max_lat)
    min_lng_amap, min_lat_amap = wgs84togcj02(min_lng, min_lat)

    lng_gap = (max_lng_amap - min_lng_amap) / lng_count
    lat_gap = (max_lat_amap - min_lat_amap) / lat_count
    data_lngs = []
    data_lats = []
    cps = []
    result = []
    for i in range(lng_count):
        lng = min_lng_amap + lng_gap * 0.5 + lng_gap * i
        if lng < max_lng_amap:
            data_lngs.append(lng)
    for i in range(lat_count):
        lat = min_lat_amap + lat_gap * 0.5 + lat_gap * i
        if lat < max_lat_amap:
            data_lats.append(lat)
    for lat in data_lats:
        for lng in data_lngs:
            cps.append([lng, lat])
    for cp in cps:
        # TODO: To judge cp whether in boundary or not
        lng, lat = gcj02towgs84(cp[0], cp[1])
        point = Point((lng, lat))
        li = obj.filter(boundary__contains=point)
        if not li:
            continue
        result.append([[cp[0] - lng_gap * 0.5, cp[1] - lat_gap * 0.5],
                       [cp[0] + lng_gap * 0.5, cp[1] - lat_gap * 0.5],
                       [cp[0] + lng_gap * 0.5, cp[1] + lat_gap * 0.5],
                       [cp[0] - lng_gap * 0.5, cp[1] + lat_gap * 0.5]])

    return result

2. 六边形(hexagonal)

原理:

20170124_01_pic_004.png

六边形可以拆分成六个等边三角形。


20170124_01_pic_010.png

每个中心点的位置每两行一个循环。同行两个中心点间距为三倍的边长,行间距为√3/2倍边长(经纬度间距离转换存在差异,现采用纬度间*0.88)

def cal_rasterized_points_from_boundary(obj, boundary_li, lng_count):
    """
    计算边界内栅格化后的中心点,返回若干个六边形的坐标数组 -- 马赛克凹凸有致版  
    蜂巢六边形切割 -- 粗略用lng的差来表示边长,因为分割后数值很小,可以忽略坐标和距离转换的误差
    :param obj: 多边形的object
    :param boundary_li: multipolygon.boundary.coords转化后的list
    :param lng_count: 横向切割数量
    :return: 
    """
    # 目前只考虑MultiPolygon包含一个多边形的情况。
    boundary_list = boundary_li[0]
    lngs = []
    lats = []
    for bl in boundary_list:
        lngs.append(bl[0])
        lats.append(bl[1])
    max_lng = float(max(lngs))
    min_lng = float(min(lngs))
    max_lat = float(max(lats))
    min_lat = float(min(lats))
    max_lng_amap, max_lat_amap = wgs84togcj02(max_lng, max_lat)
    min_lng_amap, min_lat_amap = wgs84togcj02(min_lng, min_lat)

    lng_gap = (max_lng_amap - min_lng_amap) / lng_count
    side_length = lng_gap / 3
    lat_gap = side_length * math.sqrt(3) / 2 * 0.88
    lat_count = int((max_lat_amap - min_lat_amap) / lat_gap) + 1
    data_lngs_first = []
    data_lngs_second = []
    data_lats = []
    cps = []
    result = []
    for i in range(lng_count):
        lng = min_lng_amap + lng_gap * 0.5 + lng_gap * i
        if lng < max_lng_amap:
            data_lngs_first.append(lng)
    for lng in data_lngs_first:
        sec_lng = lng - side_length * 3 / 2
        if sec_lng < max_lng_amap:
            data_lngs_second.append(sec_lng)

    for i in range(lat_count):
        lat = min_lat_amap + lat_gap * 0.5 + lat_gap * i
        if lat < max_lat_amap:
            data_lats.append(lat)

    for i, lat in enumerate(data_lats):
        # 六边形单双行中心点不对齐
        if i % 2 == 0:
            for lng in data_lngs_first:
                cps.append([lng, lat])
        else:
            for lng in data_lngs_second:
                cps.append([lng, lat])
    for cp in cps:
        lng, lat = gcj02towgs84(cp[0], cp[1])
        point = Point((lng, lat))
        li = obj.filter(boundary__contains=point)
        if not li:
            continue
        # 左上角为第一个点
        result.append([[cp[0] - side_length * 0.5, cp[1] + lat_gap],
                       [cp[0] + side_length * 0.5, cp[1] + lat_gap],
                       [cp[0] + side_length, cp[1]],
                       [cp[0] + side_length * 0.5, cp[1] - lat_gap],
                       [cp[0] - side_length * 0.5, cp[1] - lat_gap],
                       [cp[0] - side_length, cp[1]]])

    return result


def generate_comb_mosaic_item(bc_id, lng_count):
    area = BusinessCircle.objects.filter(business_circle_id=bc_id)
    if not area:
        return ""

    mp = area[0].boundary
    if len(mp) > 1:
        return "返回的是多个多边形,这样不ok"

    boundary_li = [[list(list(location)) for location in list(polygon)] for polygon in list(mp.boundary.coords)]

    result = cal_rasterized_points_from_boundary(area, boundary_li, lng_count)
    # data = [[j, i] for i in range(lat_count) for j in range(lng_count)]
    data = []
    cursor = connection.cursor()
    SQL_string_list = []
    dp_result = copy.deepcopy(result)
    for index, points in enumerate(dp_result):
        MULTIPOLYGON_txt = ""
        points.append(points[0])
        for point in points:
            lng, lat = gcj02towgs84(point[0], point[1])
            MULTIPOLYGON_txt += ", {} {}".format(lng, lat)
        MULTIPOLYGON_txt = MULTIPOLYGON_txt[2:]
        SQL_RAW = """SELECT avg(forecast_price) FROM economic_dqchina_loupaninfo
        WHERE st_contains(st_geomfromtext('MULTIPOLYGON((({})))', 4326) :: geometry, center_point :: geometry) UNION ALL """.format(MULTIPOLYGON_txt)

        SQL_string_list.append(SQL_RAW)
    SQL_RAWS = "".join(SQL_string_list)
    cursor.execute(SQL_RAWS[:-10])
    rows = cursor.fetchall()
    for index, row in enumerate(rows):
        data.append(row[0])
    item = {
        "data": [float(d) if d else 0 for d in data],
        "result": result,
    }
    # print(item["data"])
    # print(item["result"])
    return item
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容