Combine two dataloaders

Given two datasets of length 8000 and 1480 and their corresponding train and validation loaders,I would like o create a new dataloader that allows me to iterate through those loaders.

I tried using concatenate datasets as shown below

class custom_dataset(Dataset):
  def __init__(self,*data_sets):
  def __getitem__(self,i):
    return tuple(d[i] for d in self.datasets)
  def __len__(self):
    return min(len(d) for d in self.datasets)

print("Training samples {},Val Samples {}".format(len(new_train_loader),len(new_val_loader)))
Training samples 74,Val Samples 18 this is the same number of samples for dataset 2
Training samples 400,Val Samples 100 for dataset 1

Since we are concatenating our dataset shouldn’t the final no of train samples be 474 and not 74?

But then when I print the lenght of the dataset it comes out to be 1480 and not 9400 why is that?
If we concatenate two datasets having 8000 and 1480 samples shouldn’t the final dataset be of length 9400 and not 1480.

Although I am able to visualize both samples from the two datasets,I somehow feel that a majority of the samples in the first dataset have not been included.
Any suggestions on what is wrong with my following code

You are not concatenating the datasets, but are returning tuples from both while using the min(len()) of both datasets.
If you want to concatenate both dataset (i.e. iterate through them in a sequential way) use ConcatDataset.

thanks for the suggestion,I will look into it
I was able to make it work but it ends up returning only one image which could either belong to Dataset 1 or dataset 2.

Is there a way to combine my previous code with the principles of concatDataset so that I can iterate through it in a sequential way and have it return tuples from both datasets.
Meaning final output of concat dataset will be

I don’t understand what the difference to your current implementation would be as you are currently already returning tuples and are thus not concatenating the datasets.
Since both datasets might have a different length, you are using the min of them for the iteration, which sounds also right.
E.g. assuming the first dataset has 2 samples, while the second one has 4, how should the tuple creation look in this case if you are not using the min length?

although the tuple creation is correct,when I initiate training only the first 74/400 samples are taken from the dataset 1.The rest of them are ignored.

This means that during training the train error decreases and the validation error increases.
I would like to use all the training samples from both datasets and then pass them through the model.
the reason why i need to iterate through both datasets is to compute feature similiarity between the two samples from dataset 1 and dataset 2.

I tried using iter as shown in the below implementation


for iter_num in range(epochs):
  if iter_num%len_source==0:
  if iter_num%len_target==0:

This ends up working well,but I am not sure if all the batches are being taken into consideration or if it’s just one batch at each epoch.
As per my understaning,if my epochs is set to 10 then I will end up calling will only consider 10 batches from each dataset

How would you like to draw samples from both datasets if their lengths are unequal?
Currently you are limiting the length to the min length of both datasets as already explained, which is one possibility. If you want to repeat the smaller dataset until all samples from the larger one are used, you could use a modulo operation in the indexing of the smaller dataset.

If you want to repeat the smaller dataset until all samples from the larger one are used, you could use a modulo operation in the indexing of the smaller dataset.

Could you let me know how to implement the modulo operation for indexing the smaller dataset.
Should i use it in the training loop or in the custom dataset block?

You can use it in the __getitem__:

class MyDataset(Dataset):
    def __init__(self):
        self.small_dataset = torch.arange(10).view(10, 1)
        self.large_dataset = torch.arange(20).view(20, 1)
    def __getitem__(self, index):
        a = self.small_dataset[index % len(self.small_dataset)]
        b = self.large_dataset[index]
        return a, b
    def __len__(self):
        return len(self.large_dataset)

dataset = MyDataset()
# 20

loader = DataLoader(dataset, batch_size=2)

for idx, (a, b) in enumerate(loader):
    print("iter {}\na {}\nb {}".format(idx, a, b))