RORK LABEN
MAX — Rork MaxはネイティブSwiftアプリを生成し、iPhone・iPad・Apple Watch・Apple TV・Vision Pro・iMessageに対応しますNATIVE — Rork MaxはAR/LiDARスキャン、Metalの3D、ウィジェット、Live Activities、HealthKitなどネイティブ機能を解放しますFUNDING — Rorkはa16zから280万ドルを調達し、月間訪問は74万を超え、成長率は85%に達していますRN — 通常のRorkはReact Native(Expo)でiOSとAndroidアプリをまとめて生成しますFOCUS — Rorkはネイティブモバイルアプリ専業で、Web中心のBoltやLovableと一線を画しますPRICING — 無料で始められ、有料プランは月25ドルから、Rork Maxは月200ドルで2クリックのApp Store公開に対応しますMAX — Rork MaxはネイティブSwiftアプリを生成し、iPhone・iPad・Apple Watch・Apple TV・Vision Pro・iMessageに対応しますNATIVE — Rork MaxはAR/LiDARスキャン、Metalの3D、ウィジェット、Live Activities、HealthKitなどネイティブ機能を解放しますFUNDING — Rorkはa16zから280万ドルを調達し、月間訪問は74万を超え、成長率は85%に達していますRN — 通常のRorkはReact Native(Expo)でiOSとAndroidアプリをまとめて生成しますFOCUS — Rorkはネイティブモバイルアプリ専業で、Web中心のBoltやLovableと一線を画しますPRICING — 無料で始められ、有料プランは月25ドルから、Rork Maxは月200ドルで2クリックのApp Store公開に対応します
記事一覧/アプリ開発
アプリ開発/2026-06-29上級

従量制回線と低データモードを尊重して先読みを止める通信設計

画像中心の Rork(Expo)アプリで、従量制回線や低データモードのときに先読みを抑える通信設計を扱います。NetInfo の details から回線の性質を読み、画質を落とし、プリフェッチを遅延させる適応ロジックを実装コードとともに整理します。

Rork473Expo122NetInfo低データモードプリフェッチ通信最適化

プレミアム記事

私が運用している壁紙アプリのレビューに、あるとき「Wi-Fi のないところで開いたら、あっという間にギガが減った」という指摘が届きました。原因はすぐに分かりました。一覧画面で次のページの高解像度画像を片っ端から先読みしていたのです。Wi-Fi なら快適さに貢献する挙動が、従量制のモバイル回線では利用者の通信量を静かに食いつぶしていました。画像を多く扱うアプリほど、この「良かれと思った先読み」が逆効果になりやすいと感じています。

ここでは Rork で生成した Expo アプリを題材に、回線の性質を読み取って先読みと画質を適応的に切り替える通信設計を整理します。鍵になるのは、回線を「つながっているか/いないか」の二値で見るのをやめ、「従量制か」「低速か」「低データモードか」まで踏み込んで判断することです。

「オンラインか否か」だけで判断するのをやめる

多くのアプリは isConnected だけを見て、つながっていれば全部やる、という作りになっています。しかし利用者の回線は一様ではありません。NetInfo の details には、判断に使える情報がもっと含まれています。

判定軸取得元(NetInfo)意味するもの
従量制かどうかdetails.isConnectionExpensiveOS が「お金のかかる回線」と見なしている(テザリング等を含む)
回線種別type(wifi / cellular / ...)Wi-Fi かモバイルか
セルラー世代details.cellularGeneration(3g/4g/5g)速度の目安(3g なら重い先読みは避ける)
低データモード下記参照利用者が明示的に通信を絞っている

isConnectionExpensive は OS が判断した「従量制相当」のフラグで、テザリング経由の Wi-Fi なども含まれます。これを尊重するだけでも、レビューにあったような「ギガが溶ける」事故の多くは防げます。

低データモードについては補足が要ります。iOS の Low Data Mode / Android のデータセーバーは、アプリから直接「オンかどうか」を読む統一 API が乏しいのが実情です。実務では、isConnectionExpensive を低データモードの代理シグナルとして扱い、加えて「アプリ内に節約スイッチを用意して利用者自身に選ばせる」二段構えにするのが現実的です。OS の意思(従量制判定)とアプリの設定(明示スイッチ)を両方ポリシーに合流させます。

通信判定を一箇所に集約する

回線の判定を画面ごとに散らすと、必ず判断がばらつきます。回線情報を1つのポリシーに変換するレイヤーを作り、各画面はそのポリシーだけを見るようにします。

// net/networkPolicy.ts
import NetInfo, { NetInfoState } from "@react-native-community/netinfo";
 
export type PrefetchLevel = "full" | "lite" | "off";
export type ImageQuality = "high" | "medium" | "low";
 
export interface NetworkPolicy {
  prefetch: PrefetchLevel;   // 先読みの強さ
  imageQuality: ImageQuality; // 取得する画質
  allowAutoplay: boolean;     // 動画・GIF の自動再生可否
}
 
// 回線状態 + アプリの節約設定 を 1 つのポリシーへ変換する
export function derivePolicy(
  state: NetInfoState,
  saverEnabled: boolean
): NetworkPolicy {
  // つながっていない
  if (!state.isConnected) {
    return { prefetch: "off", imageQuality: "low", allowAutoplay: false };
  }
 
  const expensive = !!(state.details as any)?.isConnectionExpensive;
  const gen = (state.details as any)?.cellularGeneration as string | undefined;
  const isSlowCellular = state.type === "cellular" && (gen === "2g" || gen === "3g");
 
  // 利用者が節約を選んでいる、または OS が従量制と判断、または低速
  if (saverEnabled || expensive || isSlowCellular) {
    return { prefetch: "off", imageQuality: "medium", allowAutoplay: false };
  }
 
  // 通常のモバイル(4g/5g・非従量制)
  if (state.type === "cellular") {
    return { prefetch: "lite", imageQuality: "high", allowAutoplay: false };
  }
 
  // Wi-Fi(非従量制)— 全力で先読み
  return { prefetch: "full", imageQuality: "high", allowAutoplay: true };
}

このポリシーを React のコンテキストで配ると、各画面は useNetworkPolicy() を読むだけで一貫した判断ができます。

// net/NetworkPolicyProvider.tsx
import React, { createContext, useContext, useEffect, useState } from "react";
import NetInfo from "@react-native-community/netinfo";
import { derivePolicy, NetworkPolicy } from "./networkPolicy";
import { useDataSaver } from "../settings/useDataSaver"; // アプリ内の節約スイッチ
 
const Ctx = createContext<NetworkPolicy>({
  prefetch: "lite", imageQuality: "high", allowAutoplay: false,
});
 
export function NetworkPolicyProvider({ children }: { children: React.ReactNode }) {
  const saver = useDataSaver();
  const [policy, setPolicy] = useState<NetworkPolicy>(Ctx._currentValue);
 
  useEffect(() => {
    // 購読開始時に現在値を一度取得し、以後は変化を購読
    const unsub = NetInfo.addEventListener((state) => {
      setPolicy(derivePolicy(state, saver));
    });
    NetInfo.fetch().then((state) => setPolicy(derivePolicy(state, saver)));
    return () => unsub();
  }, [saver]);
 
  return <Ctx.Provider value={policy}>{children}</Ctx.Provider>;
}
 
export const useNetworkPolicy = () => useContext(Ctx);

ポイントは、節約スイッチ(saver)の変化も依存に入れることです。利用者が設定でスイッチを切り替えた瞬間にポリシーが再計算され、先読みの挙動が即座に追従します。

ここまでお読みいただきありがとうございます。

この記事の続きを読む

この先には、実装コードやベンチマーク結果など、実務でお役に立てる内容をご用意しています。このサイトは広告を掲載しておらず、サーバーや開発にかかる費用はメンバーの皆様のご支援で成り立っています。もしお役に立てていましたら、ご支援いただけますと大変ありがたいです。

この記事で得られること
回線が従量制・低速・低データモードのとき、自動で先読みを止めて画質を落とす適応プリフェッチの実装を手に入れられる
NetInfo の details から isConnectionExpensive やセルラー世代を読み、ポリシーに変換する一元化された通信判定レイヤーを組み立てられる
全部先読みするか何もしないかの二択をやめ、回線に応じて段階的に振る舞いを変える設計に移行できる
Stripe による安全な決済 · いつでもキャンセル可能

この記事を購入する

この先の内容をすべてお読みいただけます。一度のご購入で、いつでも何度でもアクセスできます。このサイトは広告を掲載しておらず、皆さまのご支援がサーバー費用などの運営を支えています。

または
メンバーシップなら全記事が読み放題 →
シェア

お読みいただきありがとうございます

Rork Lab は広告なしで運営しており、サーバー費用などの運営コストはメンバーシップのご支援で賄っています。実装コード・ベンチマーク・本番設計パターンなど、実務でお役立ていただける記事を毎日更新しています。もし読んでよかったと感じていただけましたら、ぜひご覧ください。

  • コピー&ペーストで使える実装コード付き
  • 毎日新しい上級ガイドを追加
  • ¥580/月 または ¥1,480 の永久アクセス
メンバーシップを見る →

関連記事

アプリ開発2026-06-27
「本当に削除しますか?」を出す前に — 取り消せる削除という選択肢
リストの項目を消すたびに確認ダイアログを出すと、ユーザーは反射的にOKを押すようになります。Rork(Expo)アプリで「取り消せる削除」を実装し、確認を出すべき場面と出さない場面を切り分ける設計ノートです。
アプリ開発2026-06-27
アップデート後の「新着情報」を、しつこくなく一度だけ出す設計 — バージョン境界とseen状態の作り方
更新後に出すアプリ内「新着情報」画面が、新規インストールにまで出たり毎回出たりして嫌われる問題を、バージョン境界とseen状態の設計で一度だけ確実に届ける。Rork(Expo)の実コードと、複数アプリで使い回す型までまとめます。
アプリ開発2026-06-27
端末の時計を戻されても「今日の1枚」が崩れない設計 — 日替わりコンテンツの日付境界とストリーク整合
日替わりの「今日のコンテンツ」がタイムゾーン移動や端末時計の巻き戻しで二重表示・欠落・ストリーク消失を起こす問題を、決定論的な日付キーと単調クロックで防ぐ設計を、Rork(Expo)アプリの実コードでまとめます。
📚RECOMMENDED BOOKS
大規模言語モデル入門
山田育矢
LLM開発
生成AIプロンプトエンジニアリング入門
我妻幸長
プロンプト
Claude CodeによるAI駆動開発入門
平川知秀
AI駆動開発
※ アフィリエイトリンクを含みます
もっと見る →