普通视图

发现新文章,点击刷新页面。
昨天 — 2026年1月17日iOS

使用pymobiledevice3进行iphone应用性能采集

作者 茫茫碌碌
2026年1月17日 16:05

执行步骤

安装依赖(未安装则执行)

pip install pymobiledevice3

启动Tunnel(系统版本>=17需要)

sudo pymobiledevice3 remote tunneld

获取性能数据(打开一个新的终端)

根据PID获取性能数据

# 获取PID
pymobiledevice3 developer dvt process-id-for-bundle-id com.xxx.xxx

# 获取指定PID的性能数据
pymobiledevice3 developer dvt sysmon process single -a pid=xxxx

获取全部进程的性能数据

pymobiledevice3 developer dvt sysmon process monitor --rsd xxxx:xxxx:xxxx::x 58524 0.01

简单方案

上面的多个步骤都是通过pymobiledevice3一个工具来实现的,因此是否可以一步就完成性能的采集?当然可以,通过深扒(并复制+改造)pymobiledevice3的源码,将所有操作封装到了一个脚本中~~~

"""
—————— 使用说明 ——————
usage: python3 ios-monitor.py [-h] [-g GAP] [-b BUNDLE_ID] [-r REPORT_HOST] [--detail] [--csv] [--debug]

ios设备性能收集

options:
  -h, --help            show this help message and exit
  -g, --gap GAP         性能数据获取时间间隔,默认1s
  -b, --bundle_id BUNDLE_ID
                        包名, 默认: com.mi.car.mobile
  -r, --report_host REPORT_HOST
                        性能采集数据上报地址
  --detail              输出详细信息
  --csv                 结果写入到CSV文件
  --debug               打印debug日志

—————— 示例 ——————
# 进行性能采集,ctrl+c 终止后写入csv文件
sudo python3 ios-monitor.py --csv

# 进行性能采集,数据上报到指定服务
sudo python3 ios-monitor.py -r 127.0.0.1:9311

"""
import argparse
import asyncio
import csv
import json
import logging
import os
import signal
import sys
import time
from functools import partial
import multiprocessing
from multiprocessing import Process

from pymobiledevice3.remote.common import TunnelProtocol
from pymobiledevice3.remote.module_imports import verify_tunnel_imports
from pymobiledevice3.remote.remote_service_discovery import RemoteServiceDiscoveryService
from pymobiledevice3.services.dvt.dvt_secure_socket_proxy import DvtSecureSocketProxyService
from pymobiledevice3.services.dvt.instruments.device_info import DeviceInfo
from pymobiledevice3.services.dvt.instruments.process_control import ProcessControl
from pymobiledevice3.services.dvt.instruments.sysmontap import Sysmontap
from pymobiledevice3.tunneld.server import TunneldRunner

import requests

TUNNELD_DEFAULT_ADDRESS = ('127.0.0.1', 49151)

logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s - %(message)s')
logger = logging.getLogger(__name__)

report_error_num = 0

csv_data_list = []
csv_fieldnames=['cpu', 'mem']

def run_tunneld():
    """ Start Tunneld service for remote tunneling
    sudo pymobiledevice3 remote tunneld
    """
    if not verify_tunnel_imports():
        logger.warning("verify_tunnel_imports false")
        return
    host = TUNNELD_DEFAULT_ADDRESS[0]
    port = TUNNELD_DEFAULT_ADDRESS[1]
    protocol = TunnelProtocol(TunnelProtocol.DEFAULT.value)

    tunneld_runner = partial(TunneldRunner.create, host, port, protocol=protocol, usb_monitor=True,
                             wifi_monitor=True, usbmux_monitor=True, mobdev2_monitor=True)

    tunneld_runner()
    return

def process_id_for_bundle_id(lockdown, app_bundle_identifier: str = "com.mi.car.mobile"):
    """ Get PID of a bundle identifier (only returns a valid value if its running). """
    with DvtSecureSocketProxyService(lockdown=lockdown) as dvt:
        return ProcessControl(dvt).process_identifier_for_bundle_identifier(app_bundle_identifier)

def sysmon_process_single(lockdown, pid, detail, report_host='', write_csv=False, tunnel_process=None):
    """ show a single snapshot of currently running processes. """
    count = 0
    result = []

    with DvtSecureSocketProxyService(lockdown=lockdown) as dvt:
        device_info = DeviceInfo(dvt)
        with Sysmontap(dvt) as sysmon:
            for process_snapshot in sysmon.iter_processes():
                count += 1
                if count < 2:
                    # first sample doesn't contain an initialized value for cpuUsage
                    continue
                for process in process_snapshot:
                    # print(process)
                    if str(process["pid"]) != str(pid):
                        continue
                    # adding "artificially" the execName field
                    process['execName'] = device_info.execname_for_pid(process['pid'])
                    result.append(process)
                # exit after single snapshot
                break
    if len(result) == 0:
        logger.info("[]")
        return
    
    cpu_usage = "%.2f" % result[0]['cpuUsage']
    mem = "%.2f" % (result[0]['memResidentSize'] / 1024 / 1024)
    
    if write_csv:
        csv_data = {'cpu': cpu_usage, 'mem': mem}
        csv_data_list.append(csv_data)

    if report_host:
        report(host=report_host, info=result[0], pid=str(pid), tunnel_process=tunnel_process)
        return

    if detail:
        logger.info(json.dumps(result, indent=4, ensure_ascii=False))
    else:
        logger.info("[CPU]{} %  [内存]{} MB".format(cpu_usage, mem))

def report(host: str, info: dict, pid: str, tunnel_process=None):
    global report_error_num
    url = 'http://%s/monitor/collect' % host
    mem = info['memResidentSize'] / 1024 / 1024
    data = {
        'device': 'ios-%s' % pid,
        'list': [0, info['cpuUsage'], 0, mem],
        'app_cpu_rate': info['cpuUsage'],
        'app_mem': mem,
        'timestamp': int(time.time() * 1000)
    }
    report_err = False
    try:
        resp = requests.post(url=url, json=data, timeout=(0.5, 0.5))
    except Exception:
        report_err = True

    if report_err is False and resp.status_code != 200:
        report_err = True

    if report_err:
        report_error_num += 1
        logger.warning("上报失败 %d" % report_error_num)
        if report_error_num > 5:
            logger.info("接收端已关闭, 监控退出")
            if tunnel_process:
                tunnel_process.terminate()
            sys.exit(0)
    else:
        cpu_usage = "%.2f" % info['cpuUsage']
        logger.info("report [CPU]{} %  [内存]{} MB".format(cpu_usage, mem))
        report_error_num = 0

def get_tunnel_addr(attemp_times=30):
    url = 'http://127.0.0.1:%d/' % TUNNELD_DEFAULT_ADDRESS[1]
    try_times = 0
    while try_times < attemp_times:
        try:
            logger.info('--- 获取设备连接信息')
            resp = requests.get(url=url, timeout=(1, 1)).json()
            for v in resp.values():
                if not v:
                    continue
                return v[0]['tunnel-address'], v[0]['tunnel-port']
        except Exception:
            pass
        try_times += 1
        time.sleep(1)
        continue
    logger.warning('--- 未找到ios设备')
    return None, None

def run_in_one(bundle_id: str, gap: int, detail: bool, report_host: str = '', write_csv: bool = False):
    logger.info('--- 连接设备')
    p = Process(target=run_tunneld, args=())
    p.start()

    def sys_exit(status: int = 0):
        p.terminate()
        # 写入csv
        if len(csv_data_list) > 0:
            logger.info('--- 写入CSV')
            filename = 'ios-monitor-result-%d.csv' % int(time.time())
            filepath = os.path.join(os.getcwd(), filename)
            with open(filepath, 'w', encoding='UTF8', newline='') as f:
                writer = csv.DictWriter(f, fieldnames=csv_fieldnames)
                writer.writeheader()
                writer.writerows(csv_data_list)
        
        logger.info(' --- 退出 ---')
        sys.exit(status)

    def signal_handler(*args, **kwargs):
        sys_exit(0)

    time.sleep(3)
    signal.signal(signal.SIGINT, signal_handler)

    addr, port = get_tunnel_addr(attemp_times=30)
    if not addr:
        sys_exit(1)

    logger.info("--- connect device: %s %d" % (addr, port))

    logger.debug('start run')
    rsd = RemoteServiceDiscoveryService(address=(addr, port))
    logger.debug('start rsd connect')
    asyncio.run(rsd.connect(), debug=True)
    time.sleep(1)
    logger.debug('get pid')
    pid = process_id_for_bundle_id(lockdown=rsd, app_bundle_identifier=bundle_id)
    logger.info('获取应用(%s)PID: %d' % (bundle_id, pid))

    while True:
        if not pid:
            pid = process_id_for_bundle_id(lockdown=rsd, app_bundle_identifier=bundle_id)
            logger.info('获取应用(%s)PID: %d' % (bundle_id, pid))
            time.sleep(0.3)
            continue

        try:
            sysmon_process_single(lockdown=rsd, pid=pid, detail=detail, report_host=report_host, write_csv=write_csv, tunnel_process=p)
            time.sleep(gap/1000)
        except Exception as err:
            logger.error('获取性能指标失败: {}'.format(err))
            addr, port = get_tunnel_addr(attemp_times=30)
            if not addr:
                sys_exit(1)

            pid = process_id_for_bundle_id(lockdown=rsd, app_bundle_identifier=bundle_id)
            logger.info('获取应用(%s)PID: %d' % (bundle_id, pid))

if __name__ == '__main__':
    multiprocessing.freeze_support()
    parser = argparse.ArgumentParser(description='ios设备性能收集', add_help=True)
    parser.add_argument('-g', '--gap', type=int, required=False, default=1000,
                        help='性能数据获取时间间隔(ms),默认1000ms')
    parser.add_argument('-b', '--bundle_id', required=False, default='com.mi.car.mobile',
                        help='包名, 默认: com.mi.car.mobile')
    parser.add_argument('-r', '--report_host', required=False, default='',
                        help='性能采集数据上报地址')
    parser.add_argument('--detail', default=False, action='store_true',
                        help='输出详细信息')
    parser.add_argument('--csv', default=False, action='store_true',
                        help='结果写入到CSV文件')
    parser.add_argument('--debug', default=False, action='store_true',
                        help='打印debug日志')

    args = parser.parse_args()

    if args.debug:
        logger.setLevel(logging.DEBUG)

    if not args.bundle_id:
        logger.error('bundle_id invalid')
        sys.exit(1)

    gap_ms = args.gap
    # 最低200ms间隔
    if gap_ms < 200:
        gap_ms = 200

    rpt_host = args.report_host
    # 上报到本机客户端
    if rpt_host in {'local', 'localhost', '*', '-', '9311'}:
        rpt_host = '127.0.0.1:9311'
    run_in_one(bundle_id=args.bundle_id, gap=gap_ms, detail=args.detail, report_host=rpt_host, write_csv=args.csv)

SwiftUI navigation stack 嵌套引起的导航错误

作者 tangzzzfan
2026年1月17日 00:26

最近我们在推进 ModernNavigation 插件化架构时,遇到了一个 SwiftUI 开发中非常经典、但也极其容易让人抓狂的“幽灵问题”:嵌套 NavigationStack。 作为一名在 iOS 领域摸爬滚打多年的开发者,我深知这种架构层面的“小瑕疵”如果不彻底理清,后续会导致手势失效、双标题、甚至 Path 状态莫名丢失等一系列连锁反应。 今天我把这个问题深度复盘了一下,总结出了一套更符合我们 Redux 思想的解决方案。

案发现场:为什么你的 Navigation 崩了? 我们在做插件化时,为了让模块独立,经常会习惯性地在 AppRoute 的 body 里写下这段代码: case .settings: NavigationStack { // 罪魁祸首在这里 SettingsView() }

当你从 HomeView(已经在一个 NavigationStack 内部)执行 router.push(.settings) 时,你就亲手制造了一个“栈中栈”。

症状分析:

  • 权限冲突:外层的 NavigationPath 想管进度,内层的 NavigationStack 也想管进度,SwiftUI 直接“摆烂”,导致侧滑返回可能直接回到 App 根部。
  • UI 灾难:你可能会在屏幕顶端看到两个叠加在一起的导航栏,或者是返回按钮莫名其妙消失。
  • Redux 状态断裂:内层栈的操作完全脱离了我们 NavigationStore 的掌控。

处方:区分“动作”而非“视图” 解决这个问题的核心思想只有一句话:Push 是一场“接力”,Sheet 是一场“派对”。

  • Push(推栈):它是当前导航流的延续,绝对不能自带 NavigationStack。
  • Sheet/Cover(弹窗):它开启了一个全新的、独立的导航流,必须自带 NavigationStack 来管理它自己的子页面。
  1. 给视图加点“环境感知” 为了让视图在“被推入”和“被弹出”时表现不同(比如弹窗时需要一个“关闭”按钮),我们引入一个 isModal 标识(不太优雅): struct SettingsView: View { var isModal: Bool = false @Environment(NavigationStore<AppRoute, AppSheet>.self) private var navStore

    var body: some View { List { ... } .navigationTitle("设置") .toolbar { if isModal { ToolbarItem(placement: .cancellationAction) { Button("关闭") { navStore.dispatch(.dismiss) } } } } } }

  2. 在路由层实现“插件化”分流 在我们的 RouteViewFactory 实现中,我们需要明确区分这两种场景。不需要写超大的 switch,而是让 Factory 能够根据上下文返回正确的包装: struct UserRouteFactory: RouteViewFactory { func view(for route: Any) -> AnyView? { // 方案 A:通过路由类型区分 if let userRoute = route as? UserRoute { return AnyView(SettingsView(isModal: false)) // 纯净视图用于 Push }

     // 方案 B:通过特定的 Sheet 路由类型
     if let sheet = route as? UserSheet {
         switch sheet {
         case .settingsModal:
             return AnyView(
                 NavigationStack { // 只有 Sheet 才包裹 Stack
                     SettingsView(isModal: true)
                 }
             )
         }
     }
     return nil
    

    } }

深度思考:Redux 架构下的单向流 在 Redux 模式下,我们的 NavigationStore 应该对这种层级关系有清晰的定义:

  • path 数组:管理的是同一个 NavigationStack 内的线性增减。
  • presentedSheet:管理的是一个全新的 UIWindow 级别的层级。 为什么我们要这么做? 这样做最大的好处是导航状态的可预测性。 当你在处理 Deep Linking(深度链接)时,你可以清晰地在代码里写:

“先 Push 到用户中心,再从用户中心 Present 一个修改头像的弹窗。”

如果每个页面都自带 NavigationStack,这种跨层级的逻辑跳转将会是调试噩梦。

总结与更新 我们要对现有的导航包进行以下约定:

  • 所有模块暴露的 Push 视图必须是“裸”的(无 Stack)。
  • 模块可以定义专有的 SheetRoute,由对应的 Factory 负责包裹 NavigationStack。
  • 统一使用 isModal 或 Environment 变量来处理导航栏交互的差异。 这样一来,我们的 ReduxRouterStack 就能保持极其精简,同时具备极强的健壮性。

【swift开发基础】33 | 访问和操作数组 - 遍历和索引

2026年1月16日 23:55

一、访问和操作数组

1. 数组遍历

1)for-in循环

  • 基本用法:通过for num in numbers形式遍历数组元素,是Swift中最基础的遍历方式
  • 控制流支持:完整支持break和continue控制语句,可以灵活控制循环流程

2)forEach方法

  • 语法特点:采用闭包形式numbers.forEach { num in ... }进行遍历

  • 控制限制:

    • break/continue:无法使用这两个控制语句跳出或跳过循环
    • return行为:仅退出当前闭包执行体,不会终止整个遍历过程
  • 示例说明:当尝试在num == 3时执行break会导致编译错误,改为return则只会跳过数字3的输出

3)同时得到索引和值

  • enumerated()方法:

    • 返回(index, value)元组序列
    • 示例:for (index, num) in numbers.enumerated()
  • 替代方案:可通过0..<numbers.count区间遍历索引,再通过下标访问值

  • 推荐实践:相比手动索引访问,更推荐使用enumerated()方法,代码更简洁清晰

4)使用Iterator遍历数组

  • 实现步骤:

    • 通过makeIterator()获取迭代器
    • 使用while let配合next()方法遍历
  • 终止条件:当next()返回nil时循环自动结束

  • 适用场景:适合需要自定义遍历逻辑的情况,但日常开发中使用频率较低

2. 索引

1)startIndex

  • 特性:始终返回0,表示数组第一个元素的位置
  • 空数组情况:当数组为空时,startIndex等于endIndex

2)endIndex

  • 定义:返回最后一个元素索引+1的位置
  • 等价关系:对于数组而言等同于count属性值
  • 特殊说明:与String不同,数组的索引都是Int类型

3)indices

  • 功能:返回数组的有效索引区间(Range)
  • 遍历应用:可通过for i in numbers.indices形式遍历所有有效索引
  • 优势:比手动指定0..<count更安全可靠

3. 代码示例

1)forEach方法应用

  • 基础输出:成功输出数组[2,3,4,5,6,7]所有元素

  • 控制尝试:

    • break/continue会导致编译错误
    • return仅跳过当前元素(如跳过数字3)

2)enumerated方法应用

  • 输出格式:同时打印索引和元素值(如"the index is: 0 2")
  • 数值处理:示例中将元素值乘以10后输出

3)使用iterator遍历数组

  • 迭代过程:通过while let num = it.next()持续获取下一个元素
  • 终止机制:当next()返回nil时自动结束循环

4)使用索引区间遍历

  • 实现方式:for i in numbers.indices配合下标访问
  • 输出效果:与enumerated()方法输出结果相同

4. 最佳实践建议

  • 简单遍历:仅需元素值时优先使用for-in循环
  • 索引需求:需要同时访问索引和值时推荐使用enumerated()方法
  • 性能考虑:避免在循环体内进行不必要的数组操作,保持遍历高效性

二、知识小结

知识点 核心内容 易混淆点/注意事项 代码示例
for-in循环 基础遍历方式,可配合break/continue控制流程 与forEach方法的关键区别在于流程控制 for number in numbers { ... }
forEach方法 闭包式遍历,语法简洁 不支持break/continue,return仅退出当前闭包 numbers.forEach { if$0 == 3 { return } }
enumerated() 同时获取索引(index)和值(value) 等效于for i in 0..<count但更优雅 for (index, num) in numbers.enumerated()
迭代器遍历 通过makeIterator()和while let组合实现 需手动处理迭代终止条件(nil) while let num = numbers.makeIterator().next()
索引属性 startIndex=0,endIndex=count 空数组时startIndex == endIndex numbers.indices返回索引区间
索引区间遍历 使用indices属性获取合法索引范围 与显式写0..<count效果相同 for i in numbers.indices { numbers[i] }
❌
❌