
在数据处理中,我们经常需要对 DataFrame 中的每一行数据应用一组特定的除数。例如,将一个多行 DataFrame 的每一行按列与一个单行 DataFrame 中的对应值进行除法运算。尽管在 Pandas 等库中,这可能通过 df.divide() 等方法直接实现,但在 Polars 中,由于其独特的设计哲学和性能优化策略,直接对齐不同高度的 DataFrame 进行逐元素运算需要特定的方法。
一个常见的误区是尝试通过复制单行 DataFrame 来使其与主 DataFrame 的行数匹配,从而进行直接的逐元素运算。
最初,开发者可能会尝试将单行除数 DataFrame 扩展成与被除数 DataFrame 相同大小,以实现逐元素除法。这种方法通常涉及使用 itertools.repeat 和 pl.concat 来创建重复的单行 DataFrame。
考虑以下示例:
from itertools import repeat
import polars as pl
# 待除的 DataFrame
data = {'a': [i for i in range(1, 5)],
'b': [i for i in range(1, 5)],
'c': [i for i in range(1, 5)],
'd': [i for i in range(1, 5)]}
df = pl.DataFrame(data)
# 除数 DataFrame (单行)
divisors = pl.DataFrame({'d1': 1, 'd2': 10, 'd3': 100, 'd4': 1000})
print("原始 DataFrame (df):")
print(df)
print("\n除数 DataFrame (divisors):")
print(divisors)
# 低效方法:重复除数 DataFrame
divisors_as_big_as_df = pl.concat([item for item in repeat(divisors, len(df))])
print("\n重复后的除数 DataFrame (divisors_as_big_as_df):")
print(divisors_as_big_as_df)
# 执行除法
divided_df = df / divisors_as_big_as_df
print("\n除法结果 (divided_df):")
print(divided_df)输出示例:
原始 DataFrame (df): shape: (4, 4) ┌─────┬─────┬─────┬─────┐ │ a ┆ b ┆ c ┆ d │ │ --- ┆ --- ┆ --- ┆ --- │ │ i64 ┆ i64 ┆ i64 ┆ i64 │ ╞═════╪═════╪═════╪═════╡ │ 1 ┆ 1 ┆ 1 ┆ 1 │ │ 2 ┆ 2 ┆ 2 ┆ 2 │ │ 3 ┆ 3 ┆ 3 ┆ 3 │ │ 4 ┆ 4 ┆ 4 ┆ 4 │ └─────┴─────┴─────┴─────┘ 除数 DataFrame (divisors): shape: (1, 4) ┌─────┬─────┬─────┬──────┐ │ d1 ┆ d2 ┆ d3 ┆ d4 │ │ --- ┆ --- ┆ --- ┆ --- │ │ i64 ┆ i64 ┆ i64 ┆ i64 │ ╞═════╪═════╪═════╪══════╡ │ 1 ┆ 10 ┆ 100 ┆ 1000 │ └─────┴─────┴──────┴──────┘ 重复后的除数 DataFrame (divisors_as_big_as_df): shape: (4, 4) ┌─────┬─────┬─────┬──────┐ │ d1 ┆ d2 ┆ d3 ┆ d4 │ │ --- ┆ --- ┆ --- ┆ --- │ │ i64 ┆ i64 ┆ i64 ┆ i64 │ ╞═════╪═════╪═════╪══════╡ │ 1 ┆ 10 ┆ 100 ┆ 1000 │ │ 1 ┆ 10 ┆ 100 ┆ 1000 │ │ 1 ┆ 10 ┆ 100 ┆ 1000 │ │ 1 ┆ 10 ┆ 100 ┆ 1000 │ └─────┴─────┴──────┴──────┘ 除法结果 (divided_df): shape: (4, 4) ┌─────┬─────┬──────┬───────┐ │ a ┆ b ┆ c ┆ d │ │ --- ┆ --- ┆ --- ┆ --- │ │ f64 ┆ f64 ┆ f64 ┆ f64 │ ╞═════╪═════╪══════╪═══════╡ │ 1.0 ┆ 0.1 ┆ 0.01 ┆ 0.001 │ │ 2.0 ┆ 0.2 ┆ 0.02 ┆ 0.002 │ │ 3.0 ┆ 0.3 ┆ 0.03 ┆ 0.003 │ │ 4.0 ┆ 0.4 ┆ 0.04 ┆ 0.004 │ └─────┴─────┴──────┴───────┘
尽管上述方法能够得到正确的结果,但其效率低下。当主 DataFrame 包含大量行时,重复单行 DataFrame 会导致创建非常大的中间 DataFrame,这不仅消耗大量内存,还会增加不必要的计算时间。
Polars 的设计理念是围绕高性能的列式操作。对于将 DataFrame 的每一列除以一个标量或单行 DataFrame 中对应的单个值,最佳实践是利用 with_columns 方法结合列选择器 pl.col()。这种方法避免了创建大型中间 DataFrame,而是直接对每列进行操作,效率显著提高。
核心思想是遍历主 DataFrame 的所有列,然后将每一列除以 divisors DataFrame 中对应列的第一个(也是唯一一个)元素。
import polars as pl
# 待除的 DataFrame
data = {'a': [i for i in range(1, 5)],
'b': [i for i in range(1, 5)],
'c': [i for i in range(1, 5)],
'd': [i for i in range(1, 5)]}
df = pl.DataFrame(data)
# 除数 DataFrame (单行)
divisors = pl.DataFrame({'d1': 1, 'd2': 10, 'd3': 100, 'd4': 1000})
print("原始 DataFrame (df):")
print(df)
print("\n除数 DataFrame (divisors):")
print(divisors)
# 推荐方法:利用 with_columns 进行列式除法
# 构建一个字典,键为原始列名,值为对应的除法表达式
# pl.col(col) 表示原始 DataFrame 中的当前列
# divisors[f"d{i+1}"] 表示除数 DataFrame 中对应的列(此处直接取其Series)
divided_df_optimized = df.with_columns(
**{col: pl.col(col) / divisors[f"d{i+1}"] for (i, col) in enumerate(df.columns)}
)
print("\n优化后的除法结果 (divided_df_optimized):")
print(divided_df_optimized)输出示例:
原始 DataFrame (df): shape: (4, 4) ┌─────┬─────┬─────┬─────┐ │ a ┆ b ┆ c ┆ d │ │ --- ┆ --- ┆ --- ┆ --- │ │ i64 ┆ i64 ┆ i64 ┆ i64 │ ╞═════╪═════╪═════╪═════╡ │ 1 ┆ 1 ┆ 1 ┆ 1 │ │ 2 ┆ 2 ┆ 2 ┆ 2 │ │ 3 ┆ 3 ┆ 3 ┆ 3 │ │ 4 ┆ 4 ┆ 4 ┆ 4 │ └─────┴─────┴─────┴─────┘ 除数 DataFrame (divisors): shape: (1, 4) ┌─────┬─────┬─────┬──────┐ │ d1 ┆ d2 ┆ d3 ┆ d4 │ │ --- ┆ --- ┆ --- ┆ --- │ │ i64 ┆ i64 ┆ i64 ┆ i64 │ ╞═════╪═════╪═════╪══════╡ │ 1 ┆ 10 ┆ 100 ┆ 1000 │ └─────┴─────┴──────┴──────┘ 优化后的除法结果 (divided_df_optimized): shape: (4, 4) ┌─────┬─────┬──────┬───────┐ │ a ┆ b ┆ c ┆ d │ │ --- ┆ --- ┆ --- ┆ --- │ │ f64 ┆ f64 ┆ f64 ┆ f64 │ ╞═════╪═════╪══════╪═══════╡ │ 1.0 ┆ 0.1 ┆ 0.01 ┆ 0.001 │ │ 2.0 ┆ 0.2 ┆ 0.02 ┆ 0.002 │ │ 3.0 ┆ 0.3 ┆ 0.03 ┆ 0.003 │ │ 4.0 ┆ 0.4 ┆ 0.04 ┆ 0.004 │ └─────┴─────┴──────┴───────┘
解释:
推荐的列式操作方法在性能上远优于通过 pl.concat 构造大型中间 DataFrame 的方法,主要原因有:
注意事项:
在 Polars 中,当需要将一个 DataFrame 的每一行按列除以一个单行 DataFrame 的对应值时,最推荐且最高效的方法是利用 df.with_columns() 结合列式操作。通过构建一个字典推导式,将主 DataFrame 的每一列与除数 DataFrame 中对应的单个值进行除法运算,可以充分利用 Polars 的性能优势,避免不必要的内存开销和计算负担。这种模式是 Polars 处理类似数据转换任务的通用且高效的范例。
以上就是Polars 中高效实现 DataFrame 逐行除法的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号