AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 77934697
Accepted
yushizhao
yushizhao
Asked: 2024-02-04 13:29:09 +0800 CST2024-02-04 13:29:09 +0800 CST 2024-02-04 13:29:09 +0800 CST

Box::new() fará uma cópia da pilha para a pilha?

  • 772

Do documento diz:

pub fn new(x: T) -> Box Aloca memória no heap e depois coloca x nele.

Mas “lugar” é uma palavra complicada. Se escrevermos

let arr_boxed = Box::new([0;1000]);

Será [0;1000]inicializado no heap no local?

Se escrevermos

let arr = [0;1000];
let arr_boxed = Box::new(arr);

O compilador será inteligente o suficiente para inicializar o [0;1000]heap em primeiro lugar?

rust
  • 1 1 respostas
  • 606 Views

1 respostas

  • Voted
  1. Best Answer
    ChrisB
    2024-02-04T14:37:46+08:002024-02-04T14:37:46+08:00

    Box::new() fará uma cópia da pilha para a pilha?

    Às vezes. A linguagem Rust não garante que essa otimização aconteça e parece deixar que o LLVM descubra isso. Por causa disso, não importa se você inicializa o array primeiro e depois o passa, pois isso é essencialmente a mesma coisa para o back-end.

    Na prática, o desempenho dependerá do caso. O exemplo que você deu é realmente especial, porque os dados são todos zeros:

    pub fn foo() -> Box<[i32; 1000]> {
        return Box::new([0; 1000]);
    }
    

    Nos meus testes , o compilador conseguiu transformar isso em uma alocação + uma chamada para memsetos dados do heap.

    Nota : apenas com as otimizações ativadas. No modo de depuração, ele será copiado.


    Por outro lado, você pode querer inicializar seus dados com um valor conhecido:

    pub fn bar(v: i32) -> Box<[i32; 1000]> {
        return Box::new([v; 1000]);
    }
    

    Para meu horror , o compilador decide inicializar todos os dados na pilha e depois chamar memcpy. (Pelo menos desenrolou o loop de preenchimento) :). Isso acontece mesmo para dados realmente grandes, como [v; 100000], que travarão seu programa com um estouro de pilha. Usar um literal conhecido em tempo de compilação (diferente de zero) [64; 100000]se comporta da mesma maneira.


    Se você realmente quiser ter certeza, você pode fazer algo assim:

    pub fn baz(v: i32) -> Box<[i32; 1000]>{
        unsafe {
            let b = std::alloc::alloc(
                std::alloc::Layout::array::<i32>(1000).unwrap_unchecked()
            ) as *mut i32;
            for i in 0..1000 {
                *b.add(i) = v;
            }
            Box::from_raw(b as *mut [i32; 1000])
        }
    }
    

    que faz a coisa certa .


    Uma versão segura bazseria:

    use std::convert::TryInto;
    
    pub fn quux(v: i32) -> Box<[i32; 1000]> {
        let mut b = Vec::with_capacity(1000);
        b.extend(std::iter::repeat(v).take(1000));
        b.into_boxed_slice().try_into().unwrap()
    }
    

    Que o compilador otimiza muito bem , para essencialmente a mesma montagem que o baz.


    Ainda mais curto seria

    vec![v; 1000].into_boxed_slice().try_into::<Box<[i32; 1000]>>().unwrap()

    qual é provavelmente a melhor versão.

    • 17

relate perguntas

  • os braços de correspondência têm tipos incompatíveis esperados ao reutilizar a função dentro da correspondência

  • Conversão de tipo de ferrugem em uma instrução de correspondência

  • Como forçar o tipo de retorno de uma correspondência para ()?

  • enums de ferrugem em representações primitivas

  • Existe uma maneira de simplificar a correspondência diretamente para Ok("VAL") em Result<String, VarError>

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle?

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Quando devo usar um std::inplace_vector em vez de um std::vector?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Marko Smith

    Estou tentando fazer o jogo pacman usando apenas o módulo Turtle Random e Math

    • 1 respostas
  • Martin Hope
    Aleksandr Dubinsky Por que a correspondência de padrões com o switch no InetAddress falha com 'não cobre todos os valores de entrada possíveis'? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer Quando devo usar um std::inplace_vector em vez de um std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB Por que o GCC gera código que executa condicionalmente uma implementação SIMD? 2024-02-17 06:17:14 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve