Значение * & и ** & в C ++ - PullRequest
       22

Значение * & и ** & в C ++

49 голосов
/ 26 апреля 2011

Я обнаружил эти символы в объявлении функции несколько раз, но я не знаю, что они означают.

Пример:

void raccogli_dati(double **& V, double **p, int N) { 
  int ultimo = 3; 
  V = new double * [N/2]; 
  for(int i=0; i < N/2; i++) { 
    V[i] = new double[N/2], std :: clog << "digita " << N/2 - i
                 << " valori per la parte superiore della matrice V: "; 
    for(int j=i; j < N/2; j++) 
      std :: cin >> V[i][j], p[ultimo++][0] = (V[i][j] /= sqrt(p[i][0]*p[j][0]));
  } 
  for(int i=1; i < N/2; i++) 
    for(int j=0; j < i; j++) 
       V[i][j] = V[j][i];
}

Ответы [ 6 ]

77 голосов
/ 26 апреля 2011

То есть, принимая параметр по ссылке. Таким образом, в первом случае вы берете параметр указателя по ссылке, поэтому любое изменение значения указателя отражается вне функции. Второе аналогично первому, с той лишь разницей, что это двойной указатель. Смотрите этот пример:

void pass_by_value(int* p)
{
    //Allocate memory for int and store the address in p
    p = new int;
}

void pass_by_reference(int*& p)
{
    p = new int;
}

int main()
{
    int* p1 = NULL;
    int* p2 = NULL;

    pass_by_value(p1); //p1 will still be NULL after this call
    pass_by_reference(p2); //p2 's value is changed to point to the newly allocate memory

    return 0;
}
14 голосов
/ 26 апреля 2011

Первый - это ссылка на указатель, второй - это ссылка на указатель на указатель.См. Также FAQ по , как отличаются указатели и ссылки .

void foo(int*& x, int**& y) {
    // modifying x or y here will modify a or b in main
}

int main() {
    int val = 42;
    int *a  = &val;
    int **b = &a;

    foo(a, b);
    return 0;
}
11 голосов
/ 26 апреля 2011

Передача указателя по ссылке, а не по значению. Это, например, позволяет изменять указатель (а не объект, на который указывает указатель) в функции таким образом, что вызывающий код видит изменение.

Сравнить:

void nochange( int* pointer ) //passed by value
{
   pointer++; // change will be discarded once function returns
}

void change( int*& pointer ) //passed by reference
{
   pointer++; // change will persist when function returns
}
5 голосов
/ 26 апреля 2011

int* - это указатель на int, поэтому int*& должен быть ссылкой на указатель на int.Аналогично, int** - это указатель на указатель на int, поэтому int**& должен быть ссылкой на указатель на указатель на int.

3 голосов
/ 26 апреля 2011

*& означает получение указателя по ссылке. Это означает, что это псевдоним для передаваемого параметра. Таким образом, это влияет на передаваемый параметр.

#include <iostream>
using namespace std;

void foo(int *ptr)
{
    ptr = new int(50);    // Modifying the pointer to point to a different location
    cout << "In foo:\t" << *ptr << "\n"; 
    delete ptr ;
}

void bar(int *& ptr)
{
    ptr = new int(80);    // Modifying the pointer to point to a different location
    cout << "In bar:\t" << *ptr << "\n"; 
    // Deleting the pointer will result the actual passed parameter dangling
}
int main()
{
    int temp = 100 ;
    int *p = &temp ;

    cout << "Before foo:\t" << *p << "\n";
    foo(p) ;
    cout << "After foo:\t" << *p << "\n";

    cout << "Before bar:\t" << *p << "\n";
    bar(p) ;
    cout << "After bar:\t" << *p << "\n";

    delete p;

    return 0;
}

Выход:

Before foo: 100
In foo: 50
After foo:  100
Before bar: 100
In bar: 80
After bar:  80
3 голосов
/ 26 апреля 2011

Чтобы понять эти фразы, давайте посмотрим на пару вещей:

typedef double Foo;
void fooFunc(Foo &_bar){ ... }

Таким образом, это передает двойной по ссылке.

typedef double* Foo;
void fooFunc(Foo &_bar){ ... }

теперь он передает указатель на double по ссылке.

typedef double** Foo;
void fooFunc(Foo &_bar){ ... }

Наконец, он передает указатель на указатель на double по ссылке. Если вы будете думать с точки зрения typedefs, как это, вы поймете правильное упорядочение символов & и * plus, что это означает.

...