Problem z deklaracją typów, rekordami, wskaźnikami...

Problem z deklaracją typów, rekordami, wskaźnikami...
R9
  • Rejestracja:ponad 16 lat
  • Ostatnio:ponad 12 lat
0

Witam!
Programuję w Delphi od jakiegoś czasu, ale teraz przyszło mi się zmierzyć z rekordami, nowymi typami i wskaźnikami - nie znam się na tym, oczywiście poczytałam o tym, jednak nadal mam problem - w momencie kompilacji wyskakuje błąd Expected ":" but "=" found w linii TElement = record czyli gdzie deklaruję strukturę.
Oto fragment mojego kodu:

Kopiuj
unit MainUnit;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls, ComCtrls;

type
  TMainForm = class(TForm)
    PLMemo: TMemo;
    ...
    SzukajButton: TButton;

    TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      ...
      Rok : Integer;

    PElement = ^Element;   // zadeklarowalam typ wskaznikowy


    procedure Wczytaj(Sender: TObject);
    procedure Zapisz(Sender: TObject; var Action: TCloseAction);
    procedure SzukajButtonClick(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }



  Root : PElement; // poczatek listy - wskaznik
  Last : PElement; // koniec listy - wskaznik

  end;

var
  MainForm: TMainForm;

implementation

uses SzukajUnit;

{$R *.dfm}

procedure TMainForm.Wczytaj(Sender: TObject);
...
end;

procedure TMainForm.Zapisz(Sender: TObject; var Action: TCloseAction);
begin
  // zapisanie bazy danych

  // posprzatanie listy
end;

procedure TMainForm.SzukajButtonClick(Sender: TObject);
begin
  SzukajForm.Visible := True;
end;

end.

Program to generalnie baza filmów, ma wczytać z pliku txt bazę do listy jednokierunkowej i ją wyświetlić (jak na razie tyle). Niestety wcześniej nie miałam styczności ze wskaźnikami, listami i nowymi typami i dlatego już mam problemy. Jak zmienię znak = na : to występuje następny problem - czyli błąd nie leży w tym znaku, wydaje mi się, że jest gdzieś indziej.
Z góry dziękuję za pomoc i pozdrawiam,
Asia

flowCRANE
Rekord musisz zapisać jako deklarację struktury, nie typu; Takie coś nie przejdzie: TElement = record, a to: TElement: record tak; Poza tym rekord trzeba zamknąć słowem end; Jeżeli chcesz zadeklarować typ - wyciąg go z klasy MainForm;
Patryk27
Moderator
  • Rejestracja:ponad 17 lat
  • Ostatnio:ponad rok
  • Lokalizacja:Wrocław
  • Postów:13042
0
Kopiuj
type
  TMainForm = class(TForm)
    PLMemo: TMemo;
    ...
    SzukajButton: TButton;
 
    TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      ...
      Rok : Integer;
 
    PElement = ^Element;   // zadeklarowalam typ wskaznikowy

1.Gdzieś zgubiłaś "end;".
2.Deklarację rekordu zaczyna się od "type"

Kopiuj
Type TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      ...
      Rok : Integer;
 end;

Tylko tak, jak próbujesz zrobić się nie da.
Musisz wywalić rekord przed klasę lub za nią.
Np:

Kopiuj
type TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      ...
      Rok : Integer;
  end;
type PElement = ^Element;   // zadeklarowalam typ wskaznikowy
type
  TMainForm = class(TForm)
    PLMemo: TMemo;
    ...
    SzukajButton: TButton;
 
 
    procedure Wczytaj(Sender: TObject);
    procedure Zapisz(Sender: TObject; var Action: TCloseAction);
    procedure SzukajButtonClick(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
 
 
 
  Root : PElement; // poczatek listy - wskaznik
  Last : PElement; // koniec listy - wskaznik
 
  end;

IMHO, byłoby lepiej, jakbyś zaczęła naukę podstaw w Free Pascalu.


edytowany 3x, ostatnio: Patryk27
R9
  • Rejestracja:ponad 16 lat
  • Ostatnio:ponad 12 lat
0

Ja już ze słowami end i type kombinowałam (jestem dziewczyną).
Wpisałam type przed deklarację struktury. Po kompilacji Delphi krzyczy Expected end but type found w miejscu gdzie to type dopisałam, a więc dopisuję end po deklaracji formy. Kompiluję - Delphi krzyczy, że metoda "wczytaj" nie istnieje, czy skasować referencję. Tylko że ta metoda istnieje.. więc odpowiadam, żeby nie kasował referencji - wtedy wywala na linijkę Next : PElement; // wskaznik na nastepny element listy, że Undeclared identifier "PElement" - a przecież jest zadeklarowany chwilę później - to może być dla niego problem, dlatego przenoszę część kodu z deklaracją struktury za deklarację "PElement". Kompiluję. I teraz błąd jest na linijce kodu z deklaracją "PElement", bo przecież Undeclared identifier "Element, czyli musiałabym przenieść tę linijkę z powrotem. Fragment zmienionego od ostatniego posta kodu wygląda teraz tak:

Kopiuj
type
  TMainForm = class(TForm)
    PLMemo: TMemo;
   ...
    SzukajButton: TButton;
    end;

    PElement = ^Element;   // zadeklarowalam typ wskaznikowy

    type TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      ...
      Rok : Integer;

    procedure Wczytaj(Sender: TObject);

EDIT:
Zrobiłam, najpierw rekord, później wskaźnik, później forma, ale wywala błąd "Undeclared identifier: 'PElement'" przy deklaracji wskaźnika w strukturze. Niestety projekt ma być napisany w Delphi, jakieś podstawy już mam (np. umiem zrobić grę kółko i krzyżyk :D), ale może akurat nie te, które by mi się tu przydały ;) Myślę, że jest w moim programie "zapętlenie" - nie mogę zrobić struktury ze wskaźnikiem przed deklaracją wskaźnika i nie mogę zrobić wskaźnika na rekord bez deklaracji rekordu... Ale jak z tej "pętli" wyjść?

Kopiuj
unit MainUnit;

interface

uses
  Windows, ...;

type TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      ...
      Rok : Integer;
type PElement = ^Element;   // zadeklarowalam typ wskaznikowy

type
  TMainForm = class(TForm)
    PLMemo: TMemo;
    ...
    SzukajButton: TButton;


    procedure Wczytaj(Sender: TObject);
    procedure Zapisz(Sender: TObject; var Action: TCloseAction);
    procedure SzukajButtonClick(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }



  Root : PElement; // poczatek listy - wskaznik
  Last : PElement; // koniec listy - wskaznik

  end;

var
  MainForm: TMainForm;

implementation

uses SzukajUnit;

{$R *.dfm}

procedure TMainForm.Wczytaj(Sender: TObject);
...
end;

procedure TMainForm.Zapisz(Sender: TObject; var Action: TCloseAction);
begin
  // zapisanie bazy danych

  // posprzatanie listy
end;

procedure TMainForm.SzukajButtonClick(Sender: TObject);
begin
  SzukajForm.Visible := True;
end;

end.
edytowany 4x, ostatnio: rajaaa92
Patryk27
Wrzuć aktualny kod.
cimak
  • Rejestracja:ponad 22 lata
  • Ostatnio:ponad 9 lat
  • Postów:1668
1

chyba tak mozna...:

Kopiuj
type PElement=^TElement;
   TElement = record
      Next : PElement; // wskaznik na nastepny element listy
      PL : String;
      Rok : Integer;
end;
R9
chyba nie -> [Error] Undeclared identifier: 'Element' przy type PElement=^TElement;
cimak
bo tam nie ma nigdzie "Element". Jest TElement.
R9
  • Rejestracja:ponad 16 lat
  • Ostatnio:ponad 12 lat
0

Faktycznie! Dzięki! Robiłam to na podstawie kursów z 4programmers, więc przepisywałam tak jak tam było. Dlaczego raz trzeba napisać PElement, raz TElement, a czasem samo Element? Rozumiem, że ma to pomóc w odnalezieniu się, czy dane słowo reprezentuje zmienną, czy typ. PElement to pointer czyli wskaźnik, a TElement to struktura. Czyli pisząc type PElement = ^TElement; deklaruję typ wskaźnikowy, który będzie wskazywał na zmienną typu strukturalnego, dobrze rozumiem? Czyli cały problem polegał na tym, że zabrakło tej jednej literki?

cimak
  • Rejestracja:ponad 22 lata
  • Ostatnio:ponad 9 lat
  • Postów:1668
1
rajaaa92 napisał(a)

Robiłam to na podstawie kursów z 4programmers, więc przepisywałam tak jak tam było.
Wskaźniki. Listy jedno i dwukierunkowe.
tu jest identycznie wiec albo cos przeoczylas albo korzystalas z innego zrodla w ktorym byl blad.

Dlaczego raz trzeba napisać PElement, raz TElement, a czasem samo Element?
nie ma takiego typu "Element". mozesz co najwyżej nazwac tak sobie zmienna lub pole.

Czyli pisząc type PElement = ^TElement; deklaruję typ wskaźnikowy, który będzie wskazywał na zmienną typu trukturalnego, dobrze rozumiem?
dokladnie (chociaz mozna by sie przyczepic za slowo "zmienna"- bo to nie musi byc zmienna, moze byc np elementem tablicy).

Czyli cały problem polegał na tym, że zabrakło tej jednej literki?
czytaj bledy kompilatora. jesli powiedzial [Error] Undeclared identifier: 'Element' tzn ze nie wie co to jest Element i wtedy faktycznie problemem jest to ze brakuje jednej literki.

edytowany 5x, ostatnio: cimak
R9
Prawda, korzystałam też z uczelnianej stronki, ale tam było to zdecydowanie mniej jasno opisane. Przeoczyłam tę literkę, mój błąd. Jeżeli mam dalsze problemy z tym programem, mogę dalej ciągnąć ten wątek? (Kompilator nic nie mówi, program po prostu nie działa)
cimak
mozesz, najwyzej zostanie przeniesiony do dzialu Newbie, ale lepsze to, niz twozenie osobnego watku w Newbie.
R9
  • Rejestracja:ponad 16 lat
  • Ostatnio:ponad 12 lat
0

A więc tak jak wspomniałam, program ma (jak na razie) na starcie wczytać bazę danych z pliku txt do listy jednokierunkowej a następnie ją wyświetlić (używam memo).
W OnCreate() mam procedurę Wczytaj, która wygląda tak:

Kopiuj
procedure TMainForm.Wczytaj(Sender: TObject);
var
  Baza : TextFile;
  Linia : String;
  NewOne : PElement;
  AtList : PElement;
begin
  // wczytanie bazy danych z pliku do listy i z listy do memo
  Root := nil;
  Last := nil;
  AtList := Root;
  AssignFile(Baza, 'D:\Dokumenty\Studia\PP\Delphi\02 baza filmów\baza.txt');
  if FileExists('D:\Dokumenty\Studia\PP\Delphi\02 baza filmów\baza.txt') then
    Reset(Baza)
  else
    Rewrite(Baza);
  while not Eof(Baza) do
  begin
    Readln(Baza, Linia);
    New(NewOne);
    NewOne^.Next := nil;
    NewOne^.PL := Copy(Linia, 1, Pos(';', Linia) - 1);    // wycina z linii tektu oddzielonej
    Linia := Copy(Linia, Pos(';', Linia) + 1, 1000);      // srednikami odpowiednie wartosci
    NewOne^.Oryg := Copy(Linia, 1, Pos(';', Linia) - 1);
    Linia := Copy(Linia, Pos(';', Linia) + 1, 1000);
    NewOne^.CzasH := StrToInt(Copy(Linia, 1, Pos(';', Linia) - 1));
    Linia := Copy(Linia, Pos(';', Linia) + 1, 1000);
    NewOne^.CzasM := StrToInt(Copy(Linia, 1, Pos(';', Linia) - 1));
    Linia := Copy(Linia, Pos(';', Linia) + 1, 1000);
    NewOne^.CzasS := StrToInt(Copy(Linia, 1, Pos(';', Linia) - 1));
    Linia := Copy(Linia, Pos(';', Linia) + 1, 1000);
    NewOne^.Jezyk := Copy(Linia, 1, Pos(';', Linia) - 1);
    Linia := Copy(Linia, Pos(';', Linia) + 1, 1000);
    NewOne^.Rok := StrToInt(Linia);
    if Root = nil then
    begin
      Root := NewOne;
      Last := Root;
    end else begin
      Last^.Next := NewOne;
      Last := NewOne;
    end;
  end;
  while AtList <> nil do
  begin
    PLMemo.Lines.Add(AtList^.PL);
    OrygMemo.Lines.Add(AtList^.Oryg);
    CzasMemo.Lines.Add(IntToStr(AtList^.CzasH) + ':' + IntToStr(AtList^.CzasM) + ':' +
                           IntToStr(AtList^.CzasS));
    JezykMemo.Lines.Add(AtList^.Jezyk);
    RokMemo.Lines.Add(IntToStr(AtList^.Rok));
    AtList := AtList^.Next;
  end;
end;

Również na podstawie 4programmers, dostosowane do moich potrzeb. Jednak kiedy forma się tworzy, w moich memo nie pojawia się zupełnie nic...

Patryk27
Moderator
  • Rejestracja:ponad 17 lat
  • Ostatnio:ponad rok
  • Lokalizacja:Wrocław
  • Postów:13042
0

A wywołujesz tę procedurę?


R9
  • Rejestracja:ponad 16 lat
  • Ostatnio:ponad 12 lat
0

Jest wpisana w pole "OnCreate" w zdarzeniach mojej głównej formy - czyli jest wywoływana na starcie, prawda?

Zobacz pozostałe 2 komentarze
R9
jest wywoływana, sprawdziłam jak poleciłeś, czytam z czym się je debuggera :)
Patryk27
Może plik nie istnieje?
Patryk27
To pozostaje tylko debugger.
R9
Dzięki Wam za pomoc, zajmę się tym jutro.
cimak
  • Rejestracja:ponad 22 lata
  • Ostatnio:ponad 9 lat
  • Postów:1668
0

przede wzystkim rob wieksze wciecia w kodzie, np:

Kopiuj
if Root = nil then
   begin
     Root := NewOne;
     Last := Root;
   end
else
   begin
     Last^.Next := NewOne;
     Last := NewOne;
   end;

zupelnie inaczej sie to pozniej czyta.

po drugie: jesli Ci "cos" nie dziala to pomysl jak mozesz sprawdzic w ktorym miejscu wystepuje blad: np jesli odczytujesz cos z pliku to sprawdzaj zwyklym showmessage() jaki tekst zostal odczytany. jesli tniesz tekst na kawalki to sprawdzaj czy wszystko sie zgadza.

druga sprawa: te wszystkie pos(), copy() itd... kazdy robi jak lubi ja preferuje inne rozwiazanie w takich wypadkach:
masz tu kod, mysle ze zrozumiesz jak tego uzywac:

Kopiuj
procedure TForm1.Button1Click(Sender: TObject);
var i:integer;
    sl:TStringList;
    s:String;
begin
  sl:=TStringList.Create;
  s:='Jan;Kowalski;19;4;1938;Stefan;Pracownik;0;119287;Jakis inny parametr';
  sl.Text:=StringReplace(s, ';', #13#10, [rfReplaceAll]);

  for i:=0 to sl.Count-1 do
    showmessage(sl[i]);

  sl.Free;
end;
edytowany 3x, ostatnio: cimak
Kliknij, aby dodać treść...

Pomoc 1.18.8

Typografia

Edytor obsługuje składnie Markdown, w której pojedynczy akcent *kursywa* oraz _kursywa_ to pochylenie. Z kolei podwójny akcent **pogrubienie** oraz __pogrubienie__ to pogrubienie. Dodanie znaczników ~~strike~~ to przekreślenie.

Możesz dodać formatowanie komendami , , oraz .

Ponieważ dekoracja podkreślenia jest przeznaczona na linki, markdown nie zawiera specjalnej składni dla podkreślenia. Dlatego by dodać podkreślenie, użyj <u>underline</u>.

Komendy formatujące reagują na skróty klawiszowe: Ctrl+B, Ctrl+I, Ctrl+U oraz Ctrl+S.

Linki

By dodać link w edytorze użyj komendy lub użyj składni [title](link). URL umieszczony w linku lub nawet URL umieszczony bezpośrednio w tekście będzie aktywny i klikalny.

Jeżeli chcesz, możesz samodzielnie dodać link: <a href="link">title</a>.

Wewnętrzne odnośniki

Możesz umieścić odnośnik do wewnętrznej podstrony, używając następującej składni: [[Delphi/Kompendium]] lub [[Delphi/Kompendium|kliknij, aby przejść do kompendium]]. Odnośniki mogą prowadzić do Forum 4programmers.net lub np. do Kompendium.

Wspomnienia użytkowników

By wspomnieć użytkownika forum, wpisz w formularzu znak @. Zobaczysz okienko samouzupełniające nazwy użytkowników. Samouzupełnienie dobierze odpowiedni format wspomnienia, zależnie od tego czy w nazwie użytkownika znajduje się spacja.

Znaczniki HTML

Dozwolone jest używanie niektórych znaczników HTML: <a>, <b>, <i>, <kbd>, <del>, <strong>, <dfn>, <pre>, <blockquote>, <hr/>, <sub>, <sup> oraz <img/>.

Skróty klawiszowe

Dodaj kombinację klawiszy komendą notacji klawiszy lub skrótem klawiszowym Alt+K.

Reprezentuj kombinacje klawiszowe używając taga <kbd>. Oddziel od siebie klawisze znakiem plus, np <kbd>Alt+Tab</kbd>.

Indeks górny oraz dolny

Przykład: wpisując H<sub>2</sub>O i m<sup>2</sup> otrzymasz: H2O i m2.

Składnia Tex

By precyzyjnie wyrazić działanie matematyczne, użyj składni Tex.

<tex>arcctg(x) = argtan(\frac{1}{x}) = arcsin(\frac{1}{\sqrt{1+x^2}})</tex>

Kod źródłowy

Krótkie fragmenty kodu

Wszelkie jednolinijkowe instrukcje języka programowania powinny być zawarte pomiędzy obróconymi apostrofami: `kod instrukcji` lub ``console.log(`string`);``.

Kod wielolinijkowy

Dodaj fragment kodu komendą . Fragmenty kodu zajmujące całą lub więcej linijek powinny być umieszczone w wielolinijkowym fragmencie kodu. Znaczniki ``` lub ~~~ umożliwiają kolorowanie różnych języków programowania. Możemy nadać nazwę języka programowania używając auto-uzupełnienia, kod został pokolorowany używając konkretnych ustawień kolorowania składni:

```javascript
document.write('Hello World');
```

Możesz zaznaczyć również już wklejony kod w edytorze, i użyć komendy  by zamienić go w kod. Użyj kombinacji Ctrl+`, by dodać fragment kodu bez oznaczników języka.

Tabelki

Dodaj przykładową tabelkę używając komendy . Przykładowa tabelka składa się z dwóch kolumn, nagłówka i jednego wiersza.

Wygeneruj tabelkę na podstawie szablonu. Oddziel komórki separatorem ; lub |, a następnie zaznacz szablonu.

nazwisko;dziedzina;odkrycie
Pitagoras;mathematics;Pythagorean Theorem
Albert Einstein;physics;General Relativity
Marie Curie, Pierre Curie;chemistry;Radium, Polonium

Użyj komendy by zamienić zaznaczony szablon na tabelkę Markdown.

Lista uporządkowana i nieuporządkowana

Możliwe jest tworzenie listy numerowanych oraz wypunktowanych. Wystarczy, że pierwszym znakiem linii będzie * lub - dla listy nieuporządkowanej oraz 1. dla listy uporządkowanej.

Użyj komendy by dodać listę uporządkowaną.

1. Lista numerowana
2. Lista numerowana

Użyj komendy by dodać listę nieuporządkowaną.

* Lista wypunktowana
* Lista wypunktowana
** Lista wypunktowana (drugi poziom)

Składnia Markdown

Edytor obsługuje składnię Markdown, która składa się ze znaków specjalnych. Dostępne komendy, jak formatowanie , dodanie tabelki lub fragmentu kodu są w pewnym sensie świadome otaczającej jej składni, i postarają się unikać uszkodzenia jej.

Dla przykładu, używając tylko dostępnych komend, nie możemy dodać formatowania pogrubienia do kodu wielolinijkowego, albo dodać listy do tabelki - mogłoby to doprowadzić do uszkodzenia składni.

W pewnych odosobnionych przypadkach brak nowej linii przed elementami markdown również mógłby uszkodzić składnie, dlatego edytor dodaje brakujące nowe linie. Dla przykładu, dodanie formatowania pochylenia zaraz po tabelce, mogłoby zostać błędne zinterpretowane, więc edytor doda oddzielającą nową linię pomiędzy tabelką, a pochyleniem.

Skróty klawiszowe

Skróty formatujące, kiedy w edytorze znajduje się pojedynczy kursor, wstawiają sformatowany tekst przykładowy. Jeśli w edytorze znajduje się zaznaczenie (słowo, linijka, paragraf), wtedy zaznaczenie zostaje sformatowane.

  • Ctrl+B - dodaj pogrubienie lub pogrub zaznaczenie
  • Ctrl+I - dodaj pochylenie lub pochyl zaznaczenie
  • Ctrl+U - dodaj podkreślenie lub podkreśl zaznaczenie
  • Ctrl+S - dodaj przekreślenie lub przekreśl zaznaczenie

Notacja Klawiszy

  • Alt+K - dodaj notację klawiszy

Fragment kodu bez oznacznika

  • Alt+C - dodaj pusty fragment kodu

Skróty operujące na kodzie i linijkach:

  • Alt+L - zaznaczenie całej linii
  • Alt+, Alt+ - przeniesienie linijki w której znajduje się kursor w górę/dół.
  • Tab/⌘+] - dodaj wcięcie (wcięcie w prawo)
  • Shit+Tab/⌘+[ - usunięcie wcięcia (wycięcie w lewo)

Dodawanie postów:

  • Ctrl+Enter - dodaj post
  • ⌘+Enter - dodaj post (MacOS)