[Android进阶]Binder学习(初始篇)
Android中Binder学习(初始篇)
本篇博客学习自侯亮的博客。地址为:
1 什么是Binder?
简单地说。Binder是Android平台上的一种跨进程交互技术。
该技术最早并非由Google公司提出的,它的前身是Be Inc公司开发的OpenBinder,并且在Palm中也有应用。后来OpenBinder的作者Dianne Hackborn增加了Google公司,并负责Android平台的开发工作,所以把这项技术也带进了Android。
我们知道。在Android的应用层次上。基本上已经没有过去的进程概念了。然而在实现层次,它毕竟还是要建构在一个个进程之上的。
实际上,在Android内部,那些支撑应用的组件往往会身处于不同的进程,那么应用的底层必定会牵涉大量的跨进程通信。为了保证通信的高效性。Android提供了Binder机制。
Binder机制具有两层含义:
- 是一种跨进程通信手段(IPC。 Inter-Process Communication)。
- 是一种远程过程调用手段(RPC。Remote Procedure Call)。
从实现的角度来说,Binder核心被实现成一个Linux驱动程序。并执行于内核态。这样它才干具有强大的跨进程訪问能力。
1.1 简述Binder的跨进程机制
为了理解Binder,我们能够先画一张最简单的跨进程通信示意图:

这个非常easy理解,不需赘言。到了Android平台上,IPC机制就变成了Binder机制,情况相似,仅仅只是为了便于说明问题。我们须要略微调整一下示意图:

图中A側的圆形块,表示“Binder代理方”。主要用于向远方发送语义,而B側的方形块则表示“Binder响应方”,主要用于响应语义。须要说明的是,这样的图形表示方法是我自己杜撰的,并没有正规的出处。我个人认为这样的图形非常简便,所以在分析Android架构时,会常常使用这样的表示法。
在后文中。我们能够看到。Binder代理方大概相应于C++层次的BpBinder对象。而Binder响应方则相应于BBinder对象。这两个对象在后文会具体阐述,此处不必太细究。
然而,上图的Binder代理方主要仅仅负责了“传递信息”的工作,并没有起到“远程过程调用”的作用,假设要支持远程过程调用。我们还必须提供“接口代理方”和“接口实现体”。
这样,我们的示意图就须要再调整一下,例如以下:

从图中能够看到。A进程并不直接和BpBinder(Binder代理)打交道。而是通过调用BpInterface(接口代理)的成员函数来完毕远程调用的。此时,BpBinder已经被聚合进BpInterface了,它在BpInterface内部完毕了一切跨进程的机制。还有一方面。与BpInterface相对的响应端实体就是BnInterface(接口实现)了。须要注意的是,BnInterface是继承于BBinder的,它并没有採用聚合的方式来包括一个BBinder对象,所以上图中B側的BnInterface块和BBinder块的背景图案是同样的。
这样看来,对于远程调用的client而言。主要搞的就是两个东西,一个是“Binder代理”。一个是“接口代理”。而服务端主要搞的则是“接口实现体”。
由于binder是一种跨进程通信机制,所以还须要一个专门的管理器来为通信两端牵线搭桥,这个管理器就是Service Manager Service。只是眼下我们能够先放下Service Manager Service。以后再具体研究。
2 Binder相关接口和类
Android的整个跨进程通信机制都是基于Binder的,这样的机制不但会在底层使用,也会在上层使用,所以必须提供Java和C++两个层次的支持。
2.1 Java层次的binder元素
Java层次里并没有我们前文图中所表示的BpBinder、BpInterface、BBinder等较低层次的概念,取而代之的是IBinder接口、IInterface等接口。Android要求全部的Binder实体都必须实现IBinder接口,该接口的定义截选例如以下:
【frameworks/base/core/java/android/os/IBinder.java】
public interface IBinder
{
    . . . . . .
    public String getInterfaceDescriptor() throws RemoteException;
    public boolean pingBinder();
    public boolean isBinderAlive();
    public IInterface queryLocalInterface(String descriptor);
    public void dump(FileDescriptor fd, String[] args) throws RemoteException;
    public void dumpAsync(FileDescriptor fd, String[] args) throws RemoteException;
    public boolean transact(int code, Parcel data, Parcel reply, int flags)
throws RemoteException;
    public interface DeathRecipient
    {
        public void binderDied();
    }
    public void linkToDeath(DeathRecipient recipient, int flags)throws RemoteException;
    public boolean unlinkToDeath(DeathRecipient recipient, int flags);
}
另外,无论是代理方还是实体方,都必须实现IInterface接口:
public interface IInterface
{
    public IBinder asBinder();
}Java层次中,与Binder相关的接口或类的继承关系例如以下:

在实际使用中。我们并不须要编写上图的XXXXNative、XXXXProxy,它们会由ADT依据我们编写的aidl脚本自己主动生成。
用户仅仅需继承XXXXNative编写一个具体的XXXXService就可以。这个XXXXService就是远程通信的服务实体类,而XXXXProxy则是其相应的代理类。
关于Java层次的binder组件,我们就先说这么多。主要是先介绍一个大概。就研究跨进程通信而言,事实上质内容基本上都在C++层次,Java层次仅仅是一个壳而已。以后我会写专文来打通Java层次和C++层次,看看它们是怎样通过JNI技术关联起来的。如今我们还是把注意力集中在C++层次吧。
2.2 C++层次的binder元素
在C++层次。就能看到我们前文所说的BpBinder类和BBinder类了。这两个类都继承于IBinder。IBinder的定义截选例如以下:
【frameworks/native/include/binder/IBinder.h】
class IBinder : public virtual RefBase
{
public:
    . . . . . .
    IBinder();
    virtual sp<IInterface>  queryLocalInterface(const String16& descriptor);
    virtual const String16& getInterfaceDescriptor() const = 0;
    virtual bool            isBinderAlive() const = 0;
    virtual status_t        pingBinder() = 0;
    virtual status_t        dump(int fd, const Vector<String16>& args) = 0;
    virtual status_t        transact(uint32_t code, const Parcel& data,
                                     Parcel* reply, uint32_t flags = 0) = 0;
    class DeathRecipient : public virtual RefBase
    {
    public:
        virtual void binderDied(const wp<IBinder>& who) = 0;
    };
    virtual status_t        linkToDeath(const sp<DeathRecipient>& recipient,
                                        void* cookie = NULL, uint32_t flags = 0) = 0;
    virtual status_t        unlinkToDeath(const wp<DeathRecipient>& recipient,
                                          void* cookie = NULL, uint32_t flags = 0,
                                          wp<DeathRecipient>* outRecipient = NULL) = 0;
    virtual bool            checkSubclass(const void* subclassID) const;
    typedef void (*object_cleanup_func)(const void* id, void* obj, void* cleanupCookie);
    virtual void            attachObject(const void* objectID, void* object,
                                         void* cleanupCookie, object_cleanup_func func) = 0;
    virtual void*           findObject(const void* objectID) const = 0;
    virtual void            detachObject(const void* objectID) = 0;
    virtual BBinder*        localBinder();
    virtual BpBinder*       remoteBinder();
protected:
    virtual          ~IBinder();
private:
};C++层次的继承关系图例如以下:

当中有下面几个非常关键的类:
- BpBinder
- BpInterface
- BBinder
- BnInterface
它们扮演着非常重要的角色。
2.2.1 BpBinder
BpBinder的定义截选例如以下:
class BpBinder : public IBinder
{
public:
BpBinder(int32_t handle);
    inline  int32_t     handle() const { return mHandle; }
    virtual const String16&    getInterfaceDescriptor() const;
    virtual bool        isBinderAlive() const;
    virtual status_t    pingBinder();
    virtual status_t    dump(int fd, const Vector<String16>& args);
    virtual status_t    transact(uint32_t code, const Parcel& data,
                                    Parcel* reply, uint32_t flags = 0);
    virtual status_t    linkToDeath(const sp<DeathRecipient>& recipient,
                                    void* cookie = NULL, uint32_t flags = 0);
    virtual status_t    unlinkToDeath(const wp<DeathRecipient>& recipient,
                                            void* cookie = NULL, uint32_t flags = 0,
                                            wp<DeathRecipient>* outRecipient = NULL);
. . . . . .
. . . . . .作为代理端的核心。BpBinder最重要的职责就是实现跨进程传输的传输机制,至于具体传输的是什么语义,它并不关心。
我们观察它的transact()函数的參数,能够看到全部的语义都被打包成Parcel了。
其它的成员函数,我们先不深究,待我们储备了足够的基础知识后。再回过头研究它们不迟。
2.2.2 BpInterface
还有一个重要的类是BpInterface,它的定义例如以下:
template<typename INTERFACE>
class BpInterface : public INTERFACE, public BpRefBase
{
public:
BpInterface(const sp<IBinder>& remote);
protected:
    virtual IBinder*     onAsBinder();
};其基类BpRefBase的定义例如以下:
class BpRefBase : public virtual RefBase
{
protected:
                            BpRefBase(const sp<IBinder>& o);
    virtual                 ~BpRefBase();
    virtual void            onFirstRef();
    virtual void            onLastStrongRef(const void* id);
    virtual bool            onIncStrongAttempted(uint32_t flags, const void* id);
    inline  IBinder*        remote()                { return mRemote; }
    inline  IBinder*        remote() const          { return mRemote; }
private:
BpRefBase(const BpRefBase& o);
    BpRefBase&              operator=(const BpRefBase& o);
    IBinder* const           mRemote;
    RefBase::weakref_type*  mRefs;
    volatile int32_t         mState;
};BpInterface使用了模板技术。并且由于它继承了BpRefBase。所以先天上就聚合了一个mRemote成员。这个成员记录的就是前面所说的BpBinder对象啦。
以后,我们还须要继承BpInterface<>实现我们自己的代理类。
在实际的代码中,我们全然能够创建多个聚合同一BpBinder对象的代理对象,这些代理对象就本质而言。相应着同一个远端binder实体。在Android框架中,常常把指向同一binder实体的多个代理称为token,这样即便这些代理分别处于不同的进程中,它们也具有了某种内在联系。
这个知识点须要大家关注。
2.2.3 BBinder
Binder远程通信的目标端实体必须继承于BBinder类,该类和BpBinder相对。主要关心的仅仅是传输方面的东西,不太关心所传输的语义。
class BBinder : public IBinder
{
public:
BBinder();
    virtual const String16& getInterfaceDescriptor() const;
    virtual bool        isBinderAlive() const;
    virtual status_t    pingBinder();
    virtual status_t    dump(int fd, const Vector<String16>& args);
    virtual status_t    transact(uint32_t code, const Parcel& data,
                                      Parcel* reply, uint32_t flags = 0);
    virtual status_t    linkToDeath(const sp<DeathRecipient>& recipient,
                                    void* cookie = NULL, uint32_t flags = 0);
    virtual status_t    unlinkToDeath(const wp<DeathRecipient>& recipient,
                                            void* cookie = NULL, uint32_t flags = 0,
                                            wp<DeathRecipient>* outRecipient = NULL);
    virtual void        attachObject(const void* objectID, void* object,
                                          void* cleanupCookie, object_cleanup_func func);
    virtual void*       findObject(const void* objectID) const;
    virtual void        detachObject(const void* objectID);
    virtual BBinder*    localBinder();
protected:
    virtual             ~BBinder();
    virtual status_t    onTransact(uint32_t code, const Parcel& data,
                                        Parcel* reply, uint32_t flags = 0);
private:
    BBinder(const BBinder& o);
    BBinder&    operator=(const BBinder& o);
    class Extras;
    Extras*     mExtras;
    void*       mReserved0;
};我们眼下仅仅需关心上面的transact()成员函数,其它函数留待以后再分析。transact函数的代码例如以下:
【frameworks/native/libs/binder/Binder.cpp】
status_t BBinder::transact(uint32_t code, const Parcel& data,
                           Parcel* reply, uint32_t flags)
{
    data.setDataPosition(0);
    status_t err = NO_ERROR;
    switch (code)
    {
        case PING_TRANSACTION:
            reply->writeInt32(pingBinder());
            break;
        default:
            err = onTransact(code, data, reply, flags);
            break;
    }
    if (reply != NULL)
    {
        reply->setDataPosition(0);
    }
    return err;
}看到了吗,transact()内部会调用onTransact(),从而走到用户所定义的子类的onTransact()里。
这个onTransact()的一大作用就是解析经由Binder机制传过来的语义了。
2.2.4 BnInterface
远程通信目标端的还有一个重要类是BnInterface<>,它是与BpInterface<>相相应的模板类。比較关心传输的语义。
普通情况下,服务端并不直接使用BnInterface<>。而是使用它的某个子类。
为此,我们须要编写一个新的BnXXX子类。并重载它的onTransact()成员函数。
BnInterface<>的定义例如以下:
template<typename INTERFACE>
class BnInterface : public INTERFACE, public BBinder
{
public:
    virtual sp<IInterface>      queryLocalInterface(const String16& _descriptor);
    virtual const String16&     getInterfaceDescriptor() const;
protected:
    virtual IBinder*            onAsBinder();
};如上所看到的,BnInterface<>继承于BBinder。但它并没有实现一个默认的onTransact()成员函数,所以在远程通信时,前文所说的BBinder::transact()调用的onTransact()应该就是BnInterface<>的某个子类的onTransact()成员函数。
2.3 几个重要的C++宏或模板
为了便于编写新的接口和类,Android在C++层次提供了几个重要的宏和模板,比方我们在IInterface.h文件里。能够看到DECLARE_META_INTERFACE、IMPLEMENT_META_INTERFACE的定义。
2.3.1 DECLARE_META_INTERFACE()
DECLARE_META_INTERFACE()的定义例如以下:
#define DECLARE_META_INTERFACE(INTERFACE)                               \
    static const android::String16 descriptor;                          \
    static android::sp<I##INTERFACE> asInterface(                       \
            const android::sp<android::IBinder>& obj);                  \
    virtual const android::String16& getInterfaceDescriptor() const;    \
    I##INTERFACE();                                                     \
    virtual ~I##INTERFACE();                                            \我们举个实际的样例。来说明怎样使用这个宏:

上例中ICamera内部使用了DECLARE_META_INTERFACE(Camera),我们把宏展开后。能够看到ICamera类的定义相当于:
class ICamera: public IInterface
{
public:
static const android::String16 descriptor;
static android::sp<ICamera> asInterface( const android::sp<android::IBinder>& obj);
virtual const android::String16& getInterfaceDescriptor() const;
ICamera();
virtual ~ICamera();
virtual void disconnect() = 0;
. . . . . .宏展开的部分就是中间那5行代码,当中最关键的就是asInterface()函数了,这个函数将承担把BpBinder打包成BpInterface的职责。
2.3.2 IMPLEMENT_META_INTERFACE()
与DECLARE_META_INTERFACE相对的就是IMPLEMENT_META_INTERFACE宏。它的定义例如以下:
#define IMPLEMENT_META_INTERFACE(INTERFACE, NAME)                       \
    const android::String16 I##INTERFACE::descriptor(NAME);             \
    const android::String16&                                            \
            I##INTERFACE::getInterfaceDescriptor() const {              \
        return I##INTERFACE::descriptor;                                \
    }                                                                   \
    android::sp<I##INTERFACE> I##INTERFACE::asInterface(                \
            const android::sp<android::IBinder>& obj)                   \
    {                                                                   \
        android::sp<I##INTERFACE> intr;                                 \
        if (obj != NULL) {                                              \
            intr = static_cast<I##INTERFACE*>(                          \
                obj->queryLocalInterface(                               \
                        I##INTERFACE::descriptor).get());               \
            if (intr == NULL) {                                         \
                intr = new Bp##INTERFACE(obj);                          \
            }                                                           \
        }                                                               \
        return intr;                                                    \
    }                                                                   \
    I##INTERFACE::I##INTERFACE() { }                                    \
    I##INTERFACE::~I##INTERFACE() { }                                   \当中。实现了关键的asInterface()函数。
实际使用IMPLEMENT_META_INTERFACE时,我们仅仅需把它简单地写在binder实体所处的cpp文件里就可以,举比例如以下:

当中的IMPLEMENT_META_INTERFACE(Camera, “android.hardware.ICamera”);一句相当于下面这段代码:
const android::String16 ICamera::descriptor(“android.hardware.ICamera”);
const android::String16& ICamera::getInterfaceDescriptor() const
{
return ICamera::descriptor;
}
android::sp<ICamera> ICamera::asInterface(const android::sp<android::IBinder>& obj)
{
android::sp<ICamera > intr;
if (obj != NULL)
{
intr = static_cast<ICamera*>(obj->queryLocalInterface(
ICamera::descriptor).get());
        if (intr == NULL)
{
            intr = new BpCamera(obj);
        }
}
    return intr;
}
ICamera::ICamera() { }
ICamera::~ICamera () { } 看来,当中重点实现了asInterface()成员函数。请注意。asInterface()函数中会先尝试调用queryLocalInterface()来获取intr。
此时,假设asInterface()的obj參数是个代理对象(BpBinder),那么intr = static_cast
sp<IInterface>  IBinder::queryLocalInterface(const String16& descriptor)
{
    return NULL;
}还有一方面。假设obj參数是个实现体对象(BnInterface对象)的话。那么queryLocalInterface()函数的默认返回值就是实体对象的this指针了,代码例如以下:
【frameworks/native/include/binder/IInterface.h】
template<typename INTERFACE>
inline sp<IInterface> BnInterface<INTERFACE>::queryLocalInterface(const String16& _descriptor)
{
    if (_descriptor == INTERFACE::descriptor)
        return this;
    return NULL;
}在我们所举的Camera样例中,我们要研究的是怎样将BpBinder转成BpInterface。所以如今我们仅仅阐述obj參数为BpBinder的情况。此时asInterface()函数中obj->queryLocalInterface()的返回值为NULL,于是asInterface()会走到new BpCamera(obj)一句,这一句是最关键的一句。我们知道,BpCamera继承于BpInterface,所以此时所创建的BpCamera对象正是可被App使用的BpInterface代理对象。
BpCamera的定义例如以下:
class BpCamera: public BpInterface<ICamera>
{
public:
    BpCamera(const sp<IBinder>& impl)
        : BpInterface<ICamera>(impl)
    {
    }
    // disconnect from camera service
    void disconnect()
    {
        LOGV("disconnect");
        Parcel data, reply;
        data.writeInterfaceToken(ICamera::getInterfaceDescriptor());
        remote()->transact(DISCONNECT, data, &reply);
    }
. . . . . .至此,IMPLEMENT_META_INTERFACE宏和asInterface()函数的关系就分析完毕了。
2.3.3 interface_cast
只是,我们常常使用的事实上并非asInterface()函数,而是interface_cast(),它简单包装了asInterface():
template<typename INTERFACE>
inline sp<INTERFACE> interface_cast(const sp<IBinder>& obj)
{
    return INTERFACE::asInterface(obj);
}以上就是关于C++层次中一些binder元素的介绍。下面我们再进一步分析其它细节。
3 ProcessState
前文我们已经提到过,在Android的上层架构中,已经大幅度地弱化了进程的概念。应用程序猿能看到的主要是activity、service、content provider等概念,再也找不到曾经熟悉的main()函数了。然而,底层程序(C++层次)毕竟还是得跑在一个个进程之上。如今我们就来看底层进程是怎样运用Binder机制来完毕跨进程通信的。
在每一个进程中。会有一个全局的ProcessState对象。这个非常easy理解,ProcessState的字面意思不就是“进程状态”吗。当然应该是每一个进程一个ProcessState。ProcessState的定义位于frameworks/native/include/binder/ProcessState.h中,我们仅仅截选当中的一部分:
class ProcessState : public virtual RefBase
{
public:
    static  sp<ProcessState>    self();
    . . . . . .
    void                startThreadPool();
    . . . . . .
    void                spawnPooledThread(bool isMain);
    status_t            setThreadPoolMaxThreadCount(size_t maxThreads);
private:
    friend class IPCThreadState;
    . . . . . .
    struct handle_entry
    {
        IBinder* binder;
        RefBase::weakref_type* refs;
    };
    handle_entry*       lookupHandleLocked(int32_t handle);
    int                 mDriverFD;
    void*               mVMStart;
    mutable Mutex       mLock;  // protects everything below.
    Vector<handle_entry> mHandleToObject;
    . . . . . .
    KeyedVector<String16, sp<IBinder> > mContexts;
    . . . . . .
};我们知道,Binder内核被设计成一个驱动程序,所以ProcessState里专门搞了个mDriverFD域。来记录binder驱动相应的句柄值,以便随时和binder驱动通信。ProcessState对象採用了典型的单例模式。在一个应用进程中。仅仅会有唯一的一个ProcessState对象,它将被进程中的多个线程共用,因此每一个进程里的线程事实上是共用所打开的那个驱动句柄(mDriverFD)的。示意图例如以下:

每一个进程基本上都是这样的结构,组合起来的示意图就是:
我们常见的使用ProcessState的代码例如以下:
int main(int argc, char** argv)
{
    sp<ProcessState> proc(ProcessState::self());
    . . . . . .
    . . . . . .
    ProcessState::self()->startThreadPool();
    IPCThreadState::self()->joinThreadPool();
}由于ProcessState採用的是单例模式,所以它的构造函数是private的,我们仅仅能通过调用ProcessState::self()来获取进程中唯一的一个ProcessState对象。self()函数的代码例如以下:
sp<ProcessState> ProcessState::self()
{
    Mutex::Autolock _l(gProcessMutex);
    if (gProcess != NULL) {
        return gProcess;
    }
    gProcess = new ProcessState;
    return gProcess;
}ProcessState对象构造之时,就会打开binder驱动:
ProcessState::ProcessState()
    : mDriverFD(open_driver())     // 打开binder驱动。
, mVMStart(MAP_FAILED)
    , mManagesContexts(false)
    , mBinderContextCheckFunc(NULL)
    , mBinderContextUserData(NULL)
    , mThreadPoolStarted(false)
    , mThreadPoolSeq(1)
{
    . . . . . .
    mVMStart = mmap(0, BINDER_VM_SIZE, PROT_READ, MAP_PRIVATE | MAP_NORESERVE, mDriverFD, 0);
    . . . . . .
}注意上面那句mDriverFD(open_driver()),当中的open_driver()就负责打开“/dev/binder”驱动:
static int open_driver()
{
    int fd = open("/dev/binder", O_RDWR);
    . . . . . .
    status_t result = ioctl(fd, BINDER_VERSION, &vers);
    . . . . . .
    size_t maxThreads = 15;
    result = ioctl(fd, BINDER_SET_MAX_THREADS, &maxThreads);
    . . . . . .
    return fd;
}ProcessState中还有一个比較有意思的域是mHandleToObject:
Vector<handle_entry> mHandleToObject;它是本进程中记录全部BpBinder的向量表噢,非常重要。我们前文已经说过,BpBinder是代理端的核心。如今最终看到它的藏身之处了。
在Binder架构中,应用进程是通过“binder句柄”来找到相应的BpBinder的。从这张向量表中我们能够看到,那个句柄值事实上相应着这个向量表的下标。这张表的子项类型为handle_entry。定义例如以下:
struct handle_entry
{
    IBinder* binder;
    RefBase::weakref_type* refs;
};当中的binder域。记录的就是BpBinder对象。

此篇完
[Android进阶]Binder学习(初始篇)的更多相关文章
- 《Android进阶》之第七篇 NDK的使用
		<Android进阶>之第一篇 在Java中调用C库函数 这一篇列举的方法是在NDK没有出来时候用的方式 在Android发布NDK之后,可以这样使用 一.首先下载android-ndk ... 
- 红茶一杯话Binder  (初始篇)
		1 什么是Binder? 简单地说,Binder是Android平 台上的一种跨进程交互技术.该技术最早并不是由Google公司提出的,它的前身是Be Inc公司开发的OpenBinder,而且在Pa ... 
- 《Android进阶》之第三篇 深入理解android的消息处理机制
		Android 异步消息处理机制 让你深入理解 Looper.Handler.Message三者关系 android的消息处理机制(图+源码分析)——Looper,Handler,Message an ... 
- 《Android进阶》之第六篇 Fragment 的使用2
		最近通过学习,对fragment的使用有了新的认识. 一开始接触android的时候,很是受不了这个fragment,总感觉它把一个简单的事情搞复杂啦,所以每次新建工程的时候总是固执的选择empty ... 
- 《Android进阶》之第五篇 Fragment 的使用
		http://blog.csdn.net/lmj623565791/article/details/37970961 1.Fragment的产生与介绍 Android运行在各种各样的设备中,有小屏幕的 ... 
- Android进阶笔记16:ListView篇之ListView刷新显示(全局 和 局部)
		一.ListView内容变化后,动态刷新的步骤(全局刷新): (1)更新适配器Adapter数据源:(不要使用匿名内部类) (2)调用适配器Adapter的刷新方法notifyDataSetChang ... 
- Android进阶笔记14:ListView篇之ListView性能优化
		1. 首先思考一个问题ListView如何才能提高效率 ? 当convertView为空时候,用setTag()方法为每个View绑定一个存放控件的ViewHolder对象.当convertView不 ... 
- Android进阶笔记11:ListView篇之ListView显示多种类型的条目(item)
		ListView可以显示多种类型的条目布局,这里写显示两种布局的情况,其他类似. 1. 这是MainActivity,MainActivity的布局就是一个ListView,太简单了这里就不写了,直接 ... 
- 《Android进阶》之第四篇 ViewPagerIndicator的使用
		1.先将这个开源框架下载到本地: Administrator@QH-20141231RFQJ /d/hixin $ cd ViewPagerIndicator/ Administrator@QH-20 ... 
随机推荐
- 自定义loading dialog --- 后背景透明
			自定义loading dialog --- 后背景透明 <style name="loading_dialog" parent="android:style/ ... 
- Mysql五种时间格式
			YEAR [字节数]:1 [取值范围]:1901~2155 [赋值]: 4位数字 2位字符串:'00'~'69'相当于2000~2069:'70'~'99'相当于1970~1999 2位数字:与2位数 ... 
- EF for Oracle  闪退
			EF oracle 加入实体类型时候闪退 主要原因: Oracle.ManagedDataAcces 版本和 SetupODTforVS2015 版本不一致所致. 更新后 SetupODTforVS2 ... 
- NServiceBus入门:发送一个命令(Introduction to NServiceBus: Sending a command)
			原文地址:https://docs.particular.net/tutorials/intro-to-nservicebus/2-sending-a-command/ 侵删. 能够发送和接收mess ... 
- 监控RTSP 流
			rtsp://admin:admin12345@192.168.0.100/live1.sdp 
- [NHibernate]使用AttributeNHibernate.Mapping.Attributes
			系列文章 [Nhibernate]体系结构 [NHibernate]ISessionFactory配置 [NHibernate]持久化类(Persistent Classes) [NHibernate ... 
- Office(Excel、Word)二次开发——VSTO
			Office(Excel.Word)二次开发——VSTO Office(Excel.Word)二次开发——VSTO Office二次开发模式: 1) VBA(visual studio for app ... 
- 【JSP EL】<c:if>  <c:foreach >EL表达式 获取list长度/不用循环,EL在List中直接获取第一项的内容/EL获取Map的键,Map的值
			1.EL表达式 获取list长度 <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" ... 
- RapidXml使用方法
			一.写xml 文件 [cpp] view plaincopy #include <iostream> #include "rapidxml/rapidxml.hpp" ... 
- c# 冒号:C#中两个冒号(::)的作用
			global::System.Console.WriteLine(number); 冒号在什么地方用. 点是空间下类,表示下一层的意思? 这里面::前面是GAC的标示符global,用法比较特殊,和. ... 
