C11dr 6.5.2.1 Array subscripting … «One of the expressions shall have type ‘‘pointer to complete object type’’, the other expression shall have integer type, and the result has type ‘‘type’’.» …
With []
, you get to do:
// pointer_type[integer_type]
float data[];
int n;
float total;
total = data[n];
// or the unconventional equivalent
// integer_type[pointer_type]
total = n[data]; // But let's leave that for another post
// Instead the OP did
// pointer_type[pointer_type]
// and received error: "Array subscript is not an integer"
float *p;
total = data[p];
The usage of float
is not the issue here, but the usage of a pointer.
An integer type includes types int
, unsigned
, size_t
, long
, etc.
I think the OP wanted the following (or something like this)
float deviation(float data[], int n) {
if (i <= 0) return 0;
float data_average = average(data, n);
float total = 0.0; // don't forget to set to 0.0
float *p = data;
while (p < (data + n)) {
total += (*p - data_average) * (*p - data_average);
p++;
}
return sqrt(total / n); // div by 0 averted
}
following this previous question Malloc Memory Corruption in C, now i have another problem.
I have the same code. Now I am trying to multiply the values contained in the arrays A * vc
and store in res. Then A is set to zero and i do a second multiplication with res and vc and i store the values in A. (A and Q are square matrices and mc and vc are N lines two columns matrices or arrays).
Here is my code :
int jacobi_gpu(double A[], double Q[],
double tol, long int dim){
int nrot, p, q, k, tid;
double c, s;
double *mc, *vc, *res;
int i,kc;
double vc1, vc2;
mc = (double *)malloc(2 * dim * sizeof(double));
vc = (double *)malloc(2 * dim * sizeof(double));
vc = (double *)malloc(dim * dim * sizeof(double));
if( mc == NULL || vc == NULL){
fprintf(stderr, "pb allocation matricre\n");
exit(1);
}
nrot = 0;
for(k = 0; k < dim - 1; k++){
eye(mc, dim);
eye(vc, dim);
for(tid = 0; tid < floor(dim /2); tid++){
p = (tid + k)%(dim - 1);
if(tid != 0)
q = (dim - tid + k - 1)%(dim - 1);
else
q = dim - 1;
printf("p = %d | q = %d\n", p, q);
if(fabs(A[p + q*dim]) > tol){
nrot++;
symschur2(A, dim, p, q, &c, &s);
mc[2*tid] = p; vc[2 * tid] = c;
mc[2*tid + 1] = q; vc[2*tid + 1] = -s;
mc[2*tid + 2*(dim - 2*tid) - 2] = p; vc[2*tid + 2*(dim - 2*tid) - 2 ] = s;
mc[2*tid + 2*(dim - 2*tid) - 1] = q; vc[2 * tid + 2*(dim - 2*tid) - 1 ] = c;
}
}
for( i = 0; i< dim; i++){
for(kc=0; kc < dim; kc++){
if( kc < floor(dim/2)) {
vc1 = vc[2*kc + i*dim];
vc2 = vc[2*kc + 2*(dim - 2*kc) - 2];
}else {
vc1 = vc[2*kc+1 + i*dim];
vc2 = vc[2*kc - 2*(dim - 2*kc) - 1];
}
res[kc + i*dim] = A[mc[2*kc] + i*dim]*vc1 + A[mc[2*kc + 1] + i*dim]*vc2;
}
}
zero(A, dim);
for( i = 0; i< dim; i++){
for(kc=0; kc < dim; k++){
if( k < floor(dim/2)){
vc1 = vc[2*kc + i*dim];
vc2 = vc[2*kc + 2*(dim - 2*kc) - 2];
}else {
vc1 = vc[2*kc+1 + i*dim];
vc2 = vc[2*kc - 2*(dim - 2*kc) - 1];
}
A[kc + i*dim] = res[mc[2*kc] + i*dim]*vc1 + res[mc[2*kc + 1] + i*dim]*vc2;
}
}
affiche(mc,dim,2,"Matrice creuse");
affiche(vc,dim,2,"Valeur creuse");
}
free(mc);
free(vc);
free(res);
return nrot;
}
When i try to compile, i have this error :
jacobi_gpu.c: In function ‘jacobi_gpu’:
jacobi_gpu.c:103: error: array subscript is not an integer
jacobi_gpu.c:103: error: array subscript is not an integer
jacobi_gpu.c:118: error: array subscript is not an integer
jacobi_gpu.c:118: error: array subscript is not an integer
make: *** [jacobi_gpu.o] Erreur 1
The corresponding lines are where I store the results in res and A :
res[kc + i*dim] = A[mc[2*kc] + i*dim]*vc1 + A[mc[2*kc + 1] + i*dim]*vc2;
and
A[kc + i*dim] = res[mc[2*kc] + i*dim]*vc1 + res[mc[2*kc + 1] + i*dim]*vc2;
Can someone explain me what is this error and how can i correct it?
Thanks for your help.
C11dr 6.5.2.1 Array subscripting … «One of the expressions shall have type ‘‘pointer to complete object type’’, the other expression shall have integer type, and the result has type ‘‘type’’.» …
With []
, you get to do:
// pointer_type[integer_type]
float data[];
int n;
float total;
total = data[n];
// or the unconventional equivalent
// integer_type[pointer_type]
total = n[data]; // But let's leave that for another post
// Instead the OP did
// pointer_type[pointer_type]
// and received error: "Array subscript is not an integer"
float *p;
total = data[p];
The usage of float
is not the issue here, but the usage of a pointer.
An integer type includes types int
, unsigned
, size_t
, long
, etc.
I think the OP wanted the following (or something like this)
float deviation(float data[], int n) {
if (i <= 0) return 0;
float data_average = average(data, n);
float total = 0.0; // don't forget to set to 0.0
float *p = data;
while (p < (data + n)) {
total += (*p - data_average) * (*p - data_average);
p++;
}
return sqrt(total / n); // div by 0 averted
}
-
11-06-2012
#1
Registered User
array subscript is not an integer
im getting the error when compiling the code…
Code:
float array[20]; int last; array[last[array]]; how to get red of this
-
11-06-2012
#2
SAMARAS
You have
Code:
array[last[array]];
In general,if you are sure you want something like this,imagine it with numbers and arrays.It would be something like this
if you want to access the index described by the last element of the array.
So maybe you want to say
-
11-06-2012
#3
Registered User
still the same error,i think may be its due to types of the integer variable and float arrays
-
11-06-2012
#4
SAMARAS
Have you initialized variable last? Can you please post the code you complied?
-
11-06-2012
#5
Registered User
im just checking the syntax errors
i have initialised int last = n-1;
int *l ;
l = &last;
-
11-06-2012
#6
SAMARAS
What is n?Where is the array you were talking about?
Just post all the code you have (i consider it is small from what you say )
Also it would be very kind of you if you could wrap your code in code tags.It is easy
[key]/*Your code here*/[/key] Replace key with code in order this to work.EDIT : it is ok to say
Code:
int a=5; int* p; p=&a;
Last edited by std10093; 11-06-2012 at 05:17 AM.
-
11-06-2012
#7
Registered User
Originally Posted by prathiksa
still the same error,i think may be its due to types of the integer variable and float arrays
Yes it is.
Arrays can only be indexed using values that have integral type (char, int, long, unsigned, etc). They cannot be indexed using non-integral values (float as in your case, struct types, etc).
There is also an obscure corner of C that causes «array[index]» and «index[array]» to be equivalent (assuming array is (well….) an array or a pointer, and index has integral type). Because of that, your expression «array[last[array]]» is equivalent to «array[array[last]]» which attempts to use array[last] as an index. In your code, array[last] is of type float so is not a valid index. Hence the compiler error.
Right 98% of the time, and don’t care about the other 3%.
If I seem grumpy or unhelpful in reply to you, or tell you you need to demonstrate more effort before you can expect help, it is likely you deserve it. Suck it up, Buttercup, and read this, this, and this before posting again.
-
11-06-2012
#8
SAMARAS
What i thought you wanted to do
Code:
#include <stdio.h> int main( int argc, char *argv[]) { float array[5]; int last = 4; array[0]=1.2; array[1]=1.3; array[2]=1.4; array[3]=1.5; array[4]=1.9; printf("%f\n",array[last]); return(0); }
grumpy is more experienced,so he is sure to have it better in his mind
-
11-06-2012
#9
Algorithm Dissector
You get rid of the error by fixing or removing the nonsense code.
Explain what you really want to do and we’ll show you how to write code for it that makes sense.My homepage
Advice: Take only as directed — If symptoms persist, please see your debuggerLinus Torvalds: «But it clearly is the only right way. The fact that everybody else does it some other way only means that they are wrong»
This topic has been deleted. Only users with topic management privileges can see it.
I’m newbie at C++ programming
I’m trying to write and read binary data. I can do this with no errors if I write a single number. But if I write an array then I can not correctly read that written data. For this two cases I have a picture attached to the post.
Could you explain me please what @0x2dfd78 in data variable means (see pictures in debug mode) and how to read and display array data in console?
#include <iostream>
#include <cstdio>
#include <string>
#include <vector>
#include <QDebug>
int main()
{
std::string str_file = "C:\\Users\\Tasik\\Documents\\Qt_prj\\proba.bin";
int n = str_file.length();
char char_file[100];
std::strcpy(char_file, str_file.c_str());
FILE *pFile;
pFile = fopen(char_file,"w+");
int data[5] = {1, 2, 3, 4, 5};
const int m = std::size(data);
fwrite(&data, sizeof(int), sizeof(data), pFile);
fclose (pFile);![alt text](image url)
pFile = fopen(char_file,"r");
int result;
fread(&result, sizeof(int), m, pFile);
std::cout << data << std::endl;
std::cout << result;
fclose (pFile);
return 0;
}
Hi
Here is a more c++ / Qt version
int main()
{
QVector<int> data = {1, 2, 3, 4, 5}; // using a container and not c array
QString filename{"e:/test.bin"}; // CHANGE ME! :)
QFile file(filename);
if (file.open(QFile::WriteOnly)) {
QDataStream out(&file);
out << data; // the << is an operator which you can define for you own types. QVector has one already
file.close();
}
data.clear(); // clear the array so we can see it works
if (file.open(QFile::ReadOnly)) {
QDataStream in(&file);
in >> data; // the >> is an operator which you can define for you own types. QVector has one already
}
qDebug() << data; // qDebug understand QVector and can just print it
return 0;
}
As you can hopefully see its far less low level than FILE interface.
Moreover, the << and >> we use to save the data can be used with all normal types
(int, float, etc) and also with Qt classes so you can just stream them with no extra code.
The << and >> handles the bytes sizes for you and all yo u then need to take care of is doing save and load in same
order or it fails.
It also allows for saving and loading object the exact same way if you give it the
QDataStream operators.
Like:
class budget
{
float transportation, grocery, food, stationery;
QString key;
public:
budget() {}
friend QDataStream &operator <<(QDataStream &stream, const budget &myclass) {
stream<< myclass.food;
stream<< myclass.grocery;
stream<< myclass.key;
stream<< myclass.stationery;
stream<< myclass.transportation;
return stream;
}
friend QDataStream &operator >>(QDataStream &stream, budget &myclass) {
stream >> myclass.food;
stream >> myclass.grocery;
stream >> myclass.key;
stream >> myclass.stationery;
stream >> myclass.transportation;
return stream;
}
this allows you to stream your own class 100% like vector
budget myBuget;
out << myBuget;
Overall this makes the program easier to read and also far less error prone than handling it at byte level with stuff like
fwrite(&data, sizeof(int), sizeof(data), pFile);
I hope i sold it to you
Hi
The @0x2dfd78 is the address of the first element in the c array. its a standard that c arrays
are internally a pointer to first element and hence the debugger show the starting address.
and then the elements.
Next thing to note is that you are using the FILE interface. Its a bit oldscool and normally iostreams are used for c++ programs.
Anyway, the reason it not working for the array is that you read it back into one int and not the array
int result; // this is your buffer. but file has 5 ints
fread(&result, sizeof(int), m, pFile);
What you mean is actually
fread(&data, sizeof(int), m, pFile); // use the data array as target buffer
To print it to console, you need to take each index (or use for loop)
std::cout << data[0] << std::endl;
as else it sees the pointer and just writes the address. (of the first element)
It does not know its an array or its size.
Other note is. Since you are using c++ and Qt, why not use the features Qt provides to save the data ?
Its ok if you want to use pure c++ but Qt does offer many benefits for saving say QStrings and other Qt data types.
Hi
Here is a more c++ / Qt version
int main()
{
QVector<int> data = {1, 2, 3, 4, 5}; // using a container and not c array
QString filename{"e:/test.bin"}; // CHANGE ME! :)
QFile file(filename);
if (file.open(QFile::WriteOnly)) {
QDataStream out(&file);
out << data; // the << is an operator which you can define for you own types. QVector has one already
file.close();
}
data.clear(); // clear the array so we can see it works
if (file.open(QFile::ReadOnly)) {
QDataStream in(&file);
in >> data; // the >> is an operator which you can define for you own types. QVector has one already
}
qDebug() << data; // qDebug understand QVector and can just print it
return 0;
}
As you can hopefully see its far less low level than FILE interface.
Moreover, the << and >> we use to save the data can be used with all normal types
(int, float, etc) and also with Qt classes so you can just stream them with no extra code.
The << and >> handles the bytes sizes for you and all yo u then need to take care of is doing save and load in same
order or it fails.
It also allows for saving and loading object the exact same way if you give it the
QDataStream operators.
Like:
class budget
{
float transportation, grocery, food, stationery;
QString key;
public:
budget() {}
friend QDataStream &operator <<(QDataStream &stream, const budget &myclass) {
stream<< myclass.food;
stream<< myclass.grocery;
stream<< myclass.key;
stream<< myclass.stationery;
stream<< myclass.transportation;
return stream;
}
friend QDataStream &operator >>(QDataStream &stream, budget &myclass) {
stream >> myclass.food;
stream >> myclass.grocery;
stream >> myclass.key;
stream >> myclass.stationery;
stream >> myclass.transportation;
return stream;
}
this allows you to stream your own class 100% like vector
budget myBuget;
out << myBuget;
Overall this makes the program easier to read and also far less error prone than handling it at byte level with stuff like
fwrite(&data, sizeof(int), sizeof(data), pFile);
I hope i sold it to you
@mrjj ahaha not sold yet but very thanks for such detailed answer
Tomorrow I will try your code (it is already deep night in Saint-Petersburg).
The main questions that affect on my decision is the speed of read/write binary files of weigh about N*10 Gigabytes and the possibility to use threads… How do you think should I use Qt interface or low level functions (such as fwrite/fread?)?
@Please_Help_me_D
i/o is always slow, so using Qt classes should not hurt much. however, holding such large datasets in memory efficiently can be complicated.
Can you elaborate a bit more about your problem?
Regards
@Please_Help_me_D
I’ll throw in a couple of (very slightly) controversial suggestions, just for your consideration, since you are talking about such large levels of I/O.
-
The Qt, C++ and even C stdio I/O functions like
fread
/fwrite()
use an underlying extra buffer level between the disk data and your code access. If performance is critical, and if what you are doing is very simple (e.g. just sequential access), lower level functions are namedread()
&write()
(or, depending on OS/compiler,_read
/_write()
). Further OS-specific calls are also available for asynchronous I/O, which again might improve your particular situation. You would have to time on platforms to see how much difference these make. -
I must admit I have never used this, though I have often wanted to: there is the
mmap()
(#include <sys/mman.h>
) family of calls. This «maps» (areas of) disk files directly into memory, so you do not actually do any I/O, the data appears and is accessed just like an array of bytes in memory. Again, you would have to time.
I don’t know what the experts here think about my two points.
Also, be aware that reading disk I/O is also a lot faster than writing it, especially (as I understand it) if you have an SSD, though I guess that would not apply to your large data, but again you could check timings.
First thing is to address @aha_1980’s request for more information on what you are trying to achieve.
@Please_Help_me_D
Hi
The overhead from DataStream etc will be very minor if just streaming a big 10 GB memory block the same way you would with
the FILE interface.
So its hard to suggest what will be the best solution before we know how you have the data structured etc.
Also what is the data ? 10 GB is massive
@mrjj I tried the example you gave with QVector and QFile and I liked it.
@aha_1980 @JonB So I try to describe a little more the problem.
I know Matlab pretty good and I started to learn C++ and Qt to write a program that performs mathematical operations on the data. So I’m going to read raw binary data file and store it in scientific HDF5 format (primary as a 2-dimensional array). Those files may weigh from N100 Megabytes till N100 Gigabytes. So when I read data and store in HDF5 format then I need to have access only to portions of that data. I never need to upload all the data in RAM at the same time.
In Matlab I worked with memory mapping (memmap function) technique but now I want to use HDF5 format wich is able to replace the need to use memory mapping. I’m afraid that on windows there is some difficulties with mmap.
Is in there a way in Qt to generate a sequance of numbers without loops? For example if I want an array with numbers {1, 2, 3, 4, 5} I write:
int data = {1, 2, 3, 4, 5};
But if I want to generate integer numbers from n to N with step dn I would get:
int data = {n, n+dn, n+2*dn, ... , N};
And also is there a way to get access to several elements of an array. For example:
int data = {1, 2, 3, 4, 5};
I how can get 2, 3 and 4th elements of data without loop?
Hi
Good to hear.
ok so its HDF5 format
Do note there exits libraries to use that format from c++.
But if they provide benefits for your use case or not is hard to say.
But since you want to read in the data, you will need to use mem mapped files or similar
and you might be able to get that out of the box with a library.
But if I want to generate integer numbers from n to N with step dn I would get:
int data = {n, n+dn, n+2*dn, … , N};
Hmm. Nothing really springs to mind. Why are you against a loop ?
And also is there a way to get access to several elements of an array. For example:
int data = {1, 2, 3, 4, 5};
I how can get 2, 3 and 4th elements of data without loop?
data[index] gives access. If you need to modify the value then
int &val = data[index];
val = 100; // will change the table value to 100
Do note that QFile also supports mem mapped files.
(QFile::map )
@mrjj Hi))
I installed official libraries from HDFGroup with Cpp libraries checked on while do CMake. And there also HDF5 cpp project but I don’t undestand what this project do. Is it just provide simple interface to use HDF5…
But since you want to read in the data, you will need to use mem mapped files or similar
and you might be able to get that out of the box with a library.
Didn’t undestand that… Do you mean that HDF5 libraries uses memory mapping or do I need to read big data with mem map? I read this staff HDF5 or memory mapping and since HDF5 is well known and actively used I decided to use HDF5 instead of memory mapping.
int data = {n, n+dn, n+2*dn, … , N};
Hmm. Nothing really springs to mind. Why are you against a loop ?
Well I’m from Matlab and it taught me to avoid loops (because it is slow in Matlab) and I’m slightly uncofortable now when I use loops in case I could avoid it
I need to get known with memory mapping in Qt a little better. Do you know is Qt memory mapping works on Windows? Because a week ago I was trying to install MPICH (for cluster computation, just to try) and I could not because of some error connected with lack of memap on Windows or something…
@Please_Help_me_D
Hi
- Didn’t understand that..
I meant that using whatever HDF5 uses to allow reading those large files might just work out of the box and then maybe no need for your own memmap file or similar. Was just saying you need something extra to drive such large files and it seems HDF5 does give that via its chunked file design.
(loops)
Ahh, That way. Well, there is a thing with loops in c++/Qt.
If you fill very large array in main thread, it will lag your program’s interface.
But besides that, loops are fast in c++. (generally speaking)
(Qfile map)
QFile map function should also work in window as far as i know.
Windows does support it natively and i think Qfile map uses that.
The https://github.com/ess-dmsc/h5cpp
provides a c++ wrapper for a c library.
This is often done to allow for object orientated programming with the
c library and maybe hide details behind more easy to use classes than raw C code.
you dont need to use the wrapper if you feel good with c code.
However,
I dont have experience with HDF5 format but looking over the docs, it really seems the way to go as it should provide you with anything you need to make a c++ program that can consume and produce such giga files.
and the c api dont really look that bad
https://support.hdfgroup.org/ftp/HDF5/examples/examples-by-api/hdf5-examples/1_10/C/H5D/h5ex_d_chunk.c
@mrjj Thank you for information
I am working on I/O with information you provided
@mrjj one more question. I get an error: array subscript is not an integer:
int data[5] = {1, 2, 3, 4, 5};
int ind[3] = {0, 1, 2};
int data2 = data[ind]; // here is that error . **ind** is highlighted by red
I declare ind as integer but still can’t get access to those elements of an array…
And here is similar problem expression is not determined by a constant. Failure caused by reading a variable beyond its lifetime:
std::string str_file = "C:\\Users\\Tasik\\Documents\\Qt_prj\\proba.bin";
int n = str_file.length();
char char_file[n]; // here is that error. It appears only when I launch the application
Hi,
ind
is not an integer, it’s an array of 3 integers.
Depending on your compiler you will have to allocate your char_file array on the heap using new and then delete when done with that array.
@SGaist Hello
Thank you for answer
So is there a way to extract few elements from an array at the same time without loop?
Depending on your compiler you will have to allocate your char_file array on the heap using new and then delete when done with that array.
My compiler is MSVC 2017. Could you write an example of this?
@Please_Help_me_D said in General question about arrays and I/o:
So is there a way to extract few elements from an array at the same time without loop?
Not with plain C arrays.
But you can do this with QVector: https://doc.qt.io/qt-5/qvector.html#mid
There is something you can do without copying anything: an array is just a pointer to first element, so:
int data[5] = {1, 2, 3, 4, 5};
int *data2 = &data[2]; // data2 is now [3, 4, 5].
Do you really need to copy to data2? You can simply have a variable «length» containing the length of the sub-array in data.
«My compiler is MSVC 2017. Could you write an example of this?»:
char *char_file = new char[n]; // Allocate on the heap
...
delete[] char_file; // Delete when not needed anymore
@jsulm thank you for the answer
The problem is that usually I have I know indexes are maybe like:
int data[4] = {1, 2, 3, 4, 5};
int ind[3] = {4, 2, 3};
and then I need to get access to those elements like:
int data2[3] = data[ind];
Now I read about QVector, I hope it is able to do that.
You know both examples that you wrote seems to me don’t work properly.
int data[5] = {1, 2, 3, 4, 5};
int *data2 = &data[2]; // data2 is now [3].
And:
int n = 5;
char char_file = new char[n]; // error: cannot initialize a variable of type 'char' with an rvalue of type 'char *'
delete[] char_file; // Delete when not needed anymore
Where can I read about ‘*’ and ‘&’ signs when using in such ways? What it gives?
@Please_Help_me_D said in General question about arrays and I/o:
seems to me don’t work properly
In what way? &data[2] points to 3 in data, so data2[0] == 3, data2[1] == 4 and data2[2] == 5
Please read about pointers in C/C++:
// It must be *char_file not just char_file
char *char_file = new char[n];
I edited my previous post as I forgot *
@jsulm
In what way? &data[2] points to 3 in data, so data2[0] == 3, data2[1] == 4 and data2[2] == 5
I attach the picture below. data2 is now is equal to 3 and that is it. Is it correct?
After I added * pointer the program works but seems to me that the length of char_file doesn’t depend on n. If n=4 then length of char_file=32, n=5 then char_file=32. Is it ok?
@Please_Help_me_D said in General question about arrays and I/o:
data2 is now is equal to 3 and that is it. Is it correct?
Yes it is, you can treat a pointer as an array (actually in C/C++ an array is simply a pointer to first element of the array). So, data2[0] == 3, data[1] == 4…
Just do
qDebug() << data2[1];
and see.
Regarding second question: this is debugger view. Your array is for sure 4 char in size. To verify do
char_file[4] = 1;
your app should crash.