前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >如何快速构建一个核酸点服务状态查询Web应用?Python实例详解

如何快速构建一个核酸点服务状态查询Web应用?Python实例详解

作者头像
用户8949263
发布2022-04-08 14:06:15
1.2K0
发布2022-04-08 14:06:15
举报
文章被收录于专栏:Python数据分析实例

什么是设计模式?

设计模式实软件中常见问题的典型解决方案。能根据需求进行预制蓝图,可用于解决代码中反复出现的设计问题。高质量应用程序框架设计过程广泛使用设计模式来确保代码可复用和可扩展性。

设计模式有什么用?

设计模式并非必须使用,我的项目没使用或不注重设计模式的使用也照样运行,的确,项目中不使用任何设计模式并不会影响项目的运行,但项目后期需求变动涉及二次开发时,在全新的上下文中工作,代码的维护和修改的复杂度着实让人头大。

项目中单例模式和工厂模式应该是用的最多的,单例模式应用以前的文章已经分享过。本次分享主要介绍一下简单工厂模式应用实例。

简单工厂模式案例

代码语言:javascript
复制
目录:1、查询应用效果图2、简单工厂模式案例1)项目结构目录
2)服务服务端-(Map_Load.py地图加载显示模式)3)应用客户端-Map_server_client.py4)Web应用发布-Map_app.py

查询应用效果图

←核酸采样点服务状态查询→

简单工厂模式案例

本次分享结合深圳市便民核酸点服务状态查询服务数据示例。

1、项目结构目录

假如有这样一个需求,做一个可切换地图加载模式的可视化项目

←程序项目结构→

项目主要由以上几个文件夹组成:

1、.venv--虚拟环境;.streamlit--网页主题设置

2、apps--应用集成文件夹(Mapview文件夹--应用服务端,logs文件夹--日志文件,Map_server_client--应用客户端;tools文件夹--其他脚本工具);

3、resources--资源文件夹

4、Map_app.py--web应用主文件

2、应用服务端-(地图加载显示模式-Map_Load.py)

背景:使用地址文件中的经纬度在地图上标记显示,并显示给定地点信息,提供两种加载模式供选择:Full_Load(地图一次性全加载)和Zoom_Loading(地图缩放加载)。

代码语言:javascript
复制
import pandas as pd
import folium
from folium import plugins

pd.set_option("expand_frame_repr", False)

from abc import  ABCMeta,abstractmethod

class BaseAlgorithm(metaclass=ABCMeta):
    """
    地图初始化基类
    """
    def __init__(self):
        #加载地图模式
        self.Map = folium.Map(location=[22.540477, 114.061226],
                         tiles='https://wprd01.is.autonavi.com/appmaptile?x={x}&y={y}&z={z}&lang=zh_cn&size=1&scl=1&style=7',
                         attr='高德-常规图',
                         zoom_start=10,
                         control_scale=True,
                         width='100%',)
    @abstractmethod
    def execute(self):
        pass


class Full_Load(BaseAlgorithm):
    """
   地图一次全加载
    """
    def __init__(self,data):
        # 获取数据源
        super(Full_Load, self).__init__()
        #数据中心
        self.data = data

    def execute(self):
        import base64
        encoded = base64.b64encode(open(r'.\resources\qrcode_logo.JPG', 'rb').read()).decode()

        # data映射一列数据【color】---设置数据过滤层
        for name, row in self.data.iterrows():
            html = '<h2> 欢迎关注Python数据分析实例</h2><img src="data:image/jpeg;base64,{0}"><br><h3>采样点名称:{1}</h3>' \
                   '<h3>当前等待时间:{2}</h3><h3>服务时间:{3}</h3><h3>服务人群:{4}</h3>' \
                   '<a href = https://szwj.borycloud.com/wh5/index.html#/>查询服务➡</a>'.format(encoded,row['service_name'],str(row["delay"]),row['service_time'],row['services'])
            iframe = folium.IFrame(html, width=380, height=560)
            folium.Marker(
                [row["LATITUDE"], row["LONGITUDE"]],
                # max_width设置每行显示字符数
                popup=folium.Popup(iframe,
                                        max_width='100%'), 
                tooltip=row["service_name"],
                icon=folium.Icon(icon='info-sign', color=row["color"])).add_to(self.Map)

        # self.Map.save(r'.\resources\folium_map_full.html')
        print("地图成功生成!")
        return self.Map

class Zoom_Loading(BaseAlgorithm):
    """
    地图缩放加载
    """
    def __init__(self,data):
        super(Zoom_Loading, self).__init__()
        # 获取数据源
        self.data = data

    def execute(self):
        import base64
        encoded = base64.b64encode(open(r'.\resources\qrcode_logo.JPG', 'rb').read()).decode()

        # 调用Marker可以创建标记,传入位置和信息,当鼠标放在标记上会显示出信息。
        folium.Marker([22.540477,114.061226],
                      popup=folium.Popup("深圳市", max_width=100)).add_to(self.Map)   #创建中心标记

        marker_cluster = plugins.MarkerCluster().add_to(self.Map)

        for name, row in self.data.iterrows():
            html = '<h2> 欢迎关注Python数据分析实例</h2><img src="data:image/jpeg;base64,{0}"><br><h3>采样点名称:{1}</h3>' \
                   '<h3>当前等待时间:{2}</h3><h3>服务时间:{3}</h3><h3>服务人群:{4}</h3>' \
                   '<a href = https://szwj.borycloud.com/wh5/index.html#/>查询服务➡</a>'.format(encoded,row['service_name'],str(row["delay"]),row['service_time'],row['services'])
            iframe = folium.IFrame(html, width=380, height=560)

            folium.Marker([row["LATITUDE"], row["LONGITUDE"]],
                          popup=folium.Popup(iframe, max_width='100%'),
                          tooltip=row["service_name"],
                          icon=folium.Icon(icon='info-sign', color='lightblue')).add_to(marker_cluster)
        # self.Map.save(r'.\resources\folium_map_zoom.html')
        print("地图成功生成")
        return self.Map


class ForestFactory(object):

    def make_execute(self,object_type,data):
        return eval(object_type)(data).execute()

备注:

1、简单工厂模式

代码语言:javascript
复制
简单工厂模式是创建型模式,顾名思义简单工厂模式则是选择创建出需要的对象。实质是由一个工厂类根据传入的参数,动态决定应该创建并且返回哪一个产品类(这些产品类继承自一个父类或接口)的实例。
代码语言:javascript
复制
本文应用均继承初始化BaseAlgorithm类方法,选择加载底图模式。这里创建了Full_Load和Zoom_Loading两种地图显示产品,当然可以根据业务需求拓展产品类。工厂类ForestFactory实例化来决定创建哪个产品类,在创建对象上的灵活性高。

如果一个类已经完成开发、测试和审核工作,属于某个框架或可被其他类使用的话,对其代码进行修改是有风险的。创建子类并重写原始类的部分内容以完成不同的行为。

用了设计模式给我的感受就是代码变得优雅简洁、逻辑清晰明了,再注入高大上的算法模型之后,你的 service 层代码看上去很nice。以上代码可进一步优化,对方法层面的封装,封装变化的内容。

2、Folium 简介

作为 Python 的一个可视化工具包 Folium,它通过 Leaflet 的地图服务,可以在 Jupyter Notebook 上实现可视化的地理位置作图,制作各种各样精美的地图信息。它不仅可以针对某个经纬度进行地理位置的可视化操作,还能够根据实时的人群地理位置信息来构建静态与动态热力图,甚至还能够针对经纬度的数量来进行必要的聚类可视化。

代码语言:javascript
复制

#生成地图
# 1. 初始化一个map对象
# zoom_start:地图zoom的初始级别,默认为10。假设改成11的话,那么就相当于在默认创建的地图的级别上放大一级。
# 2.底图更换
# tiles:str型,用于控制绘图调用的地图样式,默认为'OpenStreetMap',也有一些其他的内建地图样式,
# 如'Stamen Terrain'、'Stamen Toner'、'Mapbox Bright'、'Mapbox Control Room' 等;
''' lyrs可以设置为不同的参数,分别代表不同形式的地图,可以尝试1-7
        h = roads only
        m = standard roadmap
        p = terrain
        r = somehow altered roadmap
        s = satellite only
        t = terrain only
        y = hybrid
'''

3、Icon标记颜色

代码语言:javascript
复制
Icon should be one of: {'blue', 'beige', 'lightblue', 'darkred', 'lightgray', 'lightgreen', 'lightred', 'white', 'purple', 'green', 'darkgreen', 'pink', 'darkpurple', 'red', 'gray', 'darkblue', 'orange', 'black', 'cadetblue'}.

4、简单工厂模式和策略模式区别

我觉得简单工厂模式和策略模式很相似。从以上代码可以看出,工厂模式主要是返回的接口实现类的实例化对象,最后返回的结果是接口实现类中的方法,而策略模式是在实例化策略模式的时候已经创建好了,我们可以在策略模式中随意的拼接重写方法,简单来说,工厂模式只关注最后的结果,而策略模式注重的是过程。后面结合实例写一篇策略模式相关推文进一步讨论。

3、应用客户端-Map_server_client.py

代码语言:javascript
复制

from apps.tools.Map_Load import ForestFactory

class webservice_client():
    def __init__(self,BaseAlgorithm,data):
        # super(webservice_client, self).__init__()
        # 定义客户端对象
        self.BaseAlgorithm = BaseAlgorithm
        self.data = data

    @fail_data(msg='地图加载失败')
    def get_Map_model(self):
        f = ForestFactory()
        Map = f.make_execute(self.BaseAlgorithm,self.data)

        return Map

备注:这里给客户端实例化get_Map_model方法添加带参数装饰器,@fail_data(msg='地图加载失败')添加接口调用失败处理机制。下一篇文章补充。

4、web应用发布-Map_app.py

代码语言:javascript
复制

import streamlit as st
import os, sys
from apps.Map_server_client import webservice_client
import pandas as pd
from streamlit_option_menu import option_menu

class Mapview_App():

    def __init__(self, title='', **kwargs):
        self.__dict__.update(kwargs)
        self.title = title
        self.data_sources = pd.read_excel(r".\apps\Mapview\data1.xlsx",index_col=False)
        # 设置网页标题,以及使用宽屏模式
        st.set_page_config(
            page_title="Map_App",
            page_icon="🧊", 
            layout="wide",
            initial_sidebar_state="auto"  #侧边栏
        )

    def run(self):
        try:
            sidebar = self._cs_sidebar()
            self._cs_body(sidebar)
        except Exception as e:
            import streamlit as st
            st.image(os.path.join(".", "resources", "failure.png"), width=100, )
            st.error('An error has occurred,try again.')
            st.error('Error details: {}'.format(e))

    def _cs_sidebar(self):
        from apps.tools.convert_cir_image import circle
        import streamlit.components.v1 as components
        import streamlit as st

        components.html("""<marquee bgcolor="#670796" behavior="alternate"><font color='white'>欢迎进入主页面</font></marquee>""",
                        height=60)

        c1, c2, c3, c4, _ = st.sidebar.columns([3, 2, 4, 2, 2])
        # 输入logo的基础图片路径
        logo_path = r".\resources\logo.jpg"
        cir = circle(logo_path)

        c2.image(image=cir, width=120, caption='Python数据分析实例')

        with st.sidebar:
            """
            icons# bootstrap网站找到 https://icons.bootcss.com/
            """
            sidebar = option_menu("导航栏",
                                  ["首页", "标记数据源", "地图可视化",'关于'],
                                  icons=['house','cloud-upload', 'geo-alt', 'brightness-high', 'bar-chart',
                                         'bell', 'info-circle'],
                                  menu_icon="broadcast", default_index=0)

        st.sidebar.text("欢迎关注Python数据分析实例")

        # 隐藏右边的菜单以及页脚
        hide_streamlit_style = """
        <style>
        #MainMenu {visibility: hidden;}
        footer {visibility: hidden;}
        </style>
        """
        st.markdown(hide_streamlit_style, unsafe_allow_html=True)

        return sidebar

    def _cs_body(self,sidebar):

        import streamlit as st
        if sidebar == "标记数据源":
            # 侧边栏
            st.sidebar.subheader("点击数据下载")
            st.subheader("深圳市便民核酸采样点服务状态查询")
            Delay_time = st.number_input("请输入选择排队时间", step=1, max_value=60,value=30)

            @st.cache
            def load_data(data):
                info_list = []
                for i, row in data.iterrows():
                    info = {"code": row['code'], "service_name": row['service_name'],
                            "delay": row['delay'],"service_time":row['service_time'],'services':row['services']}
                    info_list.append(info)
                info_list.sort(key=lambda info: info['delay'], reverse=True)
                # 数据上传发送层
                DATA = pd.DataFrame(info_list)

                DATA.columns = ['采样点编号', '采样点名称', '等待时间','服务时间','服务人群']
                return DATA

            def filter_data(Delay_time,data_sources):
                """
                # 构建Web表格
                """
                # 加载数据表
                data = load_data(data_sources)
                # 等待时间过滤
                data = data[data["等待时间"] < float(Delay_time)]

                return data

            data = filter_data(Delay_time,self.data_sources)

            st.write('符合条件的采样点数量', data.shape[0])
            # 交互式表格
            # st.dataframe(data)
            st.table(data)

            @st.cache
            def convert_df(df):
                return df.to_csv().encode('gbk')

            csv = convert_df(data)

            st.sidebar.download_button(
                label="Download data as CSV",
                data=csv,
                file_name='large_df.csv',
                mime='text/csv', )

            st.sidebar.success('Success!')

            # 隐藏底部按钮
            hide_st_style = "<style># MainMenu {visibility: hidden;} footer {visibility: hidden;}</style>"
            st.markdown(hide_st_style, unsafe_allow_html=True)
            st.success('Success!')

        elif sidebar == "地图可视化":

            from streamlit_folium import folium_static
            def load_Map(BaseAlgorithm,data):
                # 发布数据应用
                obj = webservice_client(BaseAlgorithm,data)
                Map = obj.get_Map_model()
                return Map

            BaseAlgorithm_selectbox = st.sidebar.selectbox("选择模型", (
            'Zoom_Loading','Full_Load'))

            Map = load_Map(BaseAlgorithm_selectbox,self.data_sources)

            folium_static(Map,width=1400, height=800)

            # 隐藏底部按钮
            hide_st_style = "<style># MainMenu {visibility: hidden;} footer {visibility: hidden;}</style>"
            st.markdown(hide_st_style, unsafe_allow_html=True)
            st.success('Success!')

        elif sidebar == "关于":
            st.subheader("streamlit开发查询手册")

            from PIL import Image

            image = Image.open(r'.\resources\streamlit-cheat-sheet.png')
            st.image(image, caption='', use_column_width=True)

        else:
            import streamlit as st
            st.title("欢迎关注Python数据分析实例")
            st.subheader("号主:Brook")

            st.markdown(
                "<h2 style='text-align: center;'> "
                "<a href = https://mp.weixin.qq.com/mp/profile_ext?action=home&__biz=MzI0NzY2MDA4MA==&scene=124#wechat_redirect>➡</a>Python数据分析实例.<h2>",
                unsafe_allow_html=True)

            col_header_logo_left, col_header_logo_center, col_header_logo_right = st.columns(
                [2, 2, 2])

            col_header_logo_center.image(os.path.join(".", "resources", "qrcode_logo.jpg"), width=400, )

            MENU_LAYOUT = [1, 1, 1, 7, 2]
            _, _, col_logo, col_text, _ = st.columns(MENU_LAYOUT)
            col_logo.image(os.path.join(".", "resources", "data.png"), width=80, )
            col_text.subheader("这是一个记录数据分析开发路上有趣、有料、实用至上的Python项目实战分享库,欢迎关注!一起成长~.")

            st.markdown('<br><br>', unsafe_allow_html=True)


if __name__=="__main__":
    obj = Mapview_App()
    obj.run()

备注:

1、低代码开发平台-streamlit

Streamlit 网站:https://streamlit.io/ GitHub地址:https://github.com/streamlit/streamlit/

Streamlit app 是完全自上而下运行的脚本,你可以利用函数调用来处理代码。

Pycharm的Terminal中运行:streamlit run Map_app.py

运行之后,就会自动生成网页:http://localhost:8501/,直接浏览器访问即可。

2、convert_cir_image.py-生成圆形logo图片脚本

3、重要组件

streamlit写markdown一样写网页,代码快速生成web网页。代码较简单易理解,只要会写Python脚本,能很快上手,这里不对上面代码详细解释了,下面提供常见组件api供查阅使用。

以上是一个应用基本框架介绍,你可以随意调整代码,拓展组件,创建一个更集成、更全面、更复杂的应用。以上为本次分享的全部内容,文中已包含大部分源代码,

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-03-20,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Python数据分析实例 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Streamlit app 是完全自上而下运行的脚本,你可以利用函数调用来处理代码。
相关产品与服务
灰盒安全测试
腾讯知识图谱(Tencent Knowledge Graph,TKG)是一个集成图数据库、图计算引擎和图可视化分析的一站式平台。支持抽取和融合异构数据,支持千亿级节点关系的存储和计算,支持规则匹配、机器学习、图嵌入等图数据挖掘算法,拥有丰富的图数据渲染和展现的可视化方案。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档