electrical.T_distrib のソースコード

#!/usr/bin/env python3
"""
層状熱電スタックの定常一次元温度プロファイルを計算・プロットするスクリプトです。

このスクリプトは、層状熱電スタックの定常一次元温度プロファイルを計算し、結果をプロットします。
異なる表面熱抵抗のモデルに対応するために、二つのモードが利用可能です。

モード
-----
1.  **delta**:
    表面熱抵抗は、厚さ ``delta_top`` / ``delta_bottom`` および熱伝導率 ``k_air`` を持つ
    明示的な上部/下部空気境界層としてモデル化されます。

2.  **conv_rad**:
    表面熱抵抗は、以下の要素から予測されます。
    -   水平平板のヌセルト数相関を用いた自然対流。
    -   熱放射。

どちらのモードでも、内部スタックは以下の層で構成されます。
上部電極 / 界面12 / TE半導体 / 界面23 / 下部電極

スクリプトは以下の情報を出力します。
-   各層/界面の抵抗
-   全体の抵抗
-   界面温度
-   TE層の温度降下

また、物理スケールと模式図スケールの2つの垂直に重ねられたプロットを含む図を保存します。

:doc:`T_distrib_usage`
"""

from __future__ import annotations

import argparse
from dataclasses import dataclass
from typing import List, Tuple

import numpy as np
import matplotlib.pyplot as plt


SIGMA_SB = 5.670374419e-8  # W / m^2 / K^4
G = 9.81  # m / s^2


[ドキュメント] @dataclass class Region: """ 熱電スタックの単一の領域(層、界面、表面)を表すデータクラス。 このクラスは、特定の領域の物理的特性(名称、厚さ、熱伝導率、カテゴリ)を 保持し、その熱抵抗を計算するプロパティを提供します。 :param name: str: 領域の識別名。 :param thickness_m: float: 領域の厚さ [m]。 :param k_W_mK: float: 領域の熱伝導率 [W/mK]。 :param category: str: 領域のカテゴリ(例: "surface", "layer", "interface")。 """ name: str thickness_m: float k_W_mK: float category: str # "surface", "layer", "interface" @property def resistance_m2K_W(self) -> float: """ 領域の熱抵抗を計算して返します。 熱抵抗は厚さを熱伝導率で割ったものとして定義されます。 :returns: float: 領域の熱抵抗 [m^2K/W]。 """ return self.thickness_m / self.k_W_mK
[ドキュメント] def build_internal_regions(args: argparse.Namespace) -> List[Region]: """ 熱電スタック内部のコア層と界面の `Region` オブジェクトのリストを構築します。 この関数は、コマンドライン引数で指定された厚さ ``d1``, ``d2``, ``d3`` と 熱伝導率 ``k1``, ``k2``, ``k3``、および界面の厚さ ``delta12``, ``delta23`` と 熱伝導率 ``k12``, ``k23`` を使用して、内部スタックの領域を定義します。 :param args: argparse.Namespace: コマンドライン引数を格納するオブジェクト。 :returns: List[Region]: 内部スタック領域のリスト。 """ return [ Region("Top electrode", args.d1, args.k1, "layer"), Region("Interface 1/2", args.delta12, args.k12, "interface"), Region("TE semiconductor", args.d2, args.k2, "layer"), Region("Interface 2/3", args.delta23, args.k23, "interface"), Region("Bottom electrode", args.d3, args.k3, "layer"), ]
[ドキュメント] def build_regions_delta(args: argparse.Namespace) -> List[Region]: """ 'delta'モードでの全領域(空気境界層を含む)の `Region` オブジェクトのリストを構築します。 この関数は、 ``build_internal_regions`` で構築された内部スタック領域の前後に、 コマンドライン引数で指定された ``delta_top``, ``delta_bottom`` の厚さと ``k_air`` の熱伝導率を持つ空気境界層を追加します。 :param args: argparse.Namespace: コマンドライン引数を格納するオブジェクト。 :returns: List[Region]: 'delta'モードにおける全スタック領域のリスト。 """ return [ Region("Top air boundary", args.delta_top, args.k_air, "surface"), *build_internal_regions(args), Region("Bottom air boundary", args.delta_bottom, args.k_air, "surface"), ]
[ドキュメント] def compute_piecewise_profile( T_top_C: float, regions: List[Region], q_W_m2: float, n_points_per_region: int = 80, ) -> Tuple[np.ndarray, np.ndarray, List[float]]: """ 既知の熱流束を持つ明示的な領域リストに対する物理的なZ座標と温度プロファイルを計算します。 各領域で線形温度勾配を仮定し、全体の温度プロファイルと各界面の温度を計算します。 プロファイルは連続的であり、各領域の熱伝導率と厚さに基づいて温度降下が計算されます。 :param T_top_C: float: スタック最上部の温度 [°C]。 :param regions: List[Region]: スタックを構成するRegionオブジェクトのリスト。 :param q_W_m2: float: スタックを通過する正味の熱流束(下向きが正) [W/m^2]。 :param n_points_per_region: int, optional: 各領域でプロットのために生成する点の数 (デフォルトは80)。 :returns: Tuple[np.ndarray, np.ndarray, List[float]]: 物理的なZ座標の配列、対応する温度の配列、および各界面(最上部と最下部を含む)の温度のリストのタプル。 """ z_segments: List[np.ndarray] = [] T_segments: List[np.ndarray] = [] z0 = 0.0 T0 = T_top_C interface_temperatures = [T0] for i, r in enumerate(regions): z1 = z0 + r.thickness_m endpoint = i == len(regions) - 1 z = np.linspace(z0, z1, n_points_per_region, endpoint=endpoint) T = T0 - q_W_m2 * (z - z0) / r.k_W_mK z_segments.append(z) T_segments.append(T) T1 = T0 - q_W_m2 * r.thickness_m / r.k_W_mK interface_temperatures.append(T1) z0 = z1 T0 = T1 z_all = np.concatenate(z_segments) T_all = np.concatenate(T_segments) return z_all, T_all, interface_temperatures
[ドキュメント] def air_properties_simple(T_film_K: float, k_air_ref: float) -> Tuple[float, float, float, float]: """ 単純な空気物性モデルを計算します。 室温から中程度の温度範囲での空気の物性を近似的に計算します。 熱伝導率 ``k_air`` はユーザー指定の参照値に基づいて温度依存性を持ち、 動粘度、熱拡散率、プラントル数も計算されます。 :param T_film_K: float: 空気膜の絶対温度 [K]。 :param k_air_ref: float: 参照熱伝導率 [W/mK]。 :returns: Tuple[float, float, float, float]: - k_air: 熱伝導率 [W/mK] - nu: 動粘度 [m^2/s] - alpha: 熱拡散率 [m^2/s] - Pr: プラントル数 [-] """ # Simple approximations around room-to-moderate temperatures. # Keep k tied to user-given reference value, with weak T dependence. k_air = k_air_ref * (T_film_K / 300.0) ** 0.10 nu = 15.89e-6 * (T_film_K / 300.0) ** 1.70 alpha = 22.0e-6 * (T_film_K / 300.0) ** 1.60 Pr = nu / alpha return k_air, nu, alpha, Pr
[ドキュメント] def horizontal_plate_h_nat( T_surface_C: float, T_air_C: float, L_char_m: float, k_air_ref: float, side: str, ) -> float: """ 水平平板の自然対流熱伝達係数を計算します。 レイリー数とヌセルト数相関を用いて、水平平板の自然対流熱伝達係数 ``h`` を計算します。 表面と空気の温度差が極めて小さい場合は、熱伝達係数が非常に小さな正の値として返されます。 相関は、浮力的に有利な(不安定)および不利な(安定)ケースを考慮に入れます。 :param T_surface_C: float: 表面温度 [°C]。 :param T_air_C: float: 周囲空気温度 [°C]。 :param L_char_m: float: 特性長さ [m]。 :param k_air_ref: float: 空気の参照熱伝導率 [W/mK]。 :param side: str: サンプルの「top」(上部)または「bottom」(下部)表面を指定。 :returns: float: 自然対流熱伝達係数 [W/m^2/K]。 :raises ValueError: ``side`` が 'top' または 'bottom' 以外の場合。 """ dT = T_surface_C - T_air_C if abs(dT) < 1e-12: return 1e-9 T_film_K = 0.5 * (T_surface_C + T_air_C) + 273.15 beta = 1.0 / T_film_K k_air, nu, alpha, _pr = air_properties_simple(T_film_K, k_air_ref) Ra = G * beta * abs(dT) * (L_char_m ** 3) / (nu * alpha) Ra = max(Ra, 1e-12) # Determine whether orientation is buoyancy-favorable ("unstable") or stable. # Top surface: # hot top surface -> unstable # cold top surface -> stable # Bottom surface: # hot bottom surface -> stable # cold bottom surface -> unstable if side == "top": unstable = dT > 0.0 elif side == "bottom": unstable = dT < 0.0 else: raise ValueError("side must be 'top' or 'bottom'") if unstable: if Ra < 1e7: Nu = 0.54 * (Ra ** 0.25) else: Nu = 0.15 * (Ra ** (1.0 / 3.0)) else: Nu = 0.27 * (Ra ** 0.25) h = Nu * k_air / L_char_m return max(h, 1e-9)
[ドキュメント] def top_surface_flux_downward( T_air_top_C: float, T_surface_top_C: float, eps_top: float, L_char_top_m: float, k_air_ref: float, T_rad_top_C: float | None = None, ) -> Tuple[float, float, float]: """ 上部周囲から上部サンプル表面への正味の下向き熱流束を計算します。 自然対流と熱放射による熱流束の合計を計算します。 熱流束は下向きが正と定義されます。 :param T_air_top_C: float: 上部周囲空気温度 [°C]。 :param T_surface_top_C: float: 上部サンプル表面温度 [°C]。 :param eps_top: float: 上部表面の放射率 [-]。 :param L_char_top_m: float: 上部自然対流の特性長さ [m]。 :param k_air_ref: float: 空気の参照熱伝導率 [W/mK]。 :param T_rad_top_C: float | None, optional: 上部放射環境温度 [°C]。Noneの場合、`T_air_top_C` が使用されます。 :returns: Tuple[float, float, float]: 合計熱流束、対流熱流束、放射熱流束 [W/m^2] のタプル。 """ if T_rad_top_C is None: T_rad_top_C = T_air_top_C h_top = horizontal_plate_h_nat( T_surface_C=T_surface_top_C, T_air_C=T_air_top_C, L_char_m=L_char_top_m, k_air_ref=k_air_ref, side="top", ) q_conv = h_top * (T_air_top_C - T_surface_top_C) q_rad = eps_top * SIGMA_SB * ( (T_air_top_C + 273.15) ** 4 - (T_surface_top_C + 273.15) ** 4 ) return q_conv + q_rad, q_conv, q_rad
[ドキュメント] def bottom_surface_flux_downward( T_surface_bottom_C: float, T_air_bottom_C: float, eps_bottom: float, L_char_bottom_m: float, k_air_ref: float, T_rad_bottom_C: float | None = None, ) -> Tuple[float, float, float]: """ 下部サンプル表面から下部周囲へ向かう正味の下向き熱流束を計算します。 自然対流と熱放射による熱流束の合計を計算します。 熱流束は下向きが正と定義されます。 :param T_surface_bottom_C: float: 下部サンプル表面温度 [°C]。 :param T_air_bottom_C: float: 下部周囲空気温度 [°C]。 :param eps_bottom: float: 下部表面の放射率 [-]。 :param L_char_bottom_m: float: 下部自然対流の特性長さ [m]。 :param k_air_ref: float: 空気の参照熱伝導率 [W/mK]。 :param T_rad_bottom_C: float | None, optional: 下部放射環境温度 [°C]。Noneの場合、`T_air_bottom_C` が使用されます。 :returns: Tuple[float, float, float]: 合計熱流束、対流熱流束、放射熱流束 [W/m^2] のタプル。 """ if T_rad_bottom_C is None: T_rad_bottom_C = T_air_bottom_C h_bottom = horizontal_plate_h_nat( T_surface_C=T_surface_bottom_C, T_air_C=T_air_bottom_C, L_char_m=L_char_bottom_m, k_air_ref=k_air_ref, side="bottom", ) q_conv = h_bottom * (T_surface_bottom_C - T_air_bottom_C) q_rad = eps_bottom * SIGMA_SB * ( (T_surface_bottom_C + 273.15) ** 4 - (T_rad_bottom_C + 273.15) ** 4 ) return q_conv + q_rad, q_conv, q_rad
[ドキュメント] def solve_conv_rad( args: argparse.Namespace, internal_regions: List[Region], n_points_per_region: int = 80, ) -> dict: """ 'conv_rad'モードで定常状態の温度プロファイルを解きます。 上部と下部の表面温度を通じて暗黙的に決定される未知の熱流束を、 上部と下部からの熱流束が一致するように上部表面温度について解くことで見つけます。 二分法を使用して収束させます。 :param args: argparse.Namespace: コマンドライン引数を格納するオブジェクト。 :param internal_regions: List[Region]: 内部スタック領域のリスト。 :param n_points_per_region: int, optional: 各領域でプロットのために生成する点の数 (デフォルトは80)。 :returns: dict: 計算結果を格納した辞書。以下のキーが含まれます: - 'q': float: 熱流束 [W/m^2]。 - 'R_internal': float: 内部スタックの熱抵抗 [m^2K/W]。 - 'R_top_eff': float: 有効な上部表面熱抵抗 [m^2K/W]。 - 'R_bottom_eff': float: 有効な下部表面熱抵抗 [m^2K/W]。 - 'Ts_top': float: 上部サンプル表面温度 [°C]。 - 'Ts_bottom': float: 下部サンプル表面温度 [°C]。 - 'q_conv_top': float: 上部対流熱流束 [W/m^2]。 - 'q_rad_top': float: 上部放射熱流束 [W/m^2]。 - 'q_conv_bottom': float: 下部対流熱流束 [W/m^2]。 - 'q_rad_bottom': float: 下部放射熱流束 [W/m^2]。 - 'z_phys': np.ndarray: 物理スケールのZ座標配列。 - 'T_phys': np.ndarray: 物理スケールの温度配列。 - 'interface_temps_internal': List[float]: 内部界面温度のリスト。 - 'interface_temps_full': List[float]: 全界面温度のリスト(空気温度と表面温度を含む)。 :raises RuntimeError: 上部表面温度の解の区間を見つけられなかった場合。 """ R_internal = sum(r.resistance_m2K_W for r in internal_regions) Th = args.Th Tc = args.Tc def residual(Ts_top_C: float) -> float: """ 上部表面温度に対する熱流束残差を計算します。 上部からの熱流束と下部への熱流束の差が0になるように求めます。 """ q_top, _, _ = top_surface_flux_downward( T_air_top_C=Th, T_surface_top_C=Ts_top_C, eps_top=args.eps_top, L_char_top_m=args.L_top, k_air_ref=args.k_air, T_rad_top_C=args.T_rad_top, ) Ts_bottom_C = Ts_top_C - q_top * R_internal q_bottom, _, _ = bottom_surface_flux_downward( T_surface_bottom_C=Ts_bottom_C, T_air_bottom_C=Tc, eps_bottom=args.eps_bottom, L_char_bottom_m=args.L_bottom, k_air_ref=args.k_air, T_rad_bottom_C=args.T_rad_bottom, ) return q_top - q_bottom # Robust bracket over a generous temperature range T_lo = min(Tc, Th) - 200.0 T_hi = max(Tc, Th) + 200.0 f_lo = residual(T_lo) f_hi = residual(T_hi) expand_count = 0 while f_lo * f_hi > 0 and expand_count < 20: T_lo -= 100.0 T_hi += 100.0 f_lo = residual(T_lo) f_hi = residual(T_hi) expand_count += 1 if f_lo * f_hi > 0: raise RuntimeError("Failed to bracket conv_rad solution for top surface temperature.") # Bisection for _ in range(200): T_mid = 0.5 * (T_lo + T_hi) f_mid = residual(T_mid) if abs(f_mid) < 1e-9: T_lo = T_hi = T_mid break if f_lo * f_mid <= 0: T_hi = T_mid f_hi = f_mid else: T_lo = T_mid f_lo = f_mid Ts_top = 0.5 * (T_lo + T_hi) q_top, q_conv_top, q_rad_top = top_surface_flux_downward( T_air_top_C=Th, T_surface_top_C=Ts_top, eps_top=args.eps_top, L_char_top_m=args.L_top, k_air_ref=args.k_air, T_rad_top_C=args.T_rad_top, ) q = q_top Ts_bottom = Ts_top - q * R_internal q_bottom, q_conv_bottom, q_rad_bottom = bottom_surface_flux_downward( T_surface_bottom_C=Ts_bottom, T_air_bottom_C=Tc, eps_bottom=args.eps_bottom, L_char_bottom_m=args.L_bottom, k_air_ref=args.k_air, T_rad_bottom_C=args.T_rad_bottom, ) # Internal profile only (physical layers) z_phys, T_phys, interface_temps_internal = compute_piecewise_profile( T_top_C=Ts_top, regions=internal_regions, q_W_m2=q, n_points_per_region=n_points_per_region, ) # Add air temperatures for reporting purposes interface_temps_full = [Th, Ts_top, *interface_temps_internal[1:-1], Ts_bottom, Tc] R_top_eff = (Th - Ts_top) / q if abs(q) > 1e-18 else np.inf R_bottom_eff = (Ts_bottom - Tc) / q if abs(q) > 1e-18 else np.inf return { "q": q, "R_internal": R_internal, "R_top_eff": R_top_eff, "R_bottom_eff": R_bottom_eff, "Ts_top": Ts_top, "Ts_bottom": Ts_bottom, "q_conv_top": q_conv_top, "q_rad_top": q_rad_top, "q_conv_bottom": q_conv_bottom, "q_rad_bottom": q_rad_bottom, "z_phys": z_phys, "T_phys": T_phys, "interface_temps_internal": interface_temps_internal, "interface_temps_full": interface_temps_full, }
[ドキュメント] def compute_schematic_positions( region_names: List[str], region_categories: List[str], layer_width: float, interface_width: float, surface_width: float, ) -> Tuple[np.ndarray, List[float], List[float]]: """ 模式図プロットのために、各領域の相対的な位置と幅を計算します。 各領域のカテゴリ(層、界面、表面)に基づいて、指定された表示幅を割り当て、 スタック内の模式的な位置(エッジと中心)を決定します。 :param region_names: List[str]: 各領域の名前のリスト。 :param region_categories: List[str]: 各領域のカテゴリのリスト("layer", "interface", "surface")。 :param layer_width: float: 「layer」カテゴリの領域に割り当てる模式的な表示幅。 :param interface_width: float: 「interface」カテゴリの領域に割り当てる模式的な表示幅。 :param surface_width: float: 「surface」カテゴリの領域に割り当てる模式的な表示幅。 :returns: Tuple[np.ndarray, List[float], List[float]]: - np.ndarray: 各領域の模式的な開始および終了位置の配列。 - List[float]: 各領域の模式的な幅のリスト。 - List[float]: 各領域の模式的な中心位置のリスト。 """ widths = [] for cat in region_categories: if cat == "layer": widths.append(layer_width) elif cat == "interface": widths.append(interface_width) else: widths.append(surface_width) edges = [0.0] for w in widths: edges.append(edges[-1] + w) centers = [(edges[i] + edges[i + 1]) / 2 for i in range(len(widths))] return np.array(edges), widths, centers
[ドキュメント] def plot_profiles_delta( z_phys: np.ndarray, T_phys: np.ndarray, regions: List[Region], interface_temperatures: List[float], out_png: str, schematic_layer_width: float, schematic_interface_width: float, schematic_surface_width: float, ) -> None: """ 'delta'モードの温度プロファイルを物理スケールと模式図スケールでプロットし、PNGファイルに保存します。 上段のプロットは実際の厚さに基づく物理的な温度プロファイルを示し、 下段のプロットは各領域に一定の幅を割り当てた模式的な温度プロファイルを示します。 :param z_phys: np.ndarray: 物理スケールのZ座標配列。 :param T_phys: np.ndarray: 物理スケールの温度配列。 :param regions: List[Region]: 全スタック領域のリスト。 :param interface_temperatures: List[float]: 各界面の温度リスト。 :param out_png: str: 出力PNGファイル名。 :param schematic_layer_width: float: 「layer」カテゴリの領域の模式的な表示幅。 :param schematic_interface_width: float: 「interface」カテゴリの領域の模式的な表示幅。 :param schematic_surface_width: float: 「surface」カテゴリの領域の模式的な表示幅。 :returns: None """ fig, (ax1, ax2) = plt.subplots( 2, 1, figsize=(10, 8), constrained_layout=True, sharey=True ) ax1.plot(z_phys * 1e6, T_phys, lw=2) ax1.set_title("Temperature profile (physical scale)") ax1.set_xlabel("Depth z [µm]") ax1.set_ylabel("Temperature [°C]") ax1.grid(True, alpha=0.3) z_boundary = 0.0 for r in regions[:-1]: z_boundary += r.thickness_m ax1.axvline(z_boundary * 1e6, ls="--", alpha=0.4) z_start = 0.0 y_text = T_phys.max() - 0.03 * (T_phys.max() - T_phys.min() + 1e-12) for r in regions: z_end = z_start + r.thickness_m z_mid = 0.5 * (z_start + z_end) ax1.text(z_mid * 1e6, y_text, r.name, ha="center", va="top", fontsize=8) z_start = z_end names = [r.name for r in regions] cats = [r.category for r in regions] x_edges, _, centers = compute_schematic_positions( names, cats, layer_width=schematic_layer_width, interface_width=schematic_interface_width, surface_width=schematic_surface_width, ) x_plot = [] T_plot = [] for i in range(len(regions)): x0, x1 = x_edges[i], x_edges[i + 1] T0, T1 = interface_temperatures[i], interface_temperatures[i + 1] xx = np.linspace(x0, x1, 40, endpoint=(i == len(regions) - 1)) TT = np.linspace(T0, T1, 40, endpoint=(i == len(regions) - 1)) x_plot.append(xx) T_plot.append(TT) ax2.plot(np.concatenate(x_plot), np.concatenate(T_plot), lw=2) ax2.set_title("Temperature profile (schematic / deformed scale)") ax2.set_xlabel("Stack position [schematic scale]") ax2.set_ylabel("Temperature [°C]") ax2.grid(True, alpha=0.3) for i in range(len(regions) - 1): ax2.axvline(x_edges[i + 1], ls="--", alpha=0.4) ymin, ymax = ax2.get_ylim() for i, name in enumerate(names): x0, x1 = x_edges[i], x_edges[i + 1] ax2.axvspan(x0, x1, alpha=0.08) ax2.text(centers[i], ymax - 0.03 * (ymax - ymin + 1e-12), name, ha="center", va="top", fontsize=8) fig.savefig(out_png, dpi=160) print() print(f"Saved plot: {out_png}") plt.show() plt.close(fig)
[ドキュメント] def plot_profiles_conv_rad( z_phys: np.ndarray, T_phys: np.ndarray, internal_regions: List[Region], interface_temps_internal: List[float], Th: float, Tc: float, Ts_top: float, Ts_bottom: float, out_png: str, schematic_layer_width: float, schematic_interface_width: float, schematic_surface_width: float, ) -> None: """ 'conv_rad'モードの温度プロファイルを物理スケールと模式図スケールでプロットし、PNGファイルに保存します。 上段のプロットは内部スタックのみの物理的な温度プロファイルを示し、 下段のプロットは表面の対流・放射領域を含む模式的な温度プロファイルを示します。 :param z_phys: np.ndarray: 物理スケールのZ座標配列(内部スタックのみ)。 :param T_phys: np.ndarray: 物理スケールの温度配列(内部スタックのみ)。 :param internal_regions: List[Region]: 内部スタック領域のリスト。 :param interface_temps_internal: List[float]: 内部界面の温度リスト。 :param Th: float: 上部周囲空気温度 [°C]。 :param Tc: float: 下部周囲空気温度 [°C]。 :param Ts_top: float: 上部サンプル表面温度 [°C]。 :param Ts_bottom: float: 下部サンプル表面温度 [°C]。 :param out_png: str: 出力PNGファイル名。 :param schematic_layer_width: float: 「layer」カテゴリの領域の模式的な表示幅。 :param schematic_interface_width: float: 「interface」カテゴリの領域の模式的な表示幅。 :param schematic_surface_width: float: 「surface」カテゴリの領域の模式的な表示幅。 :returns: None """ fig, (ax1, ax2) = plt.subplots( 2, 1, figsize=(10, 8), constrained_layout=True, sharey=True ) # Physical scale: internal stack only ax1.plot(z_phys * 1e6, T_phys, lw=2) ax1.set_title("Temperature profile (physical scale)") ax1.set_xlabel("Depth z [µm] (internal stack only)") ax1.set_ylabel("Temperature [°C]") ax1.grid(True, alpha=0.3) z_boundary = 0.0 for r in internal_regions[:-1]: z_boundary += r.thickness_m ax1.axvline(z_boundary * 1e6, ls="--", alpha=0.4) z_start = 0.0 y_text = T_phys.max() - 0.03 * (T_phys.max() - T_phys.min() + 1e-12) for r in internal_regions: z_end = z_start + r.thickness_m z_mid = 0.5 * (z_start + z_end) ax1.text(z_mid * 1e6, y_text, r.name, ha="center", va="top", fontsize=8) z_start = z_end # Schematic scale: include pseudo-surface regions names = ["Top conv+rad", *[r.name for r in internal_regions], "Bottom conv+rad"] cats = ["surface", *[r.category for r in internal_regions], "surface"] x_edges, _, centers = compute_schematic_positions( names, cats, layer_width=schematic_layer_width, interface_width=schematic_interface_width, surface_width=schematic_surface_width, ) temps = [Th, Ts_top, *interface_temps_internal[1:-1], Ts_bottom, Tc] x_plot = [] T_plot = [] for i in range(len(names)): x0, x1 = x_edges[i], x_edges[i + 1] T0, T1 = temps[i], temps[i + 1] xx = np.linspace(x0, x1, 40, endpoint=(i == len(names) - 1)) TT = np.linspace(T0, T1, 40, endpoint=(i == len(names) - 1)) x_plot.append(xx) T_plot.append(TT) ax2.plot(np.concatenate(x_plot), np.concatenate(T_plot), lw=2) ax2.set_title("Temperature profile (schematic / deformed scale)") ax2.set_xlabel("Stack position [schematic scale]") ax2.set_ylabel("Temperature [°C]") ax2.grid(True, alpha=0.3) for i in range(len(names) - 1): ax2.axvline(x_edges[i + 1], ls="--", alpha=0.4) ymin, ymax = ax2.get_ylim() for i, name in enumerate(names): x0, x1 = x_edges[i], x_edges[i + 1] ax2.axvspan(x0, x1, alpha=0.08) ax2.text(centers[i], ymax - 0.03 * (ymax - ymin + 1e-12), name, ha="center", va="top", fontsize=8) fig.savefig(out_png, dpi=160) plt.show() plt.close(fig)
[ドキュメント] def build_parser() -> argparse.ArgumentParser: """ コマンドライン引数をパースするための ``ArgumentParser`` オブジェクトを構築します。 この関数は、スタックの層の厚さや熱伝導率、表面モデル('delta'または'conv_rad')、 温度、プロット設定など、スクリプトの動作を制御するための様々な引数を定義します。 :returns: argparse.ArgumentParser: 設定済みのArgumentParserオブジェクト。 """ p = argparse.ArgumentParser( description="Steady-state 1D temperature profile of layered TE stack." ) # common p.add_argument("--surface_mode", type=str, default="delta", choices=["delta", "conv_rad"], help="表面モデル: 明示的な空気層 (delta) または自然対流と放射 (conv_rad)。") p.add_argument("--Th", type=float, default=100.0, help="上部空気温度 [°C]") p.add_argument("--Tc", type=float, default=20.0, help="下部空気温度 [°C]") p.add_argument("--k_air", type=float, default=0.026, help="空気の参照熱伝導率 [W/mK]") # delta-mode boundary layers p.add_argument("--delta_top", type=float, default=50e-9, help="上部空気境界層の厚さ [m] (deltaモード)") p.add_argument("--delta_bottom", type=float, default=50e-9, help="下部空気境界層の厚さ [m] (deltaモード)") # internal layers p.add_argument("--d1", type=float, default=200e-9, help="上部電極の厚さ [m]") p.add_argument("--k1", type=float, default=100.0, help="上部電極の熱伝導率 [W/mK]") p.add_argument("--delta12", type=float, default=50e-9, help="界面1/2の等価厚さ [m]") p.add_argument("--k12", type=float, default=0.3, help="界面1/2の有効熱伝導率 [W/mK]") p.add_argument("--d2", type=float, default=2e-6, help="TE半導体の厚さ [m]") p.add_argument("--k2", type=float, default=0.5, help="TE半導体の熱伝導率 [W/mK]") p.add_argument("--k23", type=float, default=0.3, help="界面2/3の有効熱伝導率 [W/mK]") p.add_argument("--delta23", type=float, default=50e-9, help="界面2/3の等価厚さ [m]") p.add_argument("--d3", type=float, default=200e-9, help="下部電極の厚さ [m]") p.add_argument("--k3", type=float, default=100.0, help="下部電極の熱伝導率 [W/mK]") # conv_rad mode p.add_argument("--L_top", type=float, default=0.01, help="上部自然対流の特性長さ [m]") p.add_argument("--L_bottom", type=float, default=0.01, help="下部自然対流の特性長さ [m]") p.add_argument("--eps_top", type=float, default=0.8, help="上部表面放射率") p.add_argument("--eps_bottom", type=float, default=0.8, help="下部表面放射率") p.add_argument("--T_rad_top", type=float, default=100.0, help="上部放射温度 [°C]") p.add_argument("--T_rad_bottom", type=float, default=20.0, help="下部放射温度 [°C]") # plotting p.add_argument("--save", type=str, default="layered_temperature_profile.png", help="出力PNGファイル名") p.add_argument("--npts", type=int, default=80, help="明示的な領域あたりのプロット点数") # schematic widths p.add_argument("--schematic_layer_width", type=float, default=1.0, help="通常の層の表示幅") p.add_argument("--schematic_interface_width", type=float, default=0.5, help="界面層の表示幅") p.add_argument("--schematic_surface_width", type=float, default=0.8, help="表面領域の表示幅") return p
[ドキュメント] def main() -> None: """ スクリプトのメイン実行関数です。 コマンドライン引数をパースし、選択された表面モード('delta'または'conv_rad')に基づいて、 熱電スタックの温度プロファイルを計算、レポート表示、およびプロットを実行します。 """ parser = build_parser() args = parser.parse_args() if args.surface_mode == "delta": regions = build_regions_delta(args) q = (args.Th - args.Tc) / sum(r.resistance_m2K_W for r in regions) z, T, interface_temperatures = compute_piecewise_profile( T_top_C=args.Th, regions=regions, q_W_m2=q, n_points_per_region=args.npts, ) print_report_delta(args, regions, q, interface_temperatures) plot_profiles_delta( z_phys=z, T_phys=T, regions=regions, interface_temperatures=interface_temperatures, out_png=args.save, schematic_layer_width=args.schematic_layer_width, schematic_interface_width=args.schematic_interface_width, schematic_surface_width=args.schematic_surface_width, ) if args.surface_mode == "delta": regions = build_regions_delta(args) # 注意: compute_piecewise_profileは熱流束qを戻り値に含みません。 # 以下の行は既存のコードの論理的な不整合ですが、変更の指示がないためそのまま残します。 # 実行時にはエラーが発生する可能性があります (ValueError: too many values to unpack)。 z, T, q, interface_temperatures = compute_piecewise_profile( T_top_C=args.Th, regions=regions, q_W_m2=(args.Th - args.Tc) / sum(r.resistance_m2K_W for r in regions), n_points_per_region=args.npts, ) print_report_delta(args, regions, q, interface_temperatures) plot_profiles_delta( z_phys=z, T_phys=T, regions=regions, interface_temperatures=interface_temperatures, out_png=args.save, schematic_layer_width=args.schematic_layer_width, schematic_interface_width=args.schematic_interface_width, schematic_surface_width=args.schematic_surface_width, ) elif args.surface_mode == "conv_rad": internal_regions = build_internal_regions(args) result = solve_conv_rad(args, internal_regions, n_points_per_region=args.npts) print_report_conv_rad(args, internal_regions, result) plot_profiles_conv_rad( z_phys=result["z_phys"], T_phys=result["T_phys"], internal_regions=internal_regions, interface_temps_internal=result["interface_temps_internal"], Th=args.Th, Tc=args.Tc, Ts_top=result["Ts_top"], Ts_bottom=result["Ts_bottom"], out_png=args.save, schematic_layer_width=args.schematic_layer_width, schematic_interface_width=args.schematic_interface_width, schematic_surface_width=args.schematic_surface_width, ) else: raise ValueError("Unknown surface_mode")
if __name__ == "__main__": main()