Ru-Board.club
← Вернуться в раздел «Прикладное программирование»

» Вопросы по программированию на C/С++

Автор: Abs62
Дата сообщения: 19.03.2006 19:11
inner
После отключения CodeGuard надо всё перекомпилировать заново. Иначе в старых obj останутся ссылки на его библиотеки.
Автор: LuckyELF
Дата сообщения: 19.03.2006 19:16
На дня писал такой код:

#define PI 3.1415

...
...

float z = 3 / 8 * PI;

почему-то z все время было равно 0, что меня сильно возмутило и я написал следующие:

float z1 = 3 / 8;

и опять получил 0, хотя по идее должно быть 0,375. Писал все это в Turbo C++ 3.0, потом еще попробывал в Borland C++ Builder 6.0 таже картина. Кто нить знает в чем дело?
Автор: EZH
Дата сообщения: 19.03.2006 20:38
LuckyELF
Попробуй написать одно из чисел как вещественное, с нулевой дробной частью. Т.е. типа 3.0
Автор: inner
Дата сообщения: 19.03.2006 22:56
Abs62

Естественно перекомпилирую. Но откуда-то в obj остаются ссылки на codeguard.
уже смирился с ним. но все-таки... это баг? Теперь даже пустой апликейшн после включения CodeGuard уже не хочет с ним расставаться. ёпт. просто полный ахтунг.

Еще и достала проблема с CodeCompetion, когда в проекте количество строк/методов достигает определенного количества, CodeCompletion перестает работать. Говорят, это фича билдера.
Автор: Swappp
Дата сообщения: 19.03.2006 23:14
LuckyELF
В этом случае происходит деление двух целых чисел и результат получается целым... надо либо как сказал EZH записать одно из чисел в виде числа с плавающей точкой либо привести одно из чисел к такому типу т.е.

Код: float z1 = (float)3 / 8;
Автор: DeVOuR
Дата сообщения: 20.03.2006 22:55
Почему у меня постоянно выскакивает окно CPU, когда я пытаюсь ввести какую-то переменную с клавиатуры? причем код абсолютно правильный, даже без ворнингов компилируется. Но как только я ввожу в запущеной программе переменную - появляется этот екран CPU. Что мне делать? и есть у кого-то gcc? Можете выложить где-то?
Автор: Labutin
Дата сообщения: 21.03.2006 16:44
Вот код:

Цитата:

int n=1;
do
{
cout << "Input n >= 10"<<endl;
cin >> n;
}
while (n<10);

если ввести вместо цифры, например, букву "a", то программа зациклится. Как это можно избежать?
Автор: wInuX
Дата сообщения: 21.03.2006 16:56
Labutin
Например так


Код:
int n=1;
do
{
cout << "Input n >= 10"<<endl;
cin >> n;
if (!cin) {
cout << "error\n";
break;
}
} while (n<10);
Автор: Labutin
Дата сообщения: 21.03.2006 16:59
wInuX
Спасибо. А как буфер очистить, чтобы не сразу на выход, а еще раз спросить пользователя?
Вот еще чего нашел:

Цитата:

if (cin.fail())
{
cin.clear();
}

Тоже подходит. Но вот как буфер введенных букв вместо цифр очистить?
Автор: wInuX
Дата сообщения: 22.03.2006 12:29
Labutin

Цитата:
Но вот как буфер введенных букв вместо цифр очистить?

вроде никак. только полная отчитска.
Автор: CrazyJohn
Дата сообщения: 26.03.2006 09:52
Возникла ошибка при компиляции проги, использующей шаблоны.
Пусть есть 3 файла (temp.h, tenp.cpp, test.cpp), соответственно первые 2 файла - это объявление и реализация класса, использующего шаблоны,т.е. что-то типа
template <class C> Temp
{
// ...
}

Вообще по ходу надо помещать объявление и реализацию в один класс (при использовании шаблонов) - тогда все работает, но если разбить на 2 файла (объявление и реализация), то ни фига не работает =(
Версия компилятора borland c++ 3.1.

В help-е пишут что-то типа: надо использовтаь опцию компилятра -Jgx при компиляции, чтобы была возможность слинковать файл с реализацией класса, использующего шаблоны, но я че-то не догоняю, как это сделать.... Из среды никак, а как компилить из командной строки - не знаю =(

Вроде бы проблема стандартная, но че-то найти ее решения не могу =(

Может кто-нибудь знает ее решение ?
Автор: Mickey_from_nsk
Дата сообщения: 27.03.2006 04:56
CrazyJohn
Вообще, не знаю как в этом диалекте борланда, но в умных книжках со ссылкой на стандарт пишут, что раздельной компиляции для шаблонов быть не должно, поскольку текст инстанциируемых методов "вкомпилевывается" в код модуля, инстанциирующего шаблон.
Шаблоны не дают никакой информации для линкера. Они работают только на уровне компиляции.
То есть, если уж ОЧЕНЬ хочется разделить код, то надо будет включать не header шаблона, а его cpp.
Автор: WiseAlex
Дата сообщения: 27.03.2006 09:07
Mickey_from_nsk

Цитата:
но в умных книжках со ссылкой на стандарт пишут, что раздельной компиляции для шаблонов быть не должно

как всегда не все так однозначно. Для этих целей предназначено ключевой слово export, однако ситуация с ним не такая уж однозначная, как хотелось бы. По этому поводу можно почитать у Саттера (Новые сложные задачи на С++. Задачи 9,10). Там например говорится о том, что export реализован только в одном компиляторе (и это не borland)

Цитата:
то надо будет включать не header шаблона, а его cpp.

уж лучше все написать в h файле, т.к. суть таже, а понятности больше

Автор: Mickey_from_nsk
Дата сообщения: 27.03.2006 09:26
WiseAlex
Кстати, в том числе, эту умную книжку я и имел в виду. Если не ошибаюсь, там он и приводит причины, по которым не стоит вообще с этим связываться.
Автор: Goul
Дата сообщения: 27.03.2006 10:06
CrazyJohn
Для того чтобы разрешить использование шаблона, необходимо, чтобы он был инстанцирован. Разовьём Ваш пример. В файле template.h:

Код: template <typename T>
class CDemo {
public:
int DoSomething(const T);
}
Автор: lungtung
Дата сообщения: 30.03.2006 09:27
Я написал программу:

#include <string.h>
#include <stdio.h>

void main( void )
{
char* string = "aaa";

strcpy( string, "Hello world from " ); // C4996
printf( "String = %s\n", string );
}

если я исползовал Borland C++3.1, то она нормально работает. Но если я исползовал Visual C++6.0, то она неработает, и появилось окно ошибочного сообщения.

Я не понял почему так? Объясните пожалуйста!!!
Автор: Goul
Дата сообщения: 30.03.2006 10:00
lungtung
И тут вместо C++ подсовывают C...
Вы завели указатель string, который указывает на строку "aaa", имеющую длину 4 (включая '\0'), а пытаетесь туда записать другую длиннющую строку "Hello world from " - конечно, будете писать свою строку непонятно куда. Просто повезло, что борландовская версия работает. Надо сперва выделить соответствующую память, а уже потом копировать:

Код: int nLen = strlen("Hello world from ");
char* string = (char*)malloc(nLen + 1); // Раз уж C, то пользуемся malloc'ом
strcpy(string, "Hello world from ");
printf("String = %s\n", string);
free(string);
Автор: lungtung
Дата сообщения: 30.03.2006 12:13
спасибо goul за ответ! Но ещё не ясно. Конечно я могу написать так:
------------------------------
int len = strlen("Hello world");
char * string = new char[len+1];
strcpy(string,"Hello world");

или:
char string[20];
strcpy(string,"Hello world");
----------------------------------
И она будет нормально работать.
Но при использовании указатель :
char* string = "12345678901234567890"; // даже 20 символов длинее "Hello world";
strcpy(string,"Hello world") ; // ---> не работает???
----------------------------------
Видно, что указатель указывает адрес, в котором сохраняет строку:"12345678901234567890". Вопрос : Почему функция strcpy в этой ситуации не работает в VC, а в Borland C она нормально работает?





Добавлено:
И ещё вопрос:
Я написал программу:
------------------------------------------------------
char* copystr(char* st1, char* st2)
{
    int len = strlen(st2);

    for(int i = 0; i < len; i++) st1[i] = st2[i];
    st1[len] = 0;

    return st1;
}

void main()
{
char* string = "aaaaaaaaa",st[] = "123";

char* res = copystr( st, "Hello " ); // нормально работает
printf( "String = %s\n", res ); // String = "Hello"
printf( "String = %s\n", st ); // String = "Hello"

copystr(string,"Hello"); // не работает?
printf( "String = %s\n", string );
}
-------------------------------------------------------------
Я не полнял что, почему она не работает в Visual C 6.0. Объясните пожалуйста!
Автор: Labutin
Дата сообщения: 30.03.2006 12:39

Цитата:
char* string = "aaaaaaaaa",st[] = "123";

char* res = copystr( st, "Hello " ); // нормально работает

под строчку st было выделено 4 байта
после copystr( st, "Hello " ); данные после этой строки были затерты
дальше поведение программы становится непредсказуемым!!!
Автор: wInuX
Дата сообщения: 30.03.2006 12:44
lungtung
В Visual C 6.0 программа разделена на сегменты с различным доступом. константы чаше всего помещаются с readonly доступом. поэтому при попытку записи ты получаешь Access Violation.

Пример с st[] = "123" сработал скоорее всего потому что компилятор провел оптимизации и забросил значение "123\0" в стек (а не в константный сегмент) с помощью команды push. Поэтому у тебя получилось изменить это значение.
Автор: strah4
Дата сообщения: 30.03.2006 14:02
Читал тут Страуструпа и в задачках ко второй главе нашел вроде бы простенький вопрос над которым бьюсь третий час.

Напишите описание ссылки на массив из 10 целых.

Указатель понятно, ссылка на целое понятно, указатель на массив вполне.

Ссылка на массив непонятно. Как-то всегда без подобного обходился, очень интерестно. Попробовал 10-ок вариантов, ни один не был пропущен компилятором. Помогите теперь вы.
Автор: wInuX
Дата сообщения: 30.03.2006 14:23
strah4

int (&a)[10];
Автор: strah4
Дата сообщения: 30.03.2006 14:34
Сенькс.

А то тут уже вот такой изврат случился, хотя и красивый.

typedef int mas[10];
mas a;
a[0]=5; // для примера
mas &b=a;
cout << b[0];

...выдает 5
Автор: lungtung
Дата сообщения: 30.03.2006 14:37
Теперь понял. Спасибо всем!
Автор: HANDLE
Дата сообщения: 01.04.2006 16:25
lungtung

Цитата:
Но при использоффании указатель :
char* string = "12345678901234567890"; // даже 20 символофф длинее "Hello world";
strcpy(string,"Hello world") ; // ---> нихуя ненапрегаиццо, нах???


Советую тебе разобраться с переменными, как они объявляются, где располагаются.
Да и с типами переменных заодно. А так же со стеком, что и когда туда помещается.

char* string = "12345678901234567890";
Не вполне корректное объявление. В данном случае string - это всего лишь указатель расположенный в стеке. В него помещается адрес строки "12345678901234567890".
А где расположена эта строка? В секции констант. А где расположена строка "Hello world"? Там же. Более соответствующее действительности объявление будет выглядеть так:
const char* string = "12345678901234567890";

Твоя программа пытается произвести операцию записи в секцию только для чтения, за что получит "штраф" либо ещё на стадии компиляции, либо во время выполнения.

char string[] = "12345678901234567890";
strcpy(string,"Hello world") ;

При таком объявлении в стек помещается уже не указатель, а массив, который будет проинициализирован. Его размер будет определяться размером инициализирующей строки.

char string[100] = "12345678901234567890";
strcpy(string,"Hello world") ;

Тоже самой, только с заданным размером string (100). И учти, что если ты копируешь в локальную переменную больше, чем она может принять, то ты испортишь содержимое стека, и программа сделает "RunTime Error"


Есть ещё один вариант - глобальные переменные.
char string[] = "12345678901234567890";

void Function()
{
strcpy(string,"Hello world") ;
}

Здесь переменная string уже не в стеке, а в секции переменных, куда можно что то записывать.
Автор: JekaCh
Дата сообщения: 02.04.2006 23:34
В инсте есть задача написать прогу для работы с внешним железом. Работа производится непосредственной записью в память определённых значений. Есть прога недописанная и не обладающая необходимым функционалом, но имеющая правильные мысли написанная на Delphi. Так как Borland Builder я знаю получше хотелось бы перевести нужные мне вещи на язык builder, но знаний не хватает .
Вот как выглядит нужный мне кусок на Delphi:
int N, MyAdress;
BaseAdress:=$D0000; -такое же прямое указание адресса в билдере не проходит
procedure WriteWord95(Address:integer; Value:word);
var Data:^word; - это насколько я понял мона заменить на unsigned short *Data;
P:pointer; - это на void *P;
begin
P:=Ptr(Address); функция Ptr переводит указанный адресс в указатель, как сделать это в Билдере?
Data:=P;
Data^:=Value;
end;
MyAdress=BaseAdress+$20*N;
WriteWord95 (MyAdress, 4);
Как то же самое будет выглядеть в билдере? Если кто знает помогите плиз!
Автор: Mickey_from_nsk
Дата сообщения: 03.04.2006 09:12
JekaCh
Примерно так.

Код:
typedef unsined short word;

int BaseAddress = 0xD0000;

void WriteWord95(int address, word value)
{
word* Data = Address;
*Data = value;
}

void main(void)
{
int N = ?????;
int MyAddress = BaseAddress+0x20*N;
WriteWord95(MyAddress, 4);
}

Автор: JekaCh
Дата сообщения: 03.04.2006 16:56
Я уж понял что в ВыньXp ничего не выйдет но в инсте старые компы там вынь98 так что всё нормально с допуском в память . А за помощь большое спасибо завтра в инсте попробую .
А ещё вопросик если я хочу прочитать то это будет выглядеть так?

Цитата:
typedef unsigned char byte;
byte ReadByte95(int address)
{
byte* Data=address;
byte result=*Data;
return result;
}
?


Добавлено:
Сейчас дома попробовал скомпилировать получил ошибку к строчке word* Data = Address;

Цитата:
[C++ Error] main.cpp(12): E2034 Cannot convert 'int' to 'unsigned short *'

Автор: wInuX
Дата сообщения: 03.04.2006 18:30
JekaCh


Цитата:
...попробовал скомпилировать получил ошибку к строчке word* Data = Address;

надо добавить явное преобразование.

byte* Data= (byte*)address;
Автор: JekaCh
Дата сообщения: 03.04.2006 19:19
Спасибо, помогло! А по поводу чтения-то правильно я написал считывание из памяти?

Цитата:
typedef unsigned char byte;
byte ReadByte95(int address)
{
byte* Data= (byte*)address;
byte result=*Data;
return result;
}

Страницы: 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193

Предыдущая тема: не знаю как назвать тему :-)


Форум Ru-Board.club — поднят 15-09-2016 числа. Цель - сохранить наследие старого Ru-Board, истории становления российского интернета. Сделано для людей.