当前位置: 首页 > news >正文

制作企业免费网站优秀网站网页设计图片

制作企业免费网站,优秀网站网页设计图片,建设一个网站可以采用哪几种方案,晋江文学城写作网站背景:在项目开发过程中,难免碰到这种情况,当我们想要通过我们开发的库,调用主程序中的一些变量或者函数的时候,就会导致一些问题,因为在项目构建过程中,库都是不依赖于主程序编译的,…

背景:

在项目开发过程中,难免碰到这种情况,当我们想要通过我们开发的库,调用主程序中的一些变量或者函数的时候,就会导致一些问题,因为在项目构建过程中,库都是不依赖于主程序编译的,但是在开发库函数的时候发现,有时候我们不可避免的会和主程序之间的变量或者函数进行沟通,在这种情况下,如何在构建库的过程中,能够调用到主程序的函数? 同时又要避免链接问题(库构建的过程中应该独立构建,不依赖主程序)?

简介:

在目前的项目中,遇到了一个比较特殊的问题,主程序中包括了各种各样的管理器,其中就有一个线程池管理器,这个管理器可以接收任务,然后在后台运行。管理器是一个单例(关于如何构建模版单例,可以查看我这篇博客 -- 博客 ),我在库当中需要通过

Worker::getInstance()->getRuntime()

的方式获取Worker实例下的一个类型,现在就是问题来了,worker是在主程序处实现的,我们在库当中只是单纯的包含了库的头文件,这样在链接阶段就会提示找不到Worker相关的符号(或者因这个原因导致的其他问题)。所以有什么方式能够在写库的时候,能通过某些方式调用到App里面的一些函数吗?

思考:

首先我们要想一下,要实现这样的工作会遇到哪些问题?

第一个如果直接在库当中调用App中的某些函数,一定会遇到链接时找不到对应符号的实现。这是因为我们可以在App中调用库的函数完全是得益于target_link_library(App PRIVATE myLib),这样在编译期间就会通过链接myLib,从而在其中找到App中想要用到的myLib中的函数实现。但是我们没有办法反过来这么指定呀target_link_library(myLib PUBLIC App),因为App是一个可执行文件,是没办法被库链接的!(但是后来查到好像在某些平台可以把App所有的符号都Export出去,但是这不是通用的解决方案)。

怎么解决呢? 我想是否可以将我们想要用到的某个具体函数,封装起来,然后保存到myLib中的某个全局变量中,这样在myLib中调用的话我们直接从这个全局变量中取出这个函数的封装,然后自己拆解就行了。

设计

那接下来是要考虑如何设计这个结构了。

我们想要这么一个结构,可以将App中的一些函数封装起来,然后存储到这个结构体中,然后用到的时候,我们可以把这个结构体转换成我们想用的东西。所以这个结构体可能会封装一个函数,这个函数可以有参数或者无参数,它也可以返回任意类型的参数,否则的话这个结构体的通用性就不强。

所以我设想的是,在一个“池子”中保存某个结构体,这个结构体可以通过名字取出来,然后这个结构体可以转换成具体的格式。

所以第一步,我们使用std::<map>(std::string ,xxx ); 来做这个“池子”。

其次我们要能够通过一种格式将我们的结构体转换成当时它实际的样子,比如我一开始这个结构体存储了一个std::function<void()> fun = [](){std::cout<<"Hello"<<std::endl}; 那我用这个xxx结构体把这个函数封装完成之后,我要用的时候,可以通过xxx.cast< std::function<void()> >()再把xxx转换成这个类型,然后调用,所以这个结构体一定是要可以抹除类型信息的,我们称它为Any。

实现

好了,具体的Any实现,我直接放出来,具体可以参考《深入理解c++11 代码优化与企业级应用》。具体咱们不赘述了。

#ifndef ANY_H
#define ANY_H#include <memory>
#include <typeindex>
#include <exception>
#include <iostream>struct Any
{Any(void) : m_tpIndex(std::type_index(typeid(void))) {}Any(const Any& that) : m_ptr(that.Clone()), m_tpIndex(that.m_tpIndex) {}Any(Any && that) : m_ptr(std::move(that.m_ptr)), m_tpIndex(that.m_tpIndex) {}//创建智能指针时,对于一般的类型,通过std::decay来移除引用和cv符,从而获取原始类型template<typename U, class = typename std::enable_if<!std::is_same<typename std::decay<U>::type, Any>::value, U>::type>Any(U && value) : m_ptr(new Derived < typename std::decay<U>::type>(std::forward<U>(value))),m_tpIndex(std::type_index(typeid(typename std::decay<U>::type))){}bool IsNull() const { return !bool(m_ptr); }template<class U> bool Is() const{return m_tpIndex == std::type_index(typeid(U));}//将Any转换为实际的类型template<class U>U& AnyCast(){if (!Is<U>()){std::cout << "can not cast " << typeid(U).name() << " to " << m_tpIndex.name() << std::endl;throw std::logic_error{"bad cast"};}auto derived = dynamic_cast<Derived<U>*> (m_ptr.get());return derived->m_value;}Any& operator=(const Any& a){if (m_ptr == a.m_ptr)return *this;m_ptr = a.Clone();m_tpIndex = a.m_tpIndex;return *this;}Any& operator=(Any&& a){if (m_ptr == a.m_ptr)return *this;m_ptr = std::move(a.m_ptr);m_tpIndex = a.m_tpIndex;return *this;}private:struct Base;typedef std::unique_ptr<Base> BasePtr;struct Base{virtual ~Base() {}virtual BasePtr Clone() const = 0;};template<typename T>struct Derived : Base{template<typename U>Derived(U && value) : m_value(std::forward<U>(value)) { }BasePtr Clone() const{return BasePtr(new Derived<T>(m_value));}T m_value;};BasePtr Clone() const{if (m_ptr != nullptr)return m_ptr->Clone();return nullptr;}BasePtr m_ptr = nullptr;std::type_index m_tpIndex;
};#endif // ANY_H

然后接下来的“池子”就很简单了

struct functionWrappers{static auto GetList() -> std::map<std::string,Any>&;static std::map<std::string,Any>::iterator&& getWrapper(std::string);template <typename T>static T getFunc(std::string name){auto funcWarpper = getWrapper(name);if( funcWarpper == Qml::Register::functionWrappers::GetList().end() ){qCritical()<<"Can not find Worker warpper in Qml::Register::functionWrapper::GetList().end()"<<__PRETTY_FUNCTION__;return std::move(funcWarpper->second.AnyCast<T>());}auto func = funcWarpper->second.AnyCast<T>();if(!func){qCritical()<<"Can not convert Worker warpper function "<<__PRETTY_FUNCTION__;return func;}return func;};static Any&& getAnyFunc(const std::string& );
};

使用

使用方式如下:

//App
quick::Qml::Register::funcType addTaskCount_f("addTaskCount",[](){return quick::App::Worker::getInstance()->addTaskCount();});


//myLib
Qml::Register::functionWrappers::getFunc<std::function<void()>>("addTaskCount")();

很简单,不再赘述。

通过结合我的另一篇文章,如何静态化注册某些组件 -- 博客 , 我们可以实现程序启动之后即自动注册。

有问题欢迎讨论。

http://www.yayakq.cn/news/775268/

相关文章:

  • 怎样做免费网站如何外贸网站推广
  • 网站建设虍金手指花总主流的net快速开发框架
  • 徐州网站建设4胶州市 网站建设
  • 网站seo心态做网站一年多少钱
  • 知名做网站公司北京有哪些不错的互联网公司
  • 佛山外贸型网站建设工程八大员考试网站
  • 东莞在哪里学网站建设上海中学地址
  • 网站备案修改域名wordpress主题制作主题选项
  • 网站制作用什么语言淘宝上网站建设好便宜
  • 公司电子产品网站模板网页设计与制作第四版
  • 我做的网站平台百度搜不到网站做微信小程序
  • 淘宝客网站如何做排名用 net做网站
  • 模板网站对排名的影响网站翻页代码
  • 网站不做icp备案智慧门店管理系统app
  • 租用微信做拍卖网站网络营销主要特点有哪些
  • 恢复被百度k网站 关键词收录软件下载中心
  • 阜城网站建设全网优化推广公司
  • 做电力的系统集成公司网站建设区块链网站
  • 网站建设方法冫金手指排名26wordpress toggle
  • 淄博网站优化首选公司网页前端做购物网站的实训报告
  • 哪里可以学网站开发自己做国际网站
  • 标题设计网站苏州高端网站建设kgu
  • 新网站域名备案流程建设工程168类似的网站
  • 西青网站文化建设淄博网站建设制作
  • 延安做网站的公司网站内容及内链建设
  • 个人网站可以做资讯小说类wordpress做旅游网站
  • 南京哪家网站建设好wordpress 访客记录
  • 国外网站A国内域名注册平台
  • 聊城市建设路小学网站中宁网站建设公司
  • 东莞专业网站推广多少钱提高网站的用户体验度