Форум программистов, компьютерный форум CyberForum.ru

Массив указателей - C++

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 17, средняя оценка - 4.88
Golovastik
 Аватар для Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
09.09.2009, 18:00     Массив указателей #1
Вот,продолжаю изучать тему массивы указателей. В качестве примера, почти уже на последних страницах приводится задачка с использованием указателей в 2-мерных массивах. Набрал программу, вижу что программа не выводит русские слова по запросу,решил её изменить. Вот как получилось, программу,только начинаю разбирать,что к чему, пока не до конца въезжаю в неё. Программа вроде компилируется но работает не правильно. Не скажите, где не правильно написанно.
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <iostream>
#include  <windows.h>
#include <cstring>
using namespace std;
 
char b[256];
char *solution(const char *f)
{
CharToOem(f,b);
return b;
}
 
 
int main()
{
    setlocale(0,"");
    char *dictionary[][2] = {
    "карандаш","Инструмент для писания",
    "клавиатура", "устройство ввода данных",
    "винтовка","Огнестрельное оружие",
    "самолёт","Воздушное судно с неподвижным крылом",
    "сеть","Взаимосвязанная группа компьютеров",
    "", ""
    };
    char word[80];
    int i;
    cout<<"Введите слово: ";
    cin>>solution(word);
    for(i = 0; *dictionary[i][0]; i++)
    {
        if(!strcmp(dictionary[i][0],word))
        {
            cout<<dictionary[i][1]<<'\n';
            break;
        }
    }
    if(!*dictionary[i][0])
        cout<<solution(word) << " не найдено.\n";
    cin.ignore();
    cin.get();
}
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
M128K145
Эксперт C++
 Аватар для M128K145
8276 / 3495 / 142
Регистрация: 03.07.2009
Сообщений: 10,707
10.09.2009, 17:09     Массив указателей #21
Golovastik, solution преобразовывает char к OEM. word - на тот момент уже OEM. Чтобы корректно вывести OEM в консоль его необходимо преобразовать обратно в чар. А солюшен в той строке работает, только работает правильно, но не так как ты хочеш. А именно она берет word, который в OEM, считает его чаром и еще раз приводит к OEM, получается такой своеобразный оем в квадрате.
Т.е. давай на примере указателей. Есть переменная s. Указатель на нее *s. Чтобы получить s нам надо сделать разыменование, т.е. &(*s), а ты предлагаеш сделать *(*s). Теперь ясно?
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Golovastik
 Аватар для Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
10.09.2009, 17:44  [ТС]     Массив указателей #22
Тоесть вы хотите сказать, что повторное использование функции sulution, изменит уже вновь вывод в кодировку DOS? Я имею ввиду, если я первый раз использую вызов функции solution в строках:
C++
1
2
3
4
5
6
7
8
 for(i = 0; *dictionary[i][0]; i++)
 
        {
                if(!strcmp(solution(dictionary[i][0]),word))
                {
                        cout<<dictionary[i][1]<<'\n';
                        break;
                }
Результат будет русским,так как через этот вызов я перевёл кодировку Дос в 1251(виндоус).
А если я повторно использую вызов этой функции тоесть в строках:
C++
1
2
 if(!*dictionary[i][0])
                cout<<"Слово " <<solution(word)<< " не найдено.\n";
Тогда моя 1251 кодировка виндоус, опять перейдёт в кодировку ДОС, поэтому вы используете дополнительную функцию, которая переводит из Дос, в обратную.
M128K145
Эксперт C++
 Аватар для M128K145
8276 / 3495 / 142
Регистрация: 03.07.2009
Сообщений: 10,707
11.09.2009, 11:33     Массив указателей #23
функции solution и unsolution выполняют одностороннее преобразование, то есть если привели char c помощью CharToOemA, то повторный вызов не приведет обратно к char. Вот попробуй разобрать результат работы
этого кода
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <iostream>
#include <Windows.h>
 
char buf[250];
char* ToOem(char* text)
{
    CharToOemA(text, buf);
    return buf;
}
char buf1[250];
char* ToChar(char* text)
{
    OemToCharA(text, buf1);
    return buf1;
}
int main()
{
    setlocale(LC_ALL, "Russian");
    char text[50];
    std::cout<<"Введи текст\n";
    std::cin>>text;
    char *textOem = ToOem(text);
    char *textChar = ToChar(text);
    std::cout<<text<<'\n';
    size_t i;
    for(i = 0; i < strlen(text); ++i)
        std::cout<<text[i]<<' '<<(int)text[i]<<'\t'<<
              textOem[i]<<' '<<(int)textOem[i]<<'\t'<<
              textChar[i]<<' '<<(int)textChar[i]<<'\n';
 
    char text1[7] = "привет";
    textOem = ToOem(text1);
    textChar = ToChar(text1);
    std::cout<<text1<<'\n';
    for(i = 0; i < strlen(text1); ++i)
        std::cout<<text1[i]<<' '<<(int)text1[i]<<'\t'<<
              textOem[i]<<' '<<(int)textOem[i]<<'\t'<<
              textChar[i]<<' '<<(int)textChar[i]<<'\n';
    system("pause");
    return 0;
}
и сам код
Golovastik
 Аватар для Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
11.09.2009, 14:45  [ТС]     Массив указателей #24
Этот код вроде выводить слова, и их номер кода из таблицы ASCP. Было бы неплохо написать комент в вашем коде, а то минут 35 сижу, не всё ясно.
Вот например,каким образом выводится знак тире между числами?
Вот допустим вот код:

C++
1
(int)text[i]
Что за преобразование и зачем? Что делает функция OemToChar?
CharToOem преобразовует в кодировку 1251, а OemToChar наоборот?
M128K145
Эксперт C++
 Аватар для M128K145
8276 / 3495 / 142
Регистрация: 03.07.2009
Сообщений: 10,707
11.09.2009, 18:00     Массив указателей #25
C++
1
2
3
4
5
6
std::cout<<text[i]<<' '<<(int)text[i]<<'\t'<<//вывод символов введенного
       // текста(OEM) и его код
       textOem[i]<<' '<<(int)textOem[i]<<'\t'<<//вывод символов введенного текста
       //приведенного снова к OEM и его код
       textChar[i]<<' '<<(int)textChar[i]<<'\n';//вывод символов введенного текста
       //приведенного к char и его код
C++
1
2
3
4
5
6
std::cout<<text1[i]<<' '<<(int)text1[i]<<'\t'<<//вывод символов начального
       // текста(char) и его код
       textOem[i]<<' '<<(int)textOem[i]<<'\t'<<//вывод символов начального текста
       //приведенного к OEM и его код
       textChar[i]<<' '<<(int)textChar[i]<<'\n';//вывод символов начального текста
       //приведенного снова к char и его код
C++
1
(int)text[i]// эта строка выдает код символа
ЗЫ. Эта программа по задумке должна продемонстрировать последствия повторного приведения char к char'у и OEM к OEM'у
.::.DIMA.::.
142 / 142 / 4
Регистрация: 26.10.2008
Сообщений: 782
16.09.2009, 15:07     Массив указателей #26
Цитата Сообщение от M128K145 Посмотреть сообщение
Ну а здесь пришел великий и ужасный M128K145 и помог . Вот
ответ
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
#include <iostream>
#include <windows.h>
using namespace std;
 
char b[256];
char *solution(const char *f)
{
    CharToOemA(f,b);
    return b;
}
char *unsolution(const char *f)
{
    OemToCharA(f,b);
    return b;
}
int main()
{
    setlocale(LC_ALL,"Russian");
    char *dictionary[][2] = {
        "карандаш","Инструмент для писания",
        "клавиатура", "устройство ввода данных",
        "винтовка","Огнестрельное оружие",
        "самолёт","Воздушное судно с неподвижным крылом",
        "сеть","Взаимосвязанная группа компьютеров",
        "", ""
    };
    char word[80];
    int i;
    cout<<"Введите слово: ";
    cin>>word;
    for(i = 0; *dictionary[i][0]; i++)
    {
        if(!strcmp(solution(dictionary[i][0]),word))
        {
            cout<<dictionary[i][1]<<'\n';
            break;
        }
    }
    if(!*dictionary[i][0])
        cout<<"Слово " <<unsolution(word)<< " не найдено.\n";
    cin.ignore();
    cin.get();
}


ЗЫ. Люди, слушайте Gravity, он дело говорит
Не могу понять, зачем в данном случае создаётся массив указателей.
C++
1
2
3
4
5
6
7
8
char *dictionary[][2] = {
        "карандаш","Инструмент для писания",
        "клавиатура", "устройство ввода данных",
        "винтовка","Огнестрельное оружие",
        "самолёт","Воздушное судно с неподвижным крылом",
        "сеть","Взаимосвязанная группа компьютеров",
        "", ""
    };
Причём если сделать обычный массив, то вылетает много ошибок.
M128K145
Эксперт C++
 Аватар для M128K145
8276 / 3495 / 142
Регистрация: 03.07.2009
Сообщений: 10,707
16.09.2009, 18:47     Массив указателей #27
qwert, потому что надо реализовывать его как массив char,
вот так
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
int main()
{
    setlocale(LC_ALL,"Russian");
    char dictionary[12][90] = {
        "карандаш","Инструмент для писания",
        "клавиатура", "устройство ввода данных",
        "винтовка","Огнестрельное оружие",
        "самолёт","Воздушное судно с неподвижным крылом",
        "сеть","Взаимосвязанная группа компьютеров",
        "", ""
    };
    char word[80];
    int i;
    cout<<"Введите слово: ";
    cin>>word;
    for(i = 0; *dictionary[i]; i+=2)
        if(!strcmp(solution(dictionary[i]),word))
        {
            cout<<dictionary[i+1]<<'\n';
            break;
        }
    if(!*dictionary[i])
        cout<<"Слово " <<unsolution(word)<< " не найдено.\n";
    cin.ignore();
    cin.get();
}

Или как одномерный массив указателей,
вот
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
int main()
{
    setlocale(LC_ALL,"Russian");
    char *dictionary[12] = {
        "карандаш","Инструмент для писания",
        "клавиатура", "устройство ввода данных",
        "винтовка","Огнестрельное оружие",
        "самолёт","Воздушное судно с неподвижным крылом",
        "сеть","Взаимосвязанная группа компьютеров",
        "", ""
    };
    char word[80];
    int i;
    cout<<"Введите слово: ";
    cin>>word;
    for(i = 0; *dictionary[i]; i+=2)
        if(!strcmp(solution(dictionary[i]),word))
        {
            cout<<dictionary[i+1]<<'\n';
            break;
        }
    if(!*dictionary[i])
        cout<<"Слово " <<unsolution(word)<< " не найдено.\n";
    cin.ignore();
    cin.get();
}

Покажи как ты его реализовывал и я скажу где ты ошибся
.::.DIMA.::.
142 / 142 / 4
Регистрация: 26.10.2008
Сообщений: 782
16.09.2009, 19:01     Массив указателей #28
Я обозначил его как простой массив - ошибка в том, что не указал на сколько точно нужно выделять память.
Меня больше интересует то, что массив указателей - это переменные, хранящие в себе адресы, а здесь прямо в этом массиве записаны не ссылки а значения слов. Меня интересует, как это работает.
M128K145
Эксперт C++
 Аватар для M128K145
8276 / 3495 / 142
Регистрация: 03.07.2009
Сообщений: 10,707
16.09.2009, 19:17     Массив указателей #29
Выделен был двумерный массив указателей на строки. Значение - это и есть эти строки.
C++
1
char *str = "строка";
Физически в str будет адресс начала строки, указатель на первый символ, хоть и записуем туда значение. Точно так же и с нашим массивом. Массив указателей, а эти указатели содержат адреса наших строк
.::.DIMA.::.
142 / 142 / 4
Регистрация: 26.10.2008
Сообщений: 782
16.09.2009, 19:25     Массив указателей #30
А в чём преимущество такого подхода и где эта строка будет хранится, если мы объявляем на неё указатель, должен же быть у неё физический адресс.
M128K145
Эксперт C++
 Аватар для M128K145
8276 / 3495 / 142
Регистрация: 03.07.2009
Сообщений: 10,707
16.09.2009, 19:30     Массив указателей #31
физический адрес - это адрес на который ссылается указатель, строка хранится сразу за указателем, а первый её символ по адресу указателя
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
16.09.2009, 22:46     Массив указателей
Еще ссылки по теме:

Массив указателей C++
опп класс и массив (создать массив указателей по выборке животных, участвующих в забеге) C++
C++ Задачка. массив указателей на одномерный массив
C++ Массив указателей
Массив указателей C++

Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
.::.DIMA.::.
142 / 142 / 4
Регистрация: 26.10.2008
Сообщений: 782
16.09.2009, 22:46     Массив указателей #32
А, теперь понятно. Спасибо.
Yandex
Объявления
16.09.2009, 22:46     Массив указателей
Ответ Создать тему

Метки
chartooem, oemtochar
Опции темы

Текущее время: 05:23. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru