Форум программистов, компьютерный форум, киберфорум
Наши страницы

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 17, средняя оценка - 4.88
Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
#1

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

09.09.2009, 18:00. Просмотров 2127. Ответов 31

Вот,продолжаю изучать тему массивы указателей. В качестве примера, почти уже на последних страницах приводится задачка с использованием указателей в 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();
}
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
09.09.2009, 18:00
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Массив указателей (C++):

Массив указателей на массив строк и сортировка массива указателей - C++
Добрый день. Поступил вопрос. Есть задача. У нас встроенный массив char mass;.Мы вводим строки до тех пор, пока не будет заполнен массив...

по поводу указателей. Как правильно задавать массив указателей и его удалять? - C++
Т.е., например создаю указатель: TPoint *p_Point=NULL; а если массив? TPoint *p_MassPoint; //=?; как массив обнулить не ясно ...

Создать специфицированный шаблон функции, принимающей массив указателей на char и количество самих указателей - C++
Задача: создать специфицированный шаблон функции, принимающей массив указателей на char и количество самих указателей. Вернуть адрес...

Создать специализацию для шаблона, которая принимает массив указателей на строки и количество этих указателей - C++
Нужно создать специализацию для шаблона, которая принимает массив указателей на строки и количество этих указателей. Эта специализация...

Массив из указателей на масив из указателей на массив из int) - C++
Доброго времени суток! Возникла проблема - как на C++ создать массив из указателей на массив из указателей на int? То есть массив из n...

Чем отличаются двумерный символьный массив и массив указателей на строки - C++
Двумерный символьный массив и массив указателей на строки. Собственно чем они отличаются? Заранее спасибо.

31
Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
10.09.2009, 13:11  [ТС] #16
Да,действительно, тоесть функция unsolution,используется чисто для условия неправильного ввода.
0
M128K145
Эксперт С++
8297 / 3517 / 143
Регистрация: 03.07.2009
Сообщений: 10,706
10.09.2009, 13:16 #17
да, как преобразование
0
Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
10.09.2009, 15:02  [ТС] #18
Только до сих пор не могу понять почему OemToChar, ведь там также проверяется условие и CharToOeam стоит.
0
M128K145
Эксперт С++
8297 / 3517 / 143
Регистрация: 03.07.2009
Сообщений: 10,706
10.09.2009, 15:40 #19
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
        setlocale(LC_ALL,"Russian");//после этой строки весь ввод - OEM
        char *dictionary[][2] = {
                "карандаш","Инструмент для писания",
                "клавиатура", "устройство ввода данных",
                "винтовка","Огнестрельное оружие",
                "самолёт","Воздушное судно с неподвижным крылом",
                "сеть","Взаимосвязанная группа компьютеров",
                "", ""
        }; //это все char
        char word[80]; //тоже char
        int i;
        cout<<"Введите слово: ";
        cin>>word;//уже OEM
        for(i = 0; *dictionary[i][0]; i++)
        {
                if(!strcmp(solution(dictionary[i][0]),word))//strcmp(преобразовываем к OEM,OEM)
                {
                        cout<<dictionary[i][1]<<'\n';//выводим char
                        break;
                }
        }
        if(!*dictionary[i][0])
                cout<<"Слово " <<unsolution(word)<< " не найдено.\n";//cout<<char<<преобразовываем к char<<char;
0
Golovastik
11 / 11 / 0
Регистрация: 25.05.2009
Сообщений: 435
10.09.2009, 16:58  [ТС] #20
OemToChar - функция которая переводит с русского на английский?

Почему, функция solution не хочет работать в этой строке?
C++
1
   cout<<"Слово " <<unsolution(word)<< " не найдено.\n";
Ещё раз извините за много вопросов.
0
M128K145
Эксперт С++
8297 / 3517 / 143
Регистрация: 03.07.2009
Сообщений: 10,706
10.09.2009, 17:09 #21
Golovastik, solution преобразовывает char к OEM. word - на тот момент уже OEM. Чтобы корректно вывести OEM в консоль его необходимо преобразовать обратно в чар. А солюшен в той строке работает, только работает правильно, но не так как ты хочеш. А именно она берет word, который в OEM, считает его чаром и еще раз приводит к OEM, получается такой своеобразный оем в квадрате.
Т.е. давай на примере указателей. Есть переменная s. Указатель на нее *s. Чтобы получить s нам надо сделать разыменование, т.е. &(*s), а ты предлагаеш сделать *(*s). Теперь ясно?
0
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 кодировка виндоус, опять перейдёт в кодировку ДОС, поэтому вы используете дополнительную функцию, которая переводит из Дос, в обратную.
0
M128K145
Эксперт С++
8297 / 3517 / 143
Регистрация: 03.07.2009
Сообщений: 10,706
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;
}
и сам код
0
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 наоборот?
0
M128K145
Эксперт С++
8297 / 3517 / 143
Регистрация: 03.07.2009
Сообщений: 10,706
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'у
0
.::.DIMA.::.
143 / 143 / 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] = {
        "карандаш","Инструмент для писания",
        "клавиатура", "устройство ввода данных",
        "винтовка","Огнестрельное оружие",
        "самолёт","Воздушное судно с неподвижным крылом",
        "сеть","Взаимосвязанная группа компьютеров",
        "", ""
    };
Причём если сделать обычный массив, то вылетает много ошибок.
0
M128K145
Эксперт С++
8297 / 3517 / 143
Регистрация: 03.07.2009
Сообщений: 10,706
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();
}

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

опп класс и массив (создать массив указателей по выборке животных, участвующих в забеге) - C++
Здравствуйте! Есть задача... вкратце. Нужно создать массив указателей по выборке животных участвующих в забеге. Это Лошадь, Крокодил,...

Массив строк как массив указателей на массивы чаров - C++
Я всё правильно сделал? char* Files; Files = &quot;AHEAD&quot;; Files = &quot;LEFT&quot;; Files = &quot;RIGHT&quot;; Files = NULL; int searchFile()...

Задачка. массив указателей на одномерный массив - C++
Привет всем. Помогите, пожалуйста. Два дня мучился, писал, оказалось, что писал совсем не то, что нужно. Исходное задание такое: ...

Массив указателей - C++
int* ap; int size = 10; for(int i = 0; i&lt;size; i++) *(ap+i) = new int; Подскажите, для чего в...


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

Или воспользуйтесь поиском по форуму:
30
Ответ Создать тему
Опции темы

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