| Autor |
Nachricht |
se1111
Mitglied
Benutzerprofil
Anmeldungsdatum: 20.02.2010
Beiträge: 7
|
se1111 Mitglied
17:19:02 10.03.2010 Titel: |
map mit arrays ? |
Zitieren |
Hallo Leute,
Folgendes Problem, bei dem ich nicht weiterkomme.
Für n Identitäten (id) möchte ich 2 Arrays gleicher Art und Größe haben.
Die Relation Identität zu Objekt (hier das Array) erfolgt einfacherweise in einer Map.
Um die map am Anfang zu definieren, wollte ich es so etwa machen:
| C/C++ Code: | 1 2 3 4 5 6 7 8 | 1 2 3 4 5 6 7 8 | for (id=0 ; id<MAX ; id++)
{
int array_a[3];
int array_b[3];
array_a_map.insert(make_pair(id, &array_a));
array_b_map.insert(make_pair(id, &array_b));
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 | for (id=0 ; id<MAX ; id++)
{
int array_a[3];
int array_b[3];
array_a_map.insert(make_pair(id, &array_a));
array_b_map.insert(make_pair(id, &array_b));
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 | for (id=0 ; id<MAX ; id++)
{
int array_a[3];
int array_b[3];
array_a_map.insert(make_pair(id, &array_a));
array_b_map.insert(make_pair(id, &array_b));
}
| |
Später könnte ich mir die Pointer zurückgeben lassen, die immer auf das erste Element der Arrays zeigen und damit dann arbeiten.
Jedoch wird die Inititlisierung so wohl nicht funktionieren, da ich die Array Adressen in der Map speichere. Bei nur einer id würde es funktionieren, beim weiteren for-Durchlauf wird doch wieder dieselbe Adresse geschrieben oder? Wäre also Mist.
Wie könnte man es sonst machen? Map von Map vielleicht? Die zweite Map wäre dann das Array. |
|
|
|
 |
DocShoe
Mitglied
Benutzerprofil
Anmeldungsdatum: 02.04.2008
Beiträge: 1608
|
DocShoe Mitglied
17:24:40 10.03.2010 Titel: |
|
Zitieren |
Ich kann dir nicht ganz folgen, fürchte ich. Abgesehen davon speicherst du Adressen eines Arrays, das nach Verlassen der for-Schleife überhaupt nicht mehr existiert. Was genau hast du denn vor? |
|
|
|
 |
se1111
Mitglied
Benutzerprofil
Anmeldungsdatum: 20.02.2010
Beiträge: 7
|
se1111 Mitglied
17:34:20 10.03.2010 Titel: |
|
Zitieren |
stimmt, die Arrays müssten natürlich global sein um dauerhaft zu existieren.
Allerdings müssen sie sich ja irgendwie bei jedem Id Durchlauf unterscheiden lassen, um sie in der Map eindeutig zuzuweisen.
Den Typ der Map wollte ich so setzen:
typedef std::map<int, int*> ArrayMapType
Ich sehe aber nicht wie ich sie unterscheiden sollte.
Besser das Ganze über eine
std::map<int, map<int, int>>
machen !? |
Zuletzt bearbeitet von se1111 am 17:35:56 10.03.2010, insgesamt 3-mal bearbeitet |
|
 |
DocShoe
Mitglied
Benutzerprofil
Anmeldungsdatum: 02.04.2008
Beiträge: 1608
|
DocShoe Mitglied
17:44:39 10.03.2010 Titel: |
|
Zitieren |
Beschreib´ doch bitte mal ganz genau das Gesamtvorhaben. Man kann bestenfalls vermuten, was du vorhast, zumindest ich habe keine Ahnung, welche Adressen du wann wie wiederfinden willst. |
|
|
|
 |
asc
Mitglied
Benutzerprofil
Anmeldungsdatum: 13.01.2007
Beiträge: 5278
|
asc Mitglied
17:57:11 10.03.2010 Titel: |
Re: map mit arrays ? |
Zitieren |
| se1111 schrieb: | | Wie könnte man es sonst machen? Map von Map vielleicht? Die zweite Map wäre dann das Array. |
Ich würde es (wenn ich den dein Problem richtig verstehe) wohl auf eine der folgenden Weisen umsetzen:
std::map<int, std::vector<int> >
std::map<int, std::tr1::array<int, 3> >
Ein Array durch eine map zu ersetzen sehe ich als falsch an, da würde ich auf entweder einen vector oder dem array aus dem TR1 zurückgreifen, da diese einem Array noch am meisten ähneln. |
_________________ in theory there's no difference between theory and practice. in practice there is. (yogi berra)
In der Theorie gibt es kein Unterschied zwischen Theorie und Praxis. In der Praxis sehr wohl.
|
|
 |
se1111
Mitglied
Benutzerprofil
Anmeldungsdatum: 20.02.2010
Beiträge: 7
|
se1111 Mitglied
18:09:49 10.03.2010 Titel: |
|
Zitieren |
OK hier noch etwas genauer:
Ich habe diese zwei Arrays:
int array_a[3];
int array_b[3];
Normalerweise schreibe und lese ich direkt mit z.B.
array_a[0] = 0x10;
Da ich nun n verschiedene Kontexte habe (Identitäten genannt), kann in jeder Identität der Inhalt der Arrays anders sein. Und sie gibt es dann auch n-mal (für jede Id).
Ich muss also über eine Id<->Array Relation drauf zugreifen. Ich denke eine map ist dafür geeignet. Der Index ist die Id, der Content ist ein "Pointer auf Array".
Insgesamt habe ich also auch 2 maps, "array_a_map" und "array_b_map".
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | typedef std::map<int, int*> ArrayMapType;
// Funktion die mir den Pointer zurückgibt:
int * GetPtr2ArrayfromIdMap(int id, ArrayMapType map_name)
{
ArrayMapType::iterator iter = map_name.begin();
iter = map_name.find(id);
if( iter != map_name.end() )
return iter->second;
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | typedef std::map<int, int*> ArrayMapType;
// Funktion die mir den Pointer zurückgibt:
int * GetPtr2ArrayfromIdMap(int id, ArrayMapType map_name)
{
ArrayMapType::iterator iter = map_name.begin();
iter = map_name.find(id);
if( iter != map_name.end() )
return iter->second;
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | typedef std::map<int, int*> ArrayMapType;
// Funktion die mir den Pointer zurückgibt:
int * GetPtr2ArrayfromIdMap(int id, ArrayMapType map_name)
{
ArrayMapType::iterator iter = map_name.begin();
iter = map_name.find(id);
if( iter != map_name.end() )
return iter->second;
}
| |
| C/C++ Code: | // Array Zugriff (Lesen):
int * tmpArrayPtr = GetPtr2ArrayfromIdMap(id, array_a_map); // get Ptr to Array
int tmp__0 = *(tmpArrayPtr); // points to tmpArrayPtr[0]
int tmp__1 = *(tmpArrayPtr+1); // points to tmpArrayPtr[1]
| |
| C/C++ Code: | // Array Zugriff (Lesen):
int * tmpArrayPtr = GetPtr2ArrayfromIdMap(id, array_a_map); // get Ptr to Array
int tmp__0 = *(tmpArrayPtr); // points to tmpArrayPtr[0]
int tmp__1 = *(tmpArrayPtr+1); // points to tmpArrayPtr[1]
| |
| C/C++ Code: | // Array Zugriff (Lesen):
int * tmpArrayPtr = GetPtr2ArrayfromIdMap(id, array_a_map); // get Ptr to Array
int tmp__0 = *(tmpArrayPtr); // points to tmpArrayPtr[0]
int tmp__1 = *(tmpArrayPtr+1); // points to tmpArrayPtr[1]
| |
Die Frage ist nun, wie ich initial die map aufsetze. Ich müsste für jede map und für jede Id ein 3 dimensionales Array reservieren. Dafür brauche ich die jeweiligen Adressen der Arrays, und
diese müssen bei jedem for-Durchlauf (siehe oben) eindeutig sein. Ich muss also die Arrays unterscheiden können. Da sie aber immer gleich heißen, sind sie auch immer dieselben Objekte. |
|
|
|
 |
asc
Mitglied
Benutzerprofil
Anmeldungsdatum: 13.01.2007
Beiträge: 5278
|
asc Mitglied
18:50:15 10.03.2010 Titel: |
|
Zitieren |
| se1111 schrieb: | | OK hier noch etwas genauer: |
Ehrlich gesagt bin ich von deiner Erläuterung eher verwirrt als was anderes.
Wenn die Arrays pro Id einen eigenen Inhalt haben müssen, ist meine Variante die richtige (Ansonsten müsstest du die Arrays dynamisch allozieren oder n-mal global definieren, wobei sich n nicht zur Laufzeit unterscheiden kann).
Meine Variante:
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <array> // tr1-Header
#include <map>
int main()
{
std::map<int, std::tr1::array<int, 3> > identityArray;
// Beispiel für den schreibenden Zugriff auf ein Arrayelement zur Id 4711
identityAArray[4711][0] = 41;
// \Id/ |
// |
// Arrayposition
// Beispiel für den lesenden Zugriff auf ein Arrayelement zur Id 4711
std::cout << identityAArray[4711][0];
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <array> // tr1-Header
#include <map>
int main()
{
std::map<int, std::tr1::array<int, 3> > identityArray;
// Beispiel für den schreibenden Zugriff auf ein Arrayelement zur Id 4711
identityAArray[4711][0] = 41;
// \Id/ |
// |
// Arrayposition
// Beispiel für den lesenden Zugriff auf ein Arrayelement zur Id 4711
std::cout << identityAArray[4711][0];
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <array> // tr1-Header
#include <map>
int main()
{
std::map<int, std::tr1::array<int, 3> > identityArray;
// Beispiel für den schreibenden Zugriff auf ein Arrayelement zur Id 4711
identityAArray[4711][0] = 41;
// \Id/ |
// |
// Arrayposition
// Beispiel für den lesenden Zugriff auf ein Arrayelement zur Id 4711
std::cout << identityAArray[4711][0];
}
| |
| Code: | 1 2 3 4 5 6 7 8 | 1 2 3 4 5 6 7 8 | Was passiert beim Indexzugriff auf diese Map:
a) Existiert zu dem Key schon ein Element?
=> ja siehe c
=> nein siehe b
b) Ein neues array<int, 3> (Int-Array mit 3 defaultinitialisierten [bei int: 0]
Einträgen) wird angelegt und in die Map zum Key abgelegt.
=> weiter mit c
c) Rückgabe einer Referenz auf den Inhalt (Hier dem Array).
| |
| Code: | 1 2 3 4 5 6 7 8 | Was passiert beim Indexzugriff auf diese Map:
a) Existiert zu dem Key schon ein Element?
=> ja siehe c
=> nein siehe b
b) Ein neues array<int, 3> (Int-Array mit 3 defaultinitialisierten [bei int: 0]
Einträgen) wird angelegt und in die Map zum Key abgelegt.
=> weiter mit c
c) Rückgabe einer Referenz auf den Inhalt (Hier dem Array).
| |
| Code: | 1 2 3 4 5 6 7 8 | Was passiert beim Indexzugriff auf diese Map:
a) Existiert zu dem Key schon ein Element?
=> ja siehe c
=> nein siehe b
b) Ein neues array<int, 3> (Int-Array mit 3 defaultinitialisierten [bei int: 0]
Einträgen) wird angelegt und in die Map zum Key abgelegt.
=> weiter mit c
c) Rückgabe einer Referenz auf den Inhalt (Hier dem Array).
| |
Falls das nicht gewünscht ist: Abfrage mit find ob das Element existiert, und Zugriff per Iteratoren. |
_________________ in theory there's no difference between theory and practice. in practice there is. (yogi berra)
In der Theorie gibt es kein Unterschied zwischen Theorie und Praxis. In der Praxis sehr wohl.
Zuletzt bearbeitet von asc am 18:51:33 10.03.2010, insgesamt 2-mal bearbeitet |
|
 |
se1111
Mitglied
Benutzerprofil
Anmeldungsdatum: 20.02.2010
Beiträge: 7
|
se1111 Mitglied
10:35:53 11.03.2010 Titel: |
|
Zitieren |
Danke, werde ich mal versuchen. Allerdings habe ich hier VC++ 6.0 und ob es da die TR1 gibt, bin ich mir grad nicht sicher. |
|
|
|
 |
Michael E.
Mitglied
Benutzerprofil
Anmeldungsdatum: 25.10.2003
Beiträge: 5323
|
Michael E. Mitglied
11:50:04 11.03.2010 Titel: |
|
Zitieren |
| se1111 schrieb: | | Allerdings habe ich hier VC++ 6.0 |
Warum? |
_________________ Your password must be at least 18770 characters and cannot repeat any of your previous 30689 passwords. Please type a different password. Type a password that meets these requirements in both text boxes. (http://support.microsoft.com/kb/276304/en-us/)
|
|
 |
CptC
Unregistrierter
|
CptC Unregistrierter
12:12:53 11.03.2010 Titel: |
|
Zitieren |
| se1111 schrieb: | | Danke, werde ich mal versuchen. Allerdings habe ich hier VC++ 6.0 und ob es da die TR1 gibt, bin ich mir grad nicht sicher. |
VC++6? Ne, der hat kein TR1. Das hat ja noch nichmal richtiges C++.
Mit anderen Worten: 1997 hat angerufen und will seinen Compiler zurück . |
|
|
|
 |
asc
Mitglied
Benutzerprofil
Anmeldungsdatum: 13.01.2007
Beiträge: 5278
|
asc Mitglied
12:31:21 11.03.2010 Titel: |
|
Zitieren |
| se1111 schrieb: | | Allerdings habe ich hier VC++ 6.0 und ob es da die TR1 gibt, bin ich mir grad nicht sicher. |
Nein, gibt es nicht. Dann ist der Weg etwas umständlicher, aber mit std::vector dennoch machbar.
Dennoch die schon hier gestellte Frage: Warum VC++ 6.0? Sofern dir dies nicht durch eine Firma vorgegeben wurde rate ich dir ganz dringend zu einem Wechsel auf eine aktuelle Plattform (z.B. Visual C++ 2008 Express).
Der VC++ 6.0 hat etliche Probleme mit dem C++ Standard (logisch, der C++ Standard kam etwas danach raus).
Aber um auf die Alternativen zurück zu kommen:
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <map>
#include <vector>
int main()
{
std::map<int, std::vector<int> > identityArray;
// Beim ersten Zugriff auf Id den vector für drei Elemente initialisieren:
identityArray[4711] = std::vector<int>(3);
// Rest läuft identisch, nur muss sichergestellt sein, das vor dem Zugriff
// auf den jeweiligen vector über den Indexoperator dieser schon drei
// Elemente enthält (Ein Standardkonstruierter vector enthält anfangs 0
// Elemente).
...
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <map>
#include <vector>
int main()
{
std::map<int, std::vector<int> > identityArray;
// Beim ersten Zugriff auf Id den vector für drei Elemente initialisieren:
identityArray[4711] = std::vector<int>(3);
// Rest läuft identisch, nur muss sichergestellt sein, das vor dem Zugriff
// auf den jeweiligen vector über den Indexoperator dieser schon drei
// Elemente enthält (Ein Standardkonstruierter vector enthält anfangs 0
// Elemente).
...
}
| |
| C/C++ Code: | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <map>
#include <vector>
int main()
{
std::map<int, std::vector<int> > identityArray;
// Beim ersten Zugriff auf Id den vector für drei Elemente initialisieren:
identityArray[4711] = std::vector<int>(3);
// Rest läuft identisch, nur muss sichergestellt sein, das vor dem Zugriff
// auf den jeweiligen vector über den Indexoperator dieser schon drei
// Elemente enthält (Ein Standardkonstruierter vector enthält anfangs 0
// Elemente).
...
}
| |
|
_________________ in theory there's no difference between theory and practice. in practice there is. (yogi berra)
In der Theorie gibt es kein Unterschied zwischen Theorie und Praxis. In der Praxis sehr wohl.
|
|
 |
se1111
Mitglied
Benutzerprofil
Anmeldungsdatum: 20.02.2010
Beiträge: 7
|
se1111 Mitglied
12:22:17 12.03.2010 Titel: |
|
Zitieren |
Genau ich muss ihm die Sachen reservieren. Das war das Problem wo ich nicht sicher war.
| asc schrieb: |
// Beim ersten Zugriff auf Id den vector für drei Elemente initialisieren:
identityArray[4711] = std::vector<int>(3);
|
Ich hoff ich verstehe es richtig, dann müsste ich das ja so schreiben:
| C/C++ Code: | for (..Identity.. )
{
array_a_map.insert(make_pair(Identity, std::vector<int>(3)));
}
| |
| C/C++ Code: | for (..Identity.. )
{
array_a_map.insert(make_pair(Identity, std::vector<int>(3)));
}
| |
| C/C++ Code: | for (..Identity.. )
{
array_a_map.insert(make_pair(Identity, std::vector<int>(3)));
}
| |
|
Zuletzt bearbeitet von se1111 am 12:22:38 12.03.2010, insgesamt 1-mal bearbeitet |
|
 |
asc
Mitglied
Benutzerprofil
Anmeldungsdatum: 13.01.2007
Beiträge: 5278
|
asc Mitglied
14:03:14 12.03.2010 Titel: |
|
Zitieren |
| se1111 schrieb: | | Genau ich muss ihm die Sachen reservieren. |
Nein, DU musst die Sachen nicht zwangsweise reservieren. Zumindest wenn du mit Objekten arbeitest, die eben diese Aufgabe für dich übernehmen. Und ein Standardkonstruiertes std::tr1::array<int, 3> wäre für dich als Mapelement eigentlich genau richtig (Ist aber nicht extrem veralteten VC++ 6.0 enthalten).
Die Map stellt bereits eine Initialisierung mit dem Standardkonstruktor sicher. Sofern du den Indexoperator verwendest, ist Sichergestellt das ein entsprechendes Objekt bereits existiert. Nur muss man 3 Fälle unterscheiden:
a) Ein Standardkonstruierter Zeiger bringt dir hier keinen Nutzen.
b) Ein Standardkonstruiertes std::tr1::array<int, 3> ist eine Arrayklasse mit 3 int-Werten (Im wesentlichen ein Wrapper um ein C-Array).
c) Ein Standardkonstruiertes std::vector ist ein vector mit 0 Elementen. Da du 3 benötigst musst du selbst sicherstellen das die beim lesenden Zugriff auch enthalten sind. Dies geht entweder durch eine Zuweisung mit einem entsprechenden Vector, oder ein resize...
| se1111 schrieb: | Ich hoff ich verstehe es richtig, dann müsste ich das ja so schreiben:
| C/C++ Code: | for (..Identity.. )
{
array_a_map.insert(make_pair(Identity, std::vector<int>(3)));
}
| |
| C/C++ Code: | for (..Identity.. )
{
array_a_map.insert(make_pair(Identity, std::vector<int>(3)));
}
| |
| C/C++ Code: | for (..Identity.. )
{
array_a_map.insert(make_pair(Identity, std::vector<int>(3)));
}
| |
|
Kürzer Wäre:
| C/C++ Code: | for (..Identity.. )
{
array_a_map[Identity] = std::vector<int>(3);
// ODER
array_a_map[Identity].resize(3);
}
| |
| C/C++ Code: | for (..Identity.. )
{
array_a_map[Identity] = std::vector<int>(3);
// ODER
array_a_map[Identity].resize(3);
}
| |
| C/C++ Code: | for (..Identity.. )
{
array_a_map[Identity] = std::vector<int>(3);
// ODER
array_a_map[Identity].resize(3);
}
| |
Bei verwendung eines neuen Compilers mit der TR1 Unterstützung kannst du dir diesen Schritt gänzlich schenken (wie gesagt std::tr1::array). |
_________________ in theory there's no difference between theory and practice. in practice there is. (yogi berra)
In der Theorie gibt es kein Unterschied zwischen Theorie und Praxis. In der Praxis sehr wohl.
|
|
 |
asc
Mitglied
Benutzerprofil
Anmeldungsdatum: 13.01.2007
Beiträge: 5278
|
asc Mitglied
14:05:27 12.03.2010 Titel: |
|
Zitieren |
Vielleicht noch ein letztes: In C++ ist es nicht unüblich das man die Speicherverwaltung Klassen überlässt die es intern behandeln (z.B. die Containerklassen der STL). |
_________________ in theory there's no difference between theory and practice. in practice there is. (yogi berra)
In der Theorie gibt es kein Unterschied zwischen Theorie und Praxis. In der Praxis sehr wohl.
|
|
 |