Qt动态重生类厂

class QConstructHelper
{
public:
    virtual void * create() = 0;
};


template<typename T>
class QMetaTypeConstructHelper : public QConstructHelper
{
public:
    QMetaTypeConstructHelper(){}

    void * create(){return new T;}
};


template<typename T>
int qRegisterConstructHelper(const char *name)
{
    return qRegisterMetaType< QMetaTypeConstructHelper<T> >(name);
}


void * qConstructHelper(const char* name);

QObject * qObjectConstructHelper( const char* name );






//关键函数的实现
void * qConstructHelper(const char* name)
{
    int id = QMetaType::type( name );
    if (id != QMetaType::UnknownType)
    {
        QConstructHelper * creator = static_cast<QConstructHelper*>(QMetaType::create( id ));
        if(creator)
        {
            void * p = creator->create();
            QMetaType::destroy(id,creator);
            return p;
        }
    }
    return NULL;
}

QObject * qObjectConstructHelper( const char* name )
{
    return  static_cast<QObject*>(qConstructHelper(name));
}



用法示例:

    qRegisterConstructHelper<Data>("myData");
    qRegisterConstructHelper<Widget>("widget");

    Widget * w = qobject_cast<Widget*>(qObjectConstructHelper("widget"));
    if(w)
    {
        w->show();
        return a.exec();
    }




另外这个函数可以创建Qt的原生对象(除窗体对象已外的)

void * qMetaTypeCreate(const char *name)
{
    int id = QMetaType::type( name );
    if (id != QMetaType::UnknownType)
    {
        return QMetaType::create( id );
    }
    return NULL;
}
在使用 Qt 5.14 开发 Android 应用时,获取相册中图片的存储路径是一个相对常见的需求,尤其是在需要访问用户选择的照片进行处理或上传等操作时。由于 Android 系统的安全机制限制,直接获取文件路径的方式可能并不适用,特别是在 Android 10(API level 29)及以上版本中引入了作用域存储(Scoped Storage)。 ### 获取相册图片存储路径的方法 #### 方法一:通过 `QFileDialog` 获取 URI 并解析路径 在 Qt 中,可以使用 `QFileDialog::getOpenFileName()` 或 `QFileDialog::getOpenFileNames()` 杜撰一个文件选择对话框来让用户从相册中选择图片[^1]。虽然返回的是文件路径,但在 Android 上可能返回的是内容 URI(如 `content://media/external/images/media/...`),而不是实际的文件路径。 为了将内容 URI 转换为实际的文件路径,可以通过 `QAndroidJniObject` 调用 Java 层的方法实现路径转换: ```cpp #include <QAndroidJniObject> #include <QtAndroid> QString getPathFromUri(const QUrl &uri) { QAndroidJniObject context = QtAndroid::androidContext(); QAndroidJniObject contentResolver = context.callObjectMethod("getContentResolver", "()Landroid/content/ContentResolver;"); QAndroidJniObject uriString = QAndroidJniObject::fromString(uri.toString()); QAndroidJniObject uriObj = QAndroidJniObject::callStaticObjectMethod("android/net/Uri", "parse", "(Ljava/lang/String;)Landroid/net/Uri;", uriString.object<jstring>()); QAndroidJniObject cursor = contentResolver.callObjectMethod("query", "(Landroid/net/Uri;[Ljava/lang/String;Ljava/lang/String;[Ljava/lang/String;Ljava/lang/String;)Landroid/database/Cursor;", uriObj.object<jobject>(), nullptr, nullptr, nullptr, nullptr); QString path; if (cursor.isValid()) { int columnIndex = cursor->callIntMethod("getColumnIndex", "(Ljava/lang/String;)I", QAndroidJniObject::fromString("_data").object<jstring>()); if (columnIndex >= 0) { cursor->callMethod<void>("moveToFirst", "()Z"); QAndroidJniObject data = cursor->getObject<jstring>(); path = data.toString(); } cursor->callMethod<void>("close", "()V"); } return path; } ``` 调用该函数示例: ```cpp QUrl selectedFile = QFileDialog::getOpenFileUrl(nullptr, "选择图片", QUrl(), "Images (*.png *.jpg)"); if (!selectedFile.isEmpty()) { QString filePath = getPathFromUri(selectedFile); qDebug() << "文件路径:" << filePath; } ``` #### 方法二:使用 `QtAndroid` API 直接访问媒体库 另一种方法是使用 `QtAndroid` 提供的 JNI 接口,直接访问 Android 的媒体库以获取图片路径。这种方法适用于需要更精细控制的情况,例如列出所有图片并展示给用户选择。 #### 注意事项 - **权限请求**:在 Android 6.0(API level 23)及以上版本中,必须在运行时请求读取外部存储的权限(`READ_EXTERNAL_STORAGE`)。可以通过 `QtAndroid::requestPermissionsSync()` 请求权限[^2]。 - **作用域存储**:从 Android 10 开始,Google 引入了作用域存储机制,应用默认只能访问自己的私有目录。如果确实需要访问公共目录中的文件,可以在 `AndroidManifest.xml` 中添加 `requestLegacyStorage` 标志以启用旧的存储模式[^3]: ```xml <application android:requestLegacyStorage="true" ...> </application> ``` - **兼容性问题**:不同 Android 版本对文件系统的访问方式有所不同,建议在多种设备上测试代码以确保兼容性。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wolfseek

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值