C++ 如何通过指针获取构造函数内函数返回值的地址

C++ 如何通过指针获取构造函数内函数返回值的地址,c++,winapi,gdi+,C++,Winapi,Gdi+,我试图在构造函数中访问函数的返回值地址。我有一个指向类构造函数的指针,但无论我尝试什么组合,它都会给我一个错误。通过这个例子,我认为我走上了正确的道路: graphics.DrawLine(&p_CustomButton->frontPen, 1, 1, objWidth - 1, objWidth - 1); 我了解指针是如何工作的,读了很多关于它的书,我确信这纯粹是关于语法的。我在上面的例子中遇到了错误编译器错误C2276,所以我看了一下,但当我尝试推荐的方法时,它也不起作用

我试图在构造函数中访问函数的返回值地址。我有一个指向类构造函数的指针,但无论我尝试什么组合,它都会给我一个错误。通过这个例子,我认为我走上了正确的道路:

graphics.DrawLine(&p_CustomButton->frontPen, 1, 1, objWidth - 1, objWidth - 1);
我了解指针是如何工作的,读了很多关于它的书,我确信这纯粹是关于语法的。我在上面的例子中遇到了错误编译器错误C2276,所以我看了一下,但当我尝试推荐的方法时,它也不起作用:

graphics.DrawLine(&CustomButton::frontPen, 1, 1, objWidth - 1, objWidth - 1);
现在的错误是:

错误C2664“Gdiplus::Status Gdiplus::Graphics::DrawLine(const Gdiplus::Pen,const Gdiplus::Point&,const Gdiplus::Point&”):无法将参数1从“Gdiplus::Pen(u_uThisCall CustomButton::)(void)”转换为“const Gdiplus::Pen*”

首先,它不应该是无效的,因为它被声明了(我认为),第二个声明它为const也没有帮助。但我发现我的思路是正确的,因为编译器不再抱怨无效的语法

在我的研究过程中,我看了一眼,(这里他们的建议与MSDN相同),(我尝试了所有可能的带括号的组合,但几乎没有成功)还有更多的页面,我无法再进一步了(我也尝试了使用“*”,但对我来说没有任何意义,因为我不想再做一个指针)

我是C++新手,在这个问题上花了2个多小时。我试图尽可能清楚地描述这个问题;如果我做了一些错误的张贴像这样的问题,请让我知道,我会避免在我的下一篇文章

完整代码:

#include <windows.h>
#include <commctrl.h>
#include <winuser.h>
#include <gdiplus.h>
#include <string.h>
#include "CustomButton.h"

CustomButton::CustomButton()
{
    width = 0;
    height = 0;
    type = 0;
    customButton = NULL;

    Gdiplus::SolidBrush frontBrush(Gdiplus::Color(255, 160, 160, 160)); // Create a Brush object.
    Gdiplus::Pen frontPen(&frontBrush, 1);
}

CustomButton::~CustomButton()
{
    if (customButton)
        DestroyWindow(customButton);
}

/*set-get functions*/
HWND CustomButton::getButton()
{
    return customButton;
}

void CustomButton::DrawRoundRect(HDC hdc, int X, int Y, int RectWidth, int RectHeight, int CornerRadius)
{
    Gdiplus::Graphics graphics(hdc);
    graphics.SetSmoothingMode(Gdiplus::SmoothingModeHighQuality);

    Gdiplus::GraphicsPath gfxPath;

    gfxPath.AddLine(X + CornerRadius, Y, X + RectWidth - (CornerRadius * 2), Y);
    gfxPath.AddArc(X + RectWidth - (CornerRadius * 2), Y, CornerRadius * 2, CornerRadius * 2, 270, 90);
    gfxPath.AddLine(X + RectWidth, Y + CornerRadius, X + RectWidth, Y + RectHeight - (CornerRadius * 2));
    gfxPath.AddArc(X + RectWidth - (CornerRadius * 2), Y + RectHeight - (CornerRadius * 2), CornerRadius * 2, CornerRadius * 2, 0, 90);
    gfxPath.AddLine(X + RectWidth - (CornerRadius * 2), Y + RectHeight, X + CornerRadius, Y + RectHeight);
    gfxPath.AddArc(X, Y + RectHeight - (CornerRadius * 2), CornerRadius * 2, CornerRadius * 2, 90, 90);
    gfxPath.AddLine(X, Y + RectHeight - (CornerRadius * 2), X, Y + CornerRadius);
    gfxPath.AddArc(X, Y, CornerRadius * 2, CornerRadius * 2, 180, 90);
    gfxPath.CloseFigure();

    graphics.DrawPath(&frontPen, &gfxPath);
    graphics.DrawPath(&frontPen, &gfxPath);
}

CustomButton* CustomButton::CreateCustomButton(HINSTANCE hInstance, HWND hwnd, int pos_x, int pos_y, int width, int height, int type)
{
    CustomButton * p_CustomButton = new CustomButton;

    p_CustomButton->customButton = CreateWindowEx(0, L"BUTTON", L"OK", WS_VISIBLE | WS_CHILD | BS_OWNERDRAW, pos_x, pos_y, width, height, hwnd, (HMENU)1, (HINSTANCE)GetWindowLong(hwnd, GWLP_HINSTANCE), p_CustomButton); //BS_OWNERDRAW allows BN_CLICKED to be registered and specifies that DRAWITEM will create mask for the button
    if (p_CustomButton->customButton == NULL)
    {
        delete p_CustomButton;
        MessageBox(NULL, L"Problem creating the Button.", L"Error", 0);
        return 0;
    }

    p_CustomButton->width = width;
    p_CustomButton->height = height;
    p_CustomButton->type = type;

    if (!SetWindowSubclass(p_CustomButton->customButton, CustomButton::CustomButtonProc, 0, (DWORD_PTR)p_CustomButton))
    {
        delete p_CustomButton;
        MessageBox(NULL, L"Problem subclassing the Button.", L"Error", 0);
        return 0;
    }

    return p_CustomButton;
}

LRESULT CALLBACK CustomButton::CustomButtonProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam, UINT_PTR uIdSubclass, DWORD_PTR dwRefData)
{
    CustomButton *p_CustomButton = (CustomButton*)dwRefData;
    switch (uMsg)
    {
        case WM_NCDESTROY:
        {
            RemoveWindowSubclass(hwnd, CustomButton::CustomButtonProc, uIdSubclass);
            p_CustomButton->customButton = NULL;
            break;
        }
        case 0x8000: //WM_DRAWITEM
        {
            LPDRAWITEMSTRUCT lpdis = (DRAWITEMSTRUCT*)lParam; //assigning drawing structure to the control
            /* drawing the classic windows button */
            SetBkColor(lpdis->hDC, RGB(18, 18, 18)); //background color for the text, which is assigned to the whole button via ETO_OPAQUE in ExtTextOut
            ExtTextOut( //assigns graphic properties to the control
                lpdis->hDC,
                ((lpdis->rcItem.right - lpdis->rcItem.left)) / 2, //X position of the text inside, calculated as half the width
                ((lpdis->rcItem.bottom - lpdis->rcItem.top)) / 2, //Y
                ETO_OPAQUE, //ETO_OPAQUE = button will have the color from SetBkColor
                &lpdis->rcItem, //pointer to rectangle inside the drawing structure
                0, //text inside button
                0, //length of the text string
                NULL);
            lpdis->CtlType = ODT_BUTTON;
            lpdis->itemAction = ODA_DRAWENTIRE;

            int lineWidth = 2;
            int objWidth = 18;

            p_CustomButton->DrawRoundRect(lpdis->hDC, 0, 0, 18, 18, 2);

            Gdiplus::Graphics graphics(lpdis->hDC);
            graphics.SetSmoothingMode(Gdiplus::SmoothingModeHighQuality);

            switch (p_CustomButton->type)
            {
                case 1:
                {
                    graphics.DrawLine(&p_CustomButton->frontPen, 1, 1, objWidth - 1, objWidth - 1);
                    graphics.DrawLine(p_CustomButton->frontPen, 1, objWidth - 1, objWidth - 1, 1);
                    graphics.DrawLine(p_CustomButton->frontPen, 1, 1, objWidth - 1, objWidth - 1);
                    graphics.DrawLine(p_CustomButton->frontPen, 1, objWidth - 1, objWidth - 1, 1);
                    break;
                }
                case 2:
                {
                    graphics.DrawLine(p_CustomButton->frontPen, 0, 3, objWidth, 3);
                    break;
                }
                case 3:
                {
                    graphics.DrawLine(p_CustomButton->frontPen, 0, objWidth - 3, objWidth, objWidth - 3);
                    break;
                }
            }
            break;
        }
        case WM_LBUTTONUP:
        {
            switch (p_CustomButton->type)
            {
                case 1:
                {
                    SendMessage(GetParent(hwnd), WM_CLOSE, 0, 0);
                    break;
                }
                case 2:
                {
                    ShowWindow(GetParent(hwnd), SW_MAXIMIZE);
                    break;
                }
                case 3:
                {
                    ShowWindow(GetParent(hwnd), SW_MINIMIZE);
                    break;
                }
            }
            break;
        }
    }
    return DefSubclassProc(hwnd, uMsg, wParam, lParam);
}

您的构造函数具有以下功能:

Gdiplus::SolidBrush frontBrush(Gdiplus::Color(255, 160, 160, 160)); // Create a Brush object.
Gdiplus::Pen frontPen(&frontBrush, 1);
这将在堆栈上创建frontBrush和frontPen,然后立即销毁它们,因为它们的范围仅限于构造函数。它们需要是成员变量(如
宽度
高度
,等等),并且可以在构造函数的序言中构造:

CustomButton::CustomButton()
   : frontBrush(Gdiplus::Color(255, 160, 160, 160))
   , frontPen(&frontBrush, 1)
{
.
.
.
}

我尝试了这个解决方案,但没有成功,我又犯了一些错误。我在页眉中声明的宽度和高度是一样的,我会把它添加到原来的帖子中,也许也有问题?谢谢你帮助我,阿拉斯泰尔!过了一段时间,我通过更改标题修复了这些问题(我在声明中留下了括号,好像它是函数一样,删除了那些有帮助的内容)。我还将graphics.DrawPath(&frontPen,&gfxPath)更改为graphics.DrawPath(&(CustomButton::frontPen),&gfxPath),解决了这些行上的所有错误,但在CustomButtonProc中,错误与原始帖子中的错误相同。错误有:C2664'Gdiplus::Status Gdiplus::graphics::DrawLine(const-Gdiplus::Pen,const-Gdiplus::Point&,const-Gdiplus::Point&):无法将参数1从“const-Gdiplus::Pen-CustomButton::”转换为“const-Gdiplus::Pen*”,请尝试
graphics.DrawLine(&p_-CustomButton->frontPen
在所有调用中
graphics.DrawLine
@DanielMaczak:否决票的可能原因:文本太多(问题中有很多不相关的文本);缺失(发布的代码当然不是最小的);问题实际上是问一些基本问题(如何访问类成员);没有为编译器使用适当的警告级别(在/W4它会告诉您,您没有访问/初始化类成员)。
CustomButton::CustomButton()
   : frontBrush(Gdiplus::Color(255, 160, 160, 160))
   , frontPen(&frontBrush, 1)
{
.
.
.
}