Поделиться через


Предупреждение компилятора (уровень 3) C4996

"функция": объявлена устаревшей также "std::<имя функции>": вызов функции с параметрами, которые могут быть небезопасными — правильность передаваемых значений для этого вызова зависит от вызывающего объекта.Чтобы отключить это предупреждение, используйте -D_SCL_SECURE_NO_WARNINGS.См. документацию по использованию "проверяемых итераторов" Visual C++.

Это предупреждение или ошибка имеет много возможных значений.

Ошибка C4996 может возникнуть, если компилятор обнаружил функцию, помеченную как устаревшая. Функция может не поддерживаться в будущих выпусках. Это предупреждение можно отключить с помощью прагма-директивы warning (см. пример ниже).

Некоторые функции библиотеки CRT и стандартной библиотеки C++ устарели, и им следует предпочесть новые, более безопасные функции. Дополнительные сведения о заменяющей функции см. в документации для устаревшей функции, указанной в сообщении об ошибке. Чтобы отключить предупреждения об устаревших функциях CRT, определите _CRT_SECURE_NO_WARNINGS. Дополнительные сведения об устаревших функциях см. в разделах Функции безопасности в CRT и Безопасные библиотеки: стандартная библиотека C++.

Предупреждение C4996 можно увидеть при использовании членов файлов заголовков <hash_map> и <hash_set> в пространстве имен std. Дополнительные сведения см. в разделе Пространство имен stdext.

Предупреждение C4996 также может возникать при использовании функций библиотек MFC или ATL, не рекомендуемых к использованию по соображениям безопасности. Дополнительные сведения о том, как подавить эти предупреждения, см. в разделах _AFX_SECURE_NO_WARNINGS и _ATL_SECURE_NO_WARNINGS.

Предупреждение C4996 также может возникать и при использовании библиотеки маршалинга. В этом случае C4996 будет ошибкой, а не предупреждением. Эта ошибка будет возникать при использовании функции marshal_as для выполнения такого преобразования одного типа данных в другой, которое на самом деле выполняется с помощью Класс marshal_context. Также эта ошибка будет возникать в случае, когда библиотека маршалинга не поддерживает преобразование. Дополнительные сведения о библиотеке маршалинга см. в разделе Общие сведения о маршалировании в C++.

В первом примере предупреждение C4996 возникает в строке, в которой объявляется функция, и в строке, в которой эта функция используется.

Пример

Следующий пример приводит к возникновению предупреждения C4996.

// C4996.cpp
// compile with: /W3
// C4996 warning expected
#include <stdio.h>

// #pragma warning(disable : 4996)
void func1(void) {
   printf_s("\nIn func1");
}

__declspec(deprecated) void func1(int) {
   printf_s("\nIn func2");
}

int main() {
   func1();
   func1(1);
}

Предупреждение C4996 также может возникнуть, если при компиляции с определенным параметром _ITERATOR_DEBUG_LEVEL (по умолчанию для сборок в режиме отладки устанавливается значение 1) не используются проверяемые итераторы. Дополнительные сведения см. в разделе Проверяемые итераторы.

Следующий пример кода STL вызывает появление ошибки C4996.

// C4996_b.cpp
// compile with: /EHsc /W3 /c
#define _ITERATOR_DEBUG_LEVEL 1

#include <algorithm>
#include <iterator>

using namespace std;
using namespace stdext;

int main() {
    int a[] = { 1, 2, 3 };
    int b[] = { 10, 11, 12 };
    copy(a, a + 3, b + 1);   // C4996
    // try the following line instead
    //   copy(a, a + 3, b);
    copy(a, a + 3, checked_array_iterator<int *>(b, 3));   // OK
}

Следующий пример кода STL вызывает появление предупреждения C4996. Комментарии приведены в коде.

#include <algorithm>
#include <array>
#include <iostream>
#include <iterator>
#include <numeric>
#include <string>
#include <vector>

using namespace std;

template <typename C> void print(const string& s, const C& c) {
    cout << s;

    for (const auto& e : c) {
        cout << e << " ";
    }

    cout << endl;
}

int main()
{
    vector<int> v(16);
    iota(v.begin(), v.end(), 0);
    print("v: ", v);

    // OK: vector::iterator is checked in debug mode
    // (i.e. an overrun will trigger a debug assertion)
    vector<int> v2(16);
    transform(v.begin(), v.end(), v2.begin(), [](int n) { return n * 2; });
    print("v2: ", v2);

    // OK: back_insert_iterator is marked as checked in debug mode
    // (i.e. an overrun is impossible)
    vector<int> v3;
    transform(v.begin(), v.end(), back_inserter(v3), [](int n) { return n * 3; });
    print("v3: ", v3);

    // OK: array::iterator is checked in debug mode
    // (i.e. an overrun will trigger a debug assertion)
    array<int, 16> a4;
    transform(v.begin(), v.end(), a4.begin(), [](int n) { return n * 4; });
    print("a4: ", a4);

    // OK: Raw arrays are checked in debug mode
    // (i.e. an overrun will trigger a debug assertion)
    // NOTE: This applies only when raw arrays are given to STL algorithms!
    int a5[16];
    transform(v.begin(), v.end(), a5, [](int n) { return n * 5; });
    print("a5: ", a5);

    // WARNING C4996: Pointers cannot be checked in debug mode
    // (i.e. an overrun will trigger undefined behavior)
    int a6[16];
    int * p6 = a6;
    transform(v.begin(), v.end(), p6, [](int n) { return n * 6; });
    print("a6: ", a6);

    // OK: stdext::checked_array_iterator is checked in debug mode
    // (i.e. an overrun will trigger a debug assertion)
    int a7[16];
    int * p7 = a7;
    transform(v.begin(), v.end(), stdext::make_checked_array_iterator(p7, 16), [](int n) { return n * 7; });
    print("a7: ", a7);

    // WARNING SILENCED: stdext::unchecked_array_iterator is marked as checked in debug mode
    // (i.e. it performs no checking, so an overrun will trigger undefined behavior)
    int a8[16];
    int * p8 = a8;
    transform(v.begin(), v.end(), stdext::make_unchecked_array_iterator(p8), [](int n) { return n * 8; });
    print("a8: ", a8);
}

Следующий пример приводит к возникновению предупреждения C4996, поскольку библиотеке маршалинга для преобразования из System::String в const char * требуется контекст.

// C4996_Marshal.cpp
// compile with: /clr 
// C4996 expected
#include <stdlib.h>
#include <string.h>
#include <msclr\marshal.h>

using namespace System;
using namespace msclr::interop;

int main() {
   String^ message = gcnew String("Test String to Marshal");
   const char* result;
   result = marshal_as<const char*>( message );
   return 0;
}