Namespaces
Variants

std:: uninitialized_copy_n

From cppreference.net
Memory management library
( exposition only* )
Allocators
Uninitialized memory algorithms
Constrained uninitialized memory algorithms
Memory resources
Uninitialized storage (until C++20)
( until C++20* )
( until C++20* )
( until C++20* )

Garbage collector support (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
Definido en el encabezado <memory>
template < class InputIt, class Size, class NoThrowForwardIt >

NoThrowForwardIt uninitialized_copy_n ( InputIt first, Size count,

NoThrowForwardIt d_first ) ;
(1) (desde C++11)
(constexpr desde C++26)
template < class ExecutionPolicy, class ForwardIt,

class Size, class NoThrowForwardIt >
NoThrowForwardIt uninitialized_copy_n ( ExecutionPolicy && policy,
ForwardIt first, Size count,

NoThrowForwardIt d_first ) ;
(2) (desde C++17)
1) Copia count elementos desde un rango que comienza en first a un área de memoria no inicializada que comienza en d_first como si fuera

for ( ; count > 0 ; ++ d_first, ( void ) ++ first, -- count )
:: new ( voidify ( * d_first ) )
typename std:: iterator_traits < NoThrowForwardIt > :: value_type ( * first ) ;

Si se lanza una excepción durante la inicialización, los objetos ya construidos se destruyen en un orden no especificado.
2) Igual que (1) , pero ejecutado de acuerdo con la policy .
Esta sobrecarga participa en la resolución de sobrecarga solo si se cumplen todas las siguientes condiciones:

std:: is_execution_policy_v < std:: decay_t < ExecutionPolicy >> es true .

(hasta C++20)

std:: is_execution_policy_v < std:: remove_cvref_t < ExecutionPolicy >> es true .

(desde C++20)


Si d_first + [ 0 , count ) se superpone con first + [ 0 , count ) , el comportamiento es indefinido.

(desde C++20)

Contenidos

Parámetros

first - el inicio del rango de los elementos a copiar
count - el número de elementos a copiar
d_first - el inicio del rango de destino
policy - la política de ejecución a utilizar
Requisitos de tipo
-
InputIt debe cumplir con los requisitos de LegacyInputIterator .
-
ForwardIt debe cumplir con los requisitos de LegacyForwardIterator .
-
NoThrowForwardIt debe cumplir con los requisitos de LegacyForwardIterator .
-
Ningún incremento, asignación, comparación o indirección a través de instancias válidas de NoThrowForwardIt puede lanzar excepciones.

Valor de retorno

Iterador al elemento después del último elemento copiado.

Complejidad

Lineal en count .

Excepciones

La sobrecarga con un parámetro de plantilla llamado ExecutionPolicy reporta errores de la siguiente manera:

  • Si la ejecución de una función invocada como parte del algoritmo lanza una excepción y ExecutionPolicy es una de las políticas estándar , std::terminate es llamado. Para cualquier otra ExecutionPolicy , el comportamiento está definido por la implementación.
  • Si el algoritmo no puede asignar memoria, std::bad_alloc es lanzado.

Notas

Macro de prueba de características Valor Estándar Característica
__cpp_lib_raw_memory_algorithms 202411L (C++26) constexpr para algoritmos de memoria especializados , ( 1 )

Implementación posible

template<class InputIt, class Size, class NoThrowForwardIt>
constexpr NoThrowForwardIt uninitialized_copy_n(InputIt first, Size count,
                                                NoThrowForwardIt d_first)
{
    using T = typename std::iterator_traits<NoThrowForwardIt>::value_type;
    NoThrowForwardIt current = d_first;
    try
    {
        for (; count > 0; ++first, (void) ++current, --count)
            ::new (static_cast<void*>(std::addressof(*current))) T(*first);
    }
    catch (...)
    {
        for (; d_first != current; ++d_first)
            d_first->~T();
        throw;
    }
    return current;
}

Ejemplo

#include <algorithm>
#include <iostream>
#include <memory>
#include <string>
#include <tuple>
#include <vector>
int main()
{
    std::vector<std::string> v = {"This", "is", "an", "example"};
    std::string* p;
    std::size_t sz;
    std::tie(p, sz) = std::get_temporary_buffer<std::string>(v.size());
    sz = std::min(sz, v.size());
    std::uninitialized_copy_n(v.begin(), sz, p);
    for (std::string* i = p; i != p + sz; ++i)
    {
        std::cout << *i << ' ';
        i->~basic_string<char>();
    }
    std::cout << '\n';
    std::return_temporary_buffer(p);
}

Salida posible:

This is an example

Informes de defectos

Los siguientes informes de defectos que modifican el comportamiento se aplicaron retroactivamente a los estándares de C++ publicados anteriormente.

DR Aplicado a Comportamiento publicado Comportamiento correcto
LWG 2133 C++98 la descripción del efecto utilizaba un for loop con la expresión de iteración
++ d_first, ++ first, -- count , lo cual
resulta en búsquedas dependientes de argumento de operator,
descarta el valor
de un operando para
deshabilitar esas ADLs
LWG 2433 C++11 este algoritmo podría ser secuestrado por operator & sobrecargado utiliza std::addressof
LWG 3870 C++20 este algoritmo podría crear objetos en un almacenamiento const se mantiene deshabilitado

Véase también

copia un rango de objetos a un área de memoria no inicializada
(plantilla de función)
copia un número de objetos a un área de memoria no inicializada
(objeto función de algoritmo)