В чем разница между #import и #include в Objective-C?

Каковы различия между #import и #include в Objective-C, и есть ли моменты, когда вы должны использовать один над другим? Один устарел?

Я читал следующий учебник: http://www.otierney.net/objective-c.html и его параграф о #import и #include, кажется, противоречит сам себе или, по крайней мере, неясен.

9 ответов

Решение

Директива #import была добавлена ​​в Objective-C как улучшенная версия #include. Однако все же вопрос о том, улучшился он или нет, все еще остается предметом споров. #import гарантирует, что файл будет включен только один раз, чтобы у вас никогда не возникало проблем с рекурсивными включениями. Тем не менее, большинство приличных заголовочных файлов в любом случае защищают себя от этого, так что это не так уж и полезно.

По сути, вам решать, что вы хотите использовать. Я склонен #import заголовков для объектов Objective-C (таких как определения классов и тому подобное) и #include стандартные вещи C, которые мне нужны. Например, один из моих исходных файлов может выглядеть так:

#import <Foundation/Foundation.h>

#include <asl.h>
#include <mach/mach.h>

Кажется, есть много путаницы в отношении препроцессора.

Что делает компилятор, когда видит #include что он заменяет эту строку содержимым включенных файлов, без вопросов.

Так что если у вас есть файл a.h с этим содержанием:

typedef int my_number;

и файл b.c с этим содержанием:

#include "a.h"
#include "a.h"

файл b.c будет переведен препроцессором перед компиляцией в

typedef int my_number;
typedef int my_number;

что приведет к ошибке компилятора, так как тип my_number определяется дважды. Хотя определение одно и то же, это не допускается языком Си.

Поскольку заголовок часто используется более чем в одном месте, в C обычно используются защитные элементы. Это выглядит так:

 #ifndef _a_h_included_
 #define _a_h_included_

 typedef int my_number;

 #endif

Файл b.c по-прежнему будет содержать все содержимое заголовка дважды после предварительной обработки. Но второй экземпляр будет игнорироваться, так как макрос _a_h_included_ был бы уже определен.

Это работает очень хорошо, но имеет два недостатка. Прежде всего, должны быть включены защитные элементы, а имя макроса должно быть разным в каждом заголовке. И, во-вторых, компилятор все еще должен искать заголовочный файл и читать его так часто, как он включен.

Objective-C имеет #import инструкция препроцессора (она также может использоваться для кода C и C++ с некоторыми компиляторами и опциями). Это делает почти так же, как #include, но он также отмечает, какой файл уже был включен. #import строка заменяется только содержимым именованного файла при первом обнаружении. Каждый раз после этого это просто игнорируется.

Я согласен с Джейсоном.

Я был пойман, делая это:

#import <sys/time.h>  // to use gettimeofday() function
#import <time.h>      // to use time() function

Для GNU gcc он продолжал жаловаться, что функция time() не была определена.

Тогда я изменил #import на #include и все прошло нормально.

Причина:

Вы #import :
включает только часть с помощью #defines

Вы #import :
Нет идти. Хотя только часть уже была включена, как
Что касается #import, этот файл уже полностью включен.

Нижняя линия:

Заголовки C/C++ традиционно включают в себя части других включаемых файлов.
Так что для заголовков C/C++ используйте #include.
Для заголовков objc/objC++ используйте #import.

#include работает так же, как C #include,

#import отслеживает, какие заголовки уже включены, и игнорируется, если заголовок импортирован более одного раза в блок компиляции. Это делает ненужным использование охранников заголовка.

Нижняя строка просто использовать #import в Objective-C и не беспокойтесь, если ваши заголовки импортируют что-то более одного раза.

Я знаю, что эта ветка старая... но в "современные времена".. есть намного превосходящая "стратегия включения" через Clang's @import модули - это часто упускается из виду..

Модули улучшают доступ к API библиотек программного обеспечения, заменяя модель включения текстового препроцессора более надежной, более эффективной семантической моделью. С точки зрения пользователя, код выглядит немного иначе, поскольку используется декларация импорта, а не директива препроцессора #include:

@import Darwin; // Like including all of /usr/include. @see /usr/include/module.map

или же

@import Foundation;  //  Like #import <Foundation/Foundation.h>
@import ObjectiveC;  //  Like #import <objc/runtime.h>

Однако этот импорт модуля ведет себя совершенно иначе, чем соответствующий #include: когда компилятор видит импорт модуля выше, он загружает двоичное представление модуля и делает его API доступным для приложения напрямую. Определения препроцессора, предшествующие объявлению импорта, не влияют на предоставляемый API... потому что сам модуль был скомпилирован как отдельный, автономный модуль. Кроме того, любые флаги компоновщика, необходимые для использования модуля, будут автоматически предоставлены при импорте модуля. Эта семантическая модель импорта решает многие проблемы модели включения препроцессора.

Чтобы включить модули, передайте флаг командной строки -fmodules ака CLANG_ENABLE_MODULES в Xcode - во время компиляции. Как упомянуто выше.. эта стратегия устраняет ЛЮБОЕ и ВСЕ LDFLAGS, Например, вы можете УДАЛИТЬ любые настройки "OTHER_LDFLAGS", а также любые фазы "Связывания".

Я нахожу, что время компиляции / запуска "чувствует" гораздо более быстрое (или, возможно, "связывание" задерживается меньше), а также предоставляет отличную возможность удалить теперь посторонний файл Project-Prefix.pch, и соответствующие настройки сборки, GCC_INCREASE_PRECOMPILED_HEADER_SHARING, GCC_PRECOMPILE_PREFIX_HEADER, а также GCC_PREFIX_HEADER, так далее.

Кроме того, хотя не очень хорошо документированы... Вы можете создать module.map s для ваших собственных фреймворков и включайте их таким же удобным способом. Вы можете взглянуть на мой репозиторий ObjC-Clang-Modules github для некоторых примеров того, как реализовать такие чудеса.

Если вы знакомы с C++ и макросами, то

#import "Class.h" 

похож на

{
#pragma once

#include "class.h"
}

Это означает, что ваш класс будет загружен только один раз при запуске приложения.

#include он использовал, чтобы получить "вещи" из другого файла в тот, #include используется в. Например:

в файле: main.cpp

#include "otherfile.h"

// some stuff here using otherfile.h objects,
// functions or classes declared inside

Защита заголовка используется в верхней части каждого заголовочного файла (*.h), чтобы предотвратить включение одного и того же файла более одного раза (если это произойдет, вы получите ошибки компиляции).

в файле: otherfile.h

#ifndef OTHERFILE
#define OTHERFILE

// declare functions, classes or objects here

#endif

даже если вы положите #include "otherfile.h" n раз в вашем коде, это внутри него не будет повторно объявлено.

В одном случае у меня была глобальная переменная в одном из моих .h файлы, которые вызывали проблему, и я решил ее, добавив extern перед ней.

Если вы #include файл два раза в.h файлы, чем компилятор выдаст ошибку. Но если вы #import файла более одного раза, компилятор будет игнорировать его.

#include + guard == #import

#include guard Wiki - защита от макросов, защита заголовка или защита файла предотвращает двойное включение заголовкаpreprocessor что может замедлить время сборки

Следующий шаг

.pch [О себе] =>@import [Около]

[#import in .h или .m]

Другие вопросы по тегам