So, my idea is to keep how many AsyncDoSomething() are pending to complete, and only dispose when this count reaches to zero. My initial approach is:
public class MyClass : IDisposable {
    private delegate void AsyncDoSomethingCaller();
    private delegate void AsyncDoDisposeCaller();
    private int pendingTasks = 0;
    public DoSomething() {
     // Do whatever.
    }
    public AsyncDoSomething() {
     pendingTasks++;
     AsyncDoSomethingCaller caller = new AsyncDoSomethingCaller();
     caller.BeginInvoke( new AsyncCallback( EndDoSomethingCallback ), caller);
    }
    public Dispose() {
     AsyncDoDisposeCaller caller = new AsyncDoDisposeCaller();
     caller.BeginInvoke( new AsyncCallback( EndDoDisposeCallback ), caller);
    }
    private DoDispose() {
     WaitForPendingTasks();
     // Finally, dispose whatever managed and unmanaged resources.
    }
    private void WaitForPendingTasks() {
     while ( true ) {
      // Check if there is a pending task.
      if ( pendingTasks == 0 ) {
       return;
      }
      // Allow other threads to execute.
      Thread.Sleep( 0 );
     }
    }
    private void EndDoSomethingCallback( IAsyncResult ar ) {
     AsyncDoSomethingCaller caller = (AsyncDoSomethingCaller) ar.AsyncState;
     caller.EndInvoke( ar );
     pendingTasks--;
    }
    private void EndDoDisposeCallback( IAsyncResult ar ) {
     AsyncDoDisposeCaller caller = (AsyncDoDisposeCaller) ar.AsyncState;
     caller.EndInvoke( ar );
    }
}
Some issues may occur if two or more threads try to read / write the pendingTasks variable concurrently, so the lock keyword should be used to prevent race conditions:
public class MyClass : IDisposable {
    private delegate void AsyncDoSomethingCaller();
    private delegate void AsyncDoDisposeCaller();
    private int pendingTasks = 0;
    private readonly object lockObj = new object();
    public DoSomething() {
     // Do whatever.
    }
    public AsyncDoSomething() {
     lock ( lockObj ) {
      pendingTasks++;
      AsyncDoSomethingCaller caller = new AsyncDoSomethingCaller();
      caller.BeginInvoke( new AsyncCallback( EndDoSomethingCallback ), caller);
     }
    }
    public Dispose() {
     AsyncDoDisposeCaller caller = new AsyncDoDisposeCaller();
     caller.BeginInvoke( new AsyncCallback( EndDoDisposeCallback ), caller);
    }
    private DoDispose() {
     WaitForPendingTasks();
     // Finally, dispose whatever managed and unmanaged resources.
    }
    private void WaitForPendingTasks() {
     while ( true ) {
      // Check if there is a pending task.
      lock ( lockObj ) {
       if ( pendingTasks == 0 ) {
        return;
       }
      }
      // Allow other threads to execute.
      Thread.Sleep( 0 );
     }
    }
    private void EndDoSomethingCallback( IAsyncResult ar ) {
     lock ( lockObj ) {
      AsyncDoSomethingCaller caller = (AsyncDoSomethingCaller) ar.AsyncState;
      caller.EndInvoke( ar );
      pendingTasks--;
     }
    }
    private void EndDoDisposeCallback( IAsyncResult ar ) {
     AsyncDoDisposeCaller caller = (AsyncDoDisposeCaller) ar.AsyncState;
     caller.EndInvoke( ar );
    }
}
I see a problem with this approach. As the release of resources is asynchronously done, something like this might work:
MyClass myClass;
using ( myClass = new MyClass() ) {
    myClass.AsyncDoSomething();
}
myClass.DoSomething();
When the expected behavior should be to launch an ObjectDisposedException when DoSomething() is called outside the using clause. But I don't find this bad enough to rethink this solution.