ホームページ バックエンド開発 Python チュートリアル Plotly チャートを画像に並行して変換する

Plotly チャートを画像に並行して変換する

Jan 04, 2025 am 02:32 AM

Converting Plotly charts into images in parallel

私が働いている会社では Plotly チャートを広く使用しています。これらを使用すると、見栄えの良いインタラクティブなグラフィックを簡単に作成できます。 Plotly Express ライブラリを介した Python エクスペリエンスは素晴らしく、始めるためのハードルは低いです。

Plotly グラフには 2 つの主な使用例があります:

  • Plotly Dash を使用したインタラクティブなダッシュボードの場合。 Plotly チャートの Dash への統合は明らかに優れています。
  • PDF レポートの場合、PDF をレンダリングする前にグラフを画像に変換します。

一般的な PDF レポートの場合、特定の指標の時間の経過に伴う変化、多数のカテゴリにわたる値の分布、または隣り合う異なるカテゴリの比較を示すために、5 ~ 20 個の図を使用します。

PDF レポートを作成するには、Weasyprint、Jinja、および Plotly チャートを組み合わせて使用​​します。レポートを PDF としてレンダリングするには、まずすべてのグラフを画像としてレンダリングする必要があります。

Kaleido を使用したグラフのレンダリング

これを行うために、素晴らしい Kaleido パッケージを使用します。 Chrome ブラウザを使用してグラフをレンダリングし、画像として保存します。 API は簡単に使用できます。

from kaleido.scopes.plotly import PlotlyScope

scope = PlotlyScope()
img_bytes = scope.transform(
    figure=figure, format="png", width=1000, height=1000, scale=4,
)
ログイン後にコピー
ログイン後にコピー

これにより、Figure 内の図が高さと幅 1000 ピクセル、レンダリング スケール 4 の画像としてレンダリングされます (つまり、画像の実際の寸法は 4000 ピクセル x 4000 ピクセル)。スケールが大きいほど、最終画像の DPI が高くなり、見栄えが良くなり、最終的な PDF が大きくなります。

大量のグラフをレンダリングする

グラフのレンダリングには少し時間がかかり、多数 (10 ~ 20) のグラフをレンダリングすると、プログラムの実行時間のかなりの部分を占めることになります。 PDF レンダリング パイプラインを高速化するために、次のソリューションをデプロイしました。

内部的には、Kaleido は、グラフを画像としてレンダリングする問題を、付属の Chrome ウェブブラウザにアウトソーシングしているだけです。これは、Python 自体の場合、この画像のレンダリングは基本的に I/O を待機していることを意味します。

この特定のプロセスを高速化するには、I/O を待つだけなので、マルチスレッドを使用できます。

ランダムグラフの作成

次のようなランダムな図を作成することから始めましょう:

import pandas as pd
import numpy as np
import plotly.graph_objects as go

def get_random_figure() -> go.Figure:
    n_bars = 50
    dates = pd.date_range(start="2021-01-01", end="2021-12-31", freq="M")

    figure = go.Figure()
    for i in range(n_bars):
        values = np.random.rand(len(dates))
        figure.add_trace(go.Bar(x=dates, y=values, name=f"Label {i+1}"))

    figure.update_layout(
        dict(
            barmode="group",
            legend=dict(orientation="h", yanchor="top", xanchor="left"),
        )
    )
    figure.update_layout(yaxis=dict(tickformat=".0%"), xaxis=dict(showgrid=False))
    return figure

ログイン後にコピー
ログイン後にコピー

上記のコードを使用して、Figure を画像に変換できます。

from kaleido.scopes.plotly import PlotlyScope
import plotly.graph_objects as go

def figure_to_bytes(figure: go.Figure) -> bytes:
    scope = PlotlyScope()
    return scope.transform(figure=figure, format="png", width=1000, height=1000, scale=4)
ログイン後にコピー
ログイン後にコピー

そして最後に、後で使用するために次の定義も行います。

def transform_random_figure() -> bytes:
    return figure_to_bytes(get_random_figure())
ログイン後にコピー
ログイン後にコピー

スレッドでの画像変換の実行

Python の GIL (グローバル インタプリタ ロック) により、同時に 1 つのスレッドだけが Python コードを実行できることをご存じないかもしれません。グラフから画像への変換は Python コードではないため、スレッドを利用して多数のグラフの変換を同時に開始し、結果を収集できます。

そのために、ヘルパー クラスを定義します。

from kaleido.scopes.plotly import PlotlyScope

scope = PlotlyScope()
img_bytes = scope.transform(
    figure=figure, format="png", width=1000, height=1000, scale=4,
)
ログイン後にコピー
ログイン後にコピー

このクラスは、変換の結果 (つまり、画像のバイト) を取得するのに役立ちます。

次にしなければならないことは、Python でスレッドを操作するための標準パターンに従うことです。

  1. start() メソッドを使用して、開始したいスレッドを開始します。
  2. join() メソッドを使用して、スレッドが結果を返すのを待ちます。

スレッドはそれぞれ、transform_random_figure() を呼び出してバイトを返す必要があります。この場合、10 個のスレッドを開始します。

import pandas as pd
import numpy as np
import plotly.graph_objects as go

def get_random_figure() -> go.Figure:
    n_bars = 50
    dates = pd.date_range(start="2021-01-01", end="2021-12-31", freq="M")

    figure = go.Figure()
    for i in range(n_bars):
        values = np.random.rand(len(dates))
        figure.add_trace(go.Bar(x=dates, y=values, name=f"Label {i+1}"))

    figure.update_layout(
        dict(
            barmode="group",
            legend=dict(orientation="h", yanchor="top", xanchor="left"),
        )
    )
    figure.update_layout(yaxis=dict(tickformat=".0%"), xaxis=dict(showgrid=False))
    return figure

ログイン後にコピー
ログイン後にコピー

start() メソッドは、実際のロジックを開始するスレッドの run() メソッドも呼び出します (つまり、指定された関数を実行します。この場合、transform_random_figure() を意味します)。

結果を収集するには、スレッドの join() メソッドを使用し、結果をファイルに書き込みます。

from kaleido.scopes.plotly import PlotlyScope
import plotly.graph_objects as go

def figure_to_bytes(figure: go.Figure) -> bytes:
    scope = PlotlyScope()
    return scope.transform(figure=figure, format="png", width=1000, height=1000, scale=4)
ログイン後にコピー
ログイン後にコピー

仕組み

ここでの主な考え方は、グラフを画像に変換したいときはいつでもスレッドを開始し、このスレッドはグラフがバックグラウンドで完了するのを待つということです。

レポート全体をまとめたら、すべてのスレッドで join() を呼び出し、すべてのグラフの画像を取得してレポートに組み込みます。

この方法では、グラフなしでレポート全体をすでに生成でき、各グラフが変換されるのを待たずに済むので時間を節約できます。

まとめ

要約すると、複数の Plotly チャートを画像に変換する場合は、Python 標準ライブラリのマルチスレッド モジュールを使用して、変換プロセスを高速化します。

transform() 呼び出しをスレッドに移動し、すべてのスレッドが終了するのを待つだけで、非常に簡単に実行できます。

付録: コード

def transform_random_figure() -> bytes:
    return figure_to_bytes(get_random_figure())
ログイン後にコピー
ログイン後にコピー

以上がPlotly チャートを画像に並行して変換するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

中間の読書にどこでもfiddlerを使用するときにブラウザによって検出されないようにするにはどうすればよいですか? 中間の読書にどこでもfiddlerを使用するときにブラウザによって検出されないようにするにはどうすればよいですか? Apr 02, 2025 am 07:15 AM

fiddlereveryversings for the-middleの測定値を使用するときに検出されないようにする方法

プロジェクトの基本と問題駆動型の方法で10時間以内にコンピューター初心者プログラミングの基本を教える方法は? プロジェクトの基本と問題駆動型の方法で10時間以内にコンピューター初心者プログラミングの基本を教える方法は? Apr 02, 2025 am 07:18 AM

10時間以内にコンピューター初心者プログラミングの基本を教える方法は?コンピューター初心者にプログラミングの知識を教えるのに10時間しかない場合、何を教えることを選びますか...

Investing.comの反クローラーメカニズムをバイパスするニュースデータを取得する方法は? Investing.comの反クローラーメカニズムをバイパスするニュースデータを取得する方法は? Apr 02, 2025 am 07:03 AM

Investing.comの反クラウリング戦略を理解する多くの人々は、Investing.com(https://cn.investing.com/news/latest-news)からのニュースデータをクロールしようとします。

Python 3.6のロードピクルスファイルエラーmodulenotfounderror:ピクルスファイル「__builtin__」をロードした場合はどうすればよいですか? Python 3.6のロードピクルスファイルエラーmodulenotfounderror:ピクルスファイル「__builtin__」をロードした場合はどうすればよいですか? Apr 02, 2025 am 06:27 AM

Python 3.6のピクルスファイルの読み込みエラー:modulenotfounderror:nomodulenamed ...

Scapy Crawlerを使用するときにパイプラインファイルを書き込めない理由は何ですか? Scapy Crawlerを使用するときにパイプラインファイルを書き込めない理由は何ですか? Apr 02, 2025 am 06:45 AM

SCAPYクローラーを使用するときにパイプラインファイルを作成できない理由についての議論は、SCAPYクローラーを学習して永続的なデータストレージに使用するときに、パイプラインファイルに遭遇する可能性があります...

See all articles