К. Арнольд, Д. Гослинг - Язык программирования Java (1160779), страница 26
Текст из файла (страница 26)
Затем - получаем объект Class для заданного типа. ФункцияFindClass получает в качестве параметров среду выполнения, имя типа в виде строки языка C и логическое значение, которое определяетнеобходимость загрузки класса в том случае, если он не был ранее загружен. Функция EE возвращает текущее состояние среды выполнения.Объект, возвращаемый функцией FindClass, вставляется после конца массива и используется runtime-системой для проверки того, что каждыйзаносимый в массив элемент относится к правильному типу.
Чтобы создать массив, который может содержать любые объекты, следуетвоспользоваться классом "java/lang/Object".Реализация LocalString.sort показывает, как работает вся эта инфраструктура. Сначала давайте посмотрим, как реализована сама функцияlocal_LocalString_sort:#include "local_LocalString.h"#include <<stdlib.h>>#include ,javaString.h>>HArrayOfString *local_LocalString_sort(struct Hlocal_LocalString *this_h,HArrayOfString *strngs_h){ClassArrayOfString *in_strings;HArrayOfString *retval = NULL;ClassArrayOfString *retstrs;char **string = NULL;int i, str_cnt;if (strings_h ==NULL) { /* проверить ссылку */SignalError(EE(),"java/lang/NullPointerException", "null array");return NULL;}set_locale();in_strings = unhand(strings_h);str_cnt = obj_length(strings_h);strings = (char **)malloc(str_cnt * sizeof *strings);if (strings == NULL) {SignalError(EE(),"java/lang/OutOfMemorytException", NULL);return NULL;}for (i = 0; i << str_cnt; i++) {if (in_strings->>body[i] == NULL) {SignalError(EE(),"java/lang/NullPointerException","Null string in array");goto cleanup;}strings[i] = makeCString(in_strings->>body[i]);if (strings[i] == NULL)goto cleanup; /* функция SignalError() уже вызвана */}qsort(strings, str_cnt, sizeof *strings, cmp);retval = (HArrayOfString *)alloc_class_array("java/lang/String", str_cnt);retstrs = unhand(retval);for (i = 0; i << str_cnt; i++) {retstrs->>body[i] =makeJavaString(strings[i], strlen(strings[i]));}cleanup:free(strings);return retval;}Сначала мы проверяем, действительно ли был передан массив.
Затем устанавливается локальный контекст, как это делалось в LocalString.xfrm.Затем — создаем строковый массив, в котором будет храниться содержимое сортируемых объектов String. Для создания строкового массиванеобходимо знать, сколько строк в него входит — это число получается вызовом obj_length для дескриптора строкового массива.
Затем мыиспользуем функцию malloc для выделения памяти под указатели и проверяем возвращаемое ею значение.Проверка ошибок чрезвычайно важна. Родные методы, не анализирующие возможные ошибки, нарушают те гарантии безопасности, которыеJava предоставляет программистам. Например, если бы мы пропустили проверку равенства ссылки null, то при попытке использования этогоуказателя вместо возбуждения исключения NullPointerException все бы кончилось крахом программного потока, а возможно — и всегоприложения.Получив место для хранения строк языка C, мы в цикле перебираем элементы входного массива и сохраняем копии исходных строк в массивесортировки.
При этом мы не забываем проверять наличие null-ссылок среди этих элементов. В нашей функции была использована функцияmakeCString — главным образом для того, чтобы показать, как ей пользоваться. Кроме того, это упрощает программу, поскольку сборщик мусорабудет сам уничтожать все возвращаемые строки, в том числе и при возникновении ошибок.Теперь массив strings содержит эквиваленты исходных строк в языке C.
Мы вызываем стандартную библиотечную функцию C с именем qsort,чтобы отсортировать массив, и передаем ей в качестве последнего аргумента функцию (в данном случае — cmp):static intcmp(const void *str1, const void *str2){return strcoll(*(char **)str1, *(char **)str2);}Функция сравнения cmp преобразует свои параметры-указатели к типу char ** (qsort требует, чтобы параметры имели тип void *; предполагается,что программист сам произведет все необходимые приведения типов).
Затем вызывается стандартная библиотечная функция C с именем strcoll,которая сравнивает две строки с учетом локального контекста. Эта функция возвращает отрицательное, равное нулю или положительное число,если первая строка соответственно меньше, равна или больше второй в локальном языковом контексте упорядочения строк.
То же самоефункция qsort ожидает от своей функции сравнения, так что значение, возвращаемое strcoll, может возвращаться и самой функцией cmp.После выполнения qsort, строки оказываются отсортированными в соответствии с функцией strcoll.
Все, что остается — построить новый массивобъектов String и заполнить его результатами. Далее, строится массив с помощью рассмотренной выше функции alloc_class_array и затем в циклевызывается makeJavaString для задания каждого объекта String. Наконец, мы освобождаем массив strings, созданный функцией malloc, ивозвращаем результат.Упражнение А.6Модифицируйте класс LocalString, чтобы он мог работать с объектами, каждый из которых обладает собственным локальным контекстом, вместотого, чтобы полагаться на один общий контекст. Методы перестанут быть статическими, и понадобится новое строковое поле, определяющееконтекст. Помните, что функция POSIX с именем setlocale устанавливает локальный контекст лишь до следующего вызова setlocale.А.6 Создание объектовВы можете создавать объекты Java внутри реализаций родных методов с помощью функции execute_java_constructor:HObject *execute_java_constructor(ExecEnv *ee, char *className, ClassClass *classObj, char *signature, ...)Создает новый объект указанного типа, задаваемого одним из двух параметров className или ClassObj (не используемыйпараметр должен быть равен NULL).
Для создания объекта вызывается конструктор, описываемый строкой signature. Запараметром signature следуют параметры конструктора.Например, создание нового объекта типа Simple с помощью безаргументного конструктора класса происходит следующим образом:execute_java_contructor(NULL, "Simple", NULL, "()")В данном случае сигнатура конструктора выглядит тривиально. Чтобы воспользоваться конструктором, который получает один или несколькопараметров, необходимо включить их типы в сигнатуру, и поместить их значения в нужном порядке после строки сигнатуры.
Типы в строкесигнатуры аналогичны тем, которые возвращаются Class.getName, и используют односимвольные сокращения для примитивных типов.Применяются следующие сокращения:Z booleanI intC charJ longB byteF floatS shortD doubleЧтобы избежать конфликтов между именами классов/интерфейсов и этими буквами, типы объектов получают имена вида " Ltype" , где type —полное имя класса или интерфейса, в котором разделители-точки заменяются косой чертой, а в конце ставится точка с запятой. Например,параметр, представляющий собой объект String, будет выглядеть как " Ljava/lang/String;" . Для массивов указывается тип массива с префиксом [;так, массив значений типа long будет иметь тип " [J" .
В многомерных типах используются несколько квадратных скобок. Массив String[][] будетвыглядеть как " [[Ljava/labg/String;" .Если вам все же непонятно, как указать конкретный тип в такой строке, вы можете написать класс Java, создающий объект нужного типа, ивызвать для него метод getClass().getName().
Затем, если полученное имя является именем типа, замените все точки на /, поставьте L спереди и ;сзади, если эти символы отсутствуют.Аргументы, которые представляют собой объекты, передаются в виде указателей на дескрипторы.Ниже показано, как происходит создание двух объектов Attr; первый из них использует конструктор с одним аргументом, которому передаетсятолько имя, а второй — конструктор с двумя аргументами, которому сообщается исходное значение.oneArgAttr = (struct HAttr *)execute_java_constructor(EE(), "Attr", NULL,"(Ljava/lang/String;)", attrStr);twoArgAttr = (struct HAttr *)execute_java_constructor(EE(), "Attr", NULL,"(Ljava/lang/String;Ljava/lang/Object;)",attrStr, attrStr);Точка с запятой выполняет функцию терминатора (завершающего символа) типа, а не разделителя параметров.
Конструктор, получающий двапараметра типа long и два параметра типа double, описывается строкой " (JJDD)" .А.7 Вызов методов JavaВызов методов Java из программ на C напоминает вызов конструкторов Java. Для этого используются следующие основные функции:long *execute_java_static_method(ExecEnv *ee, ClassClass *cb, char *method_name, char *signature, ...)Выполняет статический метод класса, описываемого параметром cb.long *execute_java_dynamic_method(ExecEnv *ee, HObject *obj, char *method_name, char *signature, ...)Выполняет нестатический (динамический) метод для заданного объекта.В обоих функциях, параметр method_name является именем вызываемого метода, а signature описывает передаваемые аргументы.